S Price: $0.689925 (-1.21%)

Contract

0xd61dA7BF1a4Ed6751aea942Cf34c663c0e76BA68

Overview

S Balance

Sonic LogoSonic LogoSonic Logo0 S

S Value

$0.00

Multichain Info

No addresses found
Transaction Hash
Method
Block
From
To
Create Swap Requ...36149422025-01-12 17:16:1713 hrs ago1736702177IN
0xd61dA7BF...c0e76BA68
30 S0.000783555.5
Create Swap Requ...35701012025-01-12 12:02:2318 hrs ago1736683343IN
0xd61dA7BF...c0e76BA68
5 S0.000782965.5
Create Swap Requ...35526422025-01-12 9:45:4020 hrs ago1736675140IN
0xd61dA7BF...c0e76BA68
0.5 S0.000802535.5
Create Swap Requ...33781882025-01-11 11:15:4143 hrs ago1736594141IN
0xd61dA7BF...c0e76BA68
2 S0.000782965.5
Create Swap Requ...33610462025-01-11 8:04:5746 hrs ago1736582697IN
0xd61dA7BF...c0e76BA68
2 S0.000782965.5
Create Swap Requ...32571672025-01-10 13:38:492 days ago1736516329IN
0xd61dA7BF...c0e76BA68
2 S0.000782965.5
Create Swap Requ...32247822025-01-10 8:59:032 days ago1736499543IN
0xd61dA7BF...c0e76BA68
30 S0.000783095.5
Create Swap Requ...31699742025-01-09 21:34:423 days ago1736458482IN
0xd61dA7BF...c0e76BA68
1 S0.000782965.5
Create Swap Requ...31201382025-01-09 13:57:283 days ago1736431048IN
0xd61dA7BF...c0e76BA68
1 S0.000782965.5
Create Swap Requ...30985692025-01-09 9:45:083 days ago1736415908IN
0xd61dA7BF...c0e76BA68
200 S0.000823745.51
Withdraw Funds30814852025-01-09 6:42:083 days ago1736404928IN
0xd61dA7BF...c0e76BA68
0 S0.000454945.5
Create Swap Requ...30785692025-01-09 6:07:364 days ago1736402856IN
0xd61dA7BF...c0e76BA68
0.023 S0.000925246.5
Create Swap Requ...30785342025-01-09 6:07:104 days ago1736402830IN
0xd61dA7BF...c0e76BA68
0.016 S0.000925166.5
Create Swap Requ...30142562025-01-08 17:42:074 days ago1736358127IN
0xd61dA7BF...c0e76BA68
0.5 S0.0022421115
Create Swap Requ...29341782025-01-08 2:43:225 days ago1736304202IN
0xd61dA7BF...c0e76BA68
2.39 S0.00082365.51
Create Swap Requ...28471952025-01-07 10:53:345 days ago1736247214IN
0xd61dA7BF...c0e76BA68
31 S0.000165941.11
Create Swap Requ...27511202025-01-06 19:37:456 days ago1736192265IN
0xd61dA7BF...c0e76BA68
50 S0.000180891.21

Latest 18 internal transactions

Parent Transaction Hash Block From To
36149422025-01-12 17:16:1713 hrs ago1736702177
0xd61dA7BF...c0e76BA68
30 S
35701012025-01-12 12:02:2318 hrs ago1736683343
0xd61dA7BF...c0e76BA68
5 S
35526422025-01-12 9:45:4020 hrs ago1736675140
0xd61dA7BF...c0e76BA68
0.5 S
33781882025-01-11 11:15:4143 hrs ago1736594141
0xd61dA7BF...c0e76BA68
2 S
33610462025-01-11 8:04:5746 hrs ago1736582697
0xd61dA7BF...c0e76BA68
2 S
32571672025-01-10 13:38:492 days ago1736516329
0xd61dA7BF...c0e76BA68
2 S
32247822025-01-10 8:59:032 days ago1736499543
0xd61dA7BF...c0e76BA68
30 S
31699742025-01-09 21:34:423 days ago1736458482
0xd61dA7BF...c0e76BA68
1 S
31201382025-01-09 13:57:283 days ago1736431048
0xd61dA7BF...c0e76BA68
1 S
30985692025-01-09 9:45:083 days ago1736415908
0xd61dA7BF...c0e76BA68
200 S
30814852025-01-09 6:42:083 days ago1736404928
0xd61dA7BF...c0e76BA68
0.023 S
30814852025-01-09 6:42:083 days ago1736404928
0xd61dA7BF...c0e76BA68
0.023 S
30785692025-01-09 6:07:364 days ago1736402856
0xd61dA7BF...c0e76BA68
0.023 S
30785342025-01-09 6:07:104 days ago1736402830
0xd61dA7BF...c0e76BA68
0.016 S
30142562025-01-08 17:42:074 days ago1736358127
0xd61dA7BF...c0e76BA68
0.5 S
29341782025-01-08 2:43:225 days ago1736304202
0xd61dA7BF...c0e76BA68
2.39 S
28471952025-01-07 10:53:345 days ago1736247214
0xd61dA7BF...c0e76BA68
31 S
27511202025-01-06 19:37:456 days ago1736192265
0xd61dA7BF...c0e76BA68
50 S
Loading...
Loading

Contract Source Code Verified (Exact Match)

Contract Name:
SwapRequestInbox

Compiler Version
v0.8.19+commit.7dd6d404

Optimization Enabled:
Yes with 200 runs

Other Settings:
default evmVersion
File 1 of 17 : SwapRequestInbox.sol
// SPDX-License-Identifier: UNLICENSED
pragma solidity ^0.8.17;

import {WETH} from "solmate/src/tokens/WETH.sol";
import {ERC20, SafeTransferLib} from "solmate/src/utils/SafeTransferLib.sol";
import {Permit2HashHelper} from "../lib/Permit2HashHelper.sol";
import {RequestLib as SwapRequestLib} from "../lib/SwapRequestLib.sol";
import {SwapRequestInboxEvents} from "../common/InboxEvents.sol";
import {BasicRequest, Request as SwapRequest} from "../common/SwapRequestStructs.sol";
import {Ownable} from "../utils/Ownable.sol";
import {IBungeeGateway} from "../interfaces/IBungeeGateway.sol";
import {IPermit2} from "permit2/src/interfaces/IPermit2.sol";
import {
    InvalidChainId,
    InvalidMsgValue,
    InvalidNonce,
    TransferFailed,
    RequestDoesNotExist,
    RequestAlreadyFulfilled
} from "../common/InboxErrors.sol";

/// @title SwapRequestInbox
/// @notice Enables users to send native token same chain swap requests to Bungee Protocol auction
/// @dev To send Requests to Bungee Protocol, users are required to sign on a Request (EIP-712)
/// @dev Then submit Request + signed message to Bungee Protocol's off-chain auction service
/// @dev To enable creating orders on-chain, users can use Inboxes where request details are stored
/// @dev Request creation event is emitted which is picked up by BP's auction
/// @dev Bungee Protocol uses PERMIT-2, which uses EIP-1271 to validate stored requests on the Inbox
/// @author [email protected]
contract SwapRequestInbox is Ownable, SwapRequestInboxEvents {
    /*//////////////////////////////////////////////////////////////////////////
                                    LIBS
    //////////////////////////////////////////////////////////////////////////*/

    using SwapRequestLib for SwapRequest;
    using SafeTransferLib for ERC20;

    /*//////////////////////////////////////////////////////////////////////////
                                    CONSTANTS
    //////////////////////////////////////////////////////////////////////////*/

    /// @notice Permit2 contract address
    IPermit2 public immutable PERMIT2;

    /// @notice Wrapped native token address
    WETH public immutable WRAPPED_NATIVE_TOKEN;

    /// @notice bungeeGateway contract address
    IBungeeGateway public immutable BUNGEE_GATEWAY;

    /// @notice Address used to denote native tokens on Bungee Protocol
    address public constant NATIVE_TOKEN_ADDRESS = address(0xEeeeeEeeeEeEeeEeEeEeeEEEeeeeEeeeeeeeEEeE);

    /// @notice MAGIC_VALUE returned for valid signatures according to EIP-1271
    /// @dev https://eips.ethereum.org/EIPS/eip-1271
    bytes4 internal constant MAGIC_VALUE = 0x1626ba7e;

    /// @notice MAGIC_VALUE returned for valid signatures according to EIP-1271
    /// @dev https://eips.ethereum.org/EIPS/eip-1271
    bytes4 internal constant NON_MAGIC_VALUE = 0xffffffff;

    /*//////////////////////////////////////////////////////////////////////////
                                    PUBLIC STORAGE
    //////////////////////////////////////////////////////////////////////////*/

    /// @notice stores requests created by users
    mapping(uint256 nonce => ReceivedRequest receivedRequest) public requestInbox;

    /*//////////////////////////////////////////////////////////////////////////
                                    STRUCTS
    //////////////////////////////////////////////////////////////////////////*/

    /// @notice Struct used to store received requests against request nonce
    /// @param ogSender sender of the request
    /// @param typedDataHash typedDataHash of the Request's hash
    struct ReceivedRequest {
        address ogSender;
        bytes32 typedDataHash;
    }

    /*//////////////////////////////////////////////////////////////////////////
                                    CONSTRUCTOR
    //////////////////////////////////////////////////////////////////////////*/

    /// @dev constructor sets immutable variables and gives infinite WETH/Wrapped Native Token approval to Permit2
    /// @param _owner owner address who'll be able to call onlyOwner methods
    /// @param _permit2 permit2 contract address
    /// @param _bungeeGateway swapRequest BungeeGateway
    /// @param _wrappedNativeToken wrapped native token address
    constructor(
        address _owner,
        address _permit2,
        address _bungeeGateway,
        address payable _wrappedNativeToken
    ) Ownable(_owner) {
        PERMIT2 = IPermit2(_permit2);
        BUNGEE_GATEWAY = IBungeeGateway(_bungeeGateway);
        WRAPPED_NATIVE_TOKEN = WETH(_wrappedNativeToken);

        // Gives max Wrapped-NativeToken approval to Permit2 contract
        WRAPPED_NATIVE_TOKEN.approve(address(PERMIT2), type(uint256).max);
    }

    /*//////////////////////////////////////////////////////////////////////////
                                RECEIVE FUNCTION
    //////////////////////////////////////////////////////////////////////////*/

    /// @notice receive() function
    receive() external payable {}

    /*//////////////////////////////////////////////////////////////////////////
                                EXTERNAL FUNCTIONS
    //////////////////////////////////////////////////////////////////////////*/

    /// @notice called by the user to create a SwapRequest from native token
    /// @dev Creates SwapRequest, Wraps native token, stores typedDataHash, emits Request creation event
    /// @param swapRequest swapRequest struct
    function createSwapRequest(SwapRequest calldata swapRequest) external payable {
        // Validates Request and reverts if there are invalid details in the Request
        _checkRequestValidity(swapRequest.basicReq);

        // Wraps native asset to WETH
        WRAPPED_NATIVE_TOKEN.deposit{value: msg.value}();

        // Creates RequestHash and TypedDataHash for the Request
        (bytes32 requestHash, bytes32 typedDataHash) = _createTypedDataHash(swapRequest);

        // Store the hash against the nonce
        requestInbox[swapRequest.basicReq.nonce] = ReceivedRequest({
            ogSender: msg.sender,
            typedDataHash: typedDataHash
        });

        // Emit event
        emit SwapRequestCreated(requestHash, msg.sender, abi.encode(swapRequest));
    }

    /// @notice called by the user to unlock/withdraw funds if Request is not fulfilled
    /// @param swapRequest swapRequest struct
    function withdrawFunds(SwapRequest calldata swapRequest) external {
        // creates requestHash and typedDataHash for the Request
        (bytes32 requestHash, bytes32 typedDataHash) = _createTypedDataHash(swapRequest);

        // checks if the request was created on Inbox
        if (requestInbox[swapRequest.basicReq.nonce].typedDataHash != typedDataHash) revert RequestDoesNotExist();

        // if nonce is valid, request hasn't been extracted and funds are unlocked to the user from Inbox
        // if invalid, request has been extracted and funds are already used and swapped
        if (_isNonceValid(swapRequest.basicReq.nonce)) {
            // CASE - PRE EXTRACTION
            _withdraw(swapRequest.basicReq.nonce, swapRequest.basicReq.inputAmount, requestHash);

            // deletes request from storage
            delete requestInbox[swapRequest.basicReq.nonce];
        } else {
            /// @dev there is no post-extraction case in SwapRequests since extraction and swapping are done in one step
            revert RequestAlreadyFulfilled();
        }
    }

    /*//////////////////////////////////////////////////////////////////////////
                                Signature Verification
    //////////////////////////////////////////////////////////////////////////*/

    /// @notice Called by PERMIT2 to verify the validity of witness data signed against the nonce for transfer
    /// @param _hash PERMIT2 typedData hash
    /// @param _signature Encoded Request Details passed from PERMIT2
    function isValidSignature(bytes32 _hash, bytes calldata _signature) external view returns (bytes4 magicValue) {
        // Decodes nonce from signature
        uint256 nonce = abi.decode(_signature, (uint256));

        // Checks if hashTypedData (_hash) from PERMIT2 does not match the order stored against nonce
        if (requestInbox[nonce].typedDataHash == _hash) {
            return MAGIC_VALUE;
        } else {
            return NON_MAGIC_VALUE;
        }
    }

    /*//////////////////////////////////////////////////////////////////////////
                                ADMIN FUNCTIONS
    //////////////////////////////////////////////////////////////////////////*/

    /// @notice admin / onlyOwner function to rescue funds in case mistakenly sent to the contract
    /// @param token address of token rescued
    /// @param to address to send rescued funds to
    /// @param amount amount to be rescued
    function rescue(address token, address to, uint256 amount) external onlyOwner {
        _sendFundsFromContract(token, to, amount);
    }

    /*//////////////////////////////////////////////////////////////////////////
                                INTERNAL FUNCTIONS
    //////////////////////////////////////////////////////////////////////////*/

    /// @dev validates request details and reverts if any of the conditions are not met
    /// @param basicRequest BasicRequest struct details
    function _checkRequestValidity(BasicRequest calldata basicRequest) internal view {
        // reverts if inputAmount does not match msg.value
        if (msg.value != basicRequest.inputAmount) revert InvalidMsgValue();

        // reverts if chainId is not the current chainId
        if (basicRequest.chainId != block.chainid) revert InvalidChainId();

        // reverts if the nonce has already been used
        if (requestInbox[basicRequest.nonce].typedDataHash != bytes32(0)) revert InvalidNonce();
    }

    /// @dev creates typedDataHash which is the hash the user signs on for permit2
    /// @param swapRequest swapRequest struct
    function _createTypedDataHash(
        SwapRequest calldata swapRequest
    ) internal view returns (bytes32 requestHash, bytes32 typedDataHash) {
        // Creates Request Hash
        requestHash = swapRequest.hashRequest();

        // Hash with witness
        bytes32 hashWithWitness = Permit2HashHelper.returnHashWithWitness(
            swapRequest.basicReq.inputToken,
            swapRequest.basicReq.inputAmount,
            swapRequest.basicReq.nonce,
            swapRequest.basicReq.deadline,
            requestHash,
            SwapRequestLib.PERMIT2_ORDER_TYPE,
            swapRequest.basicReq.bungeeGateway
        );

        // Typed Data Hash
        typedDataHash = Permit2HashHelper._hashTypedData(address(PERMIT2), hashWithWitness);
    }

    /// @dev withdraws / unlocks funds back to the user if a Request is not fulfilled
    /// @dev this method is called in case the request is not yet picked up by Bungee Protocol
    /// @param nonce nonce of the Request
    /// @param amount user's inputAmount that was locked and to be withdrawn
    /// @param requestHash hash of the Request
    function _withdraw(uint256 nonce, uint256 amount, bytes32 requestHash) internal {
        // unwrap native token
        WRAPPED_NATIVE_TOKEN.withdraw(amount);

        // transfers funds to user
        _sendFundsFromContract(NATIVE_TOKEN_ADDRESS, requestInbox[nonce].ogSender, amount);

        // emits event
        emit SwapRequestWithdrawn(requestHash);
    }

    /// @dev transfers native token / ERC-20 token from the contract
    /// @param token address of token being sent
    /// @param to address the token is sent to
    /// @param amount amount being sent
    function _sendFundsFromContract(address token, address to, uint256 amount) internal {
        /// native token case
        if (token == NATIVE_TOKEN_ADDRESS) {
            (bool success, ) = to.call{value: amount, gas: 5000}("");
            if (!success) revert TransferFailed();
            return;
        }

        /// ERC20 case
        ERC20(token).safeTransfer(to, amount);
    }

    /// @notice Checks if the unordered nonce has been used on Permit2
    /// @param nonce nonce value on Permit2 whose validity is to be checked
    // @todo move to reusable lib
    function _isNonceValid(uint256 nonce) internal view returns (bool) {
        (uint256 wordPos, uint256 bitPos) = _bitmapPositions(nonce);

        // fetches the nonceBitmap value from PERMIT2
        uint256 bitmap = PERMIT2.nonceBitmap(address(this), wordPos);

        uint256 bit = 1 << bitPos;
        uint256 flipped = bitmap ^ bit;

        // checks if the nonce has been used
        if (flipped & bit == 0) return false;
        return true;
    }

    /// @notice Bit shifts nonce and returns wordPos and bitPos
    /// @param nonce nonce value
    function _bitmapPositions(uint256 nonce) private pure returns (uint256 wordPos, uint256 bitPos) {
        wordPos = uint248(nonce >> 8);
        bitPos = uint8(nonce);
    }
}

File 2 of 17 : IAllowanceTransfer.sol
// SPDX-License-Identifier: MIT
pragma solidity ^0.8.17;

import {IEIP712} from "./IEIP712.sol";

/// @title AllowanceTransfer
/// @notice Handles ERC20 token permissions through signature based allowance setting and ERC20 token transfers by checking allowed amounts
/// @dev Requires user's token approval on the Permit2 contract
interface IAllowanceTransfer is IEIP712 {
    /// @notice Thrown when an allowance on a token has expired.
    /// @param deadline The timestamp at which the allowed amount is no longer valid
    error AllowanceExpired(uint256 deadline);

    /// @notice Thrown when an allowance on a token has been depleted.
    /// @param amount The maximum amount allowed
    error InsufficientAllowance(uint256 amount);

    /// @notice Thrown when too many nonces are invalidated.
    error ExcessiveInvalidation();

    /// @notice Emits an event when the owner successfully invalidates an ordered nonce.
    event NonceInvalidation(
        address indexed owner,
        address indexed token,
        address indexed spender,
        uint48 newNonce,
        uint48 oldNonce
    );

    /// @notice Emits an event when the owner successfully sets permissions on a token for the spender.
    event Approval(
        address indexed owner,
        address indexed token,
        address indexed spender,
        uint160 amount,
        uint48 expiration
    );

    /// @notice Emits an event when the owner successfully sets permissions using a permit signature on a token for the spender.
    event Permit(
        address indexed owner,
        address indexed token,
        address indexed spender,
        uint160 amount,
        uint48 expiration,
        uint48 nonce
    );

    /// @notice Emits an event when the owner sets the allowance back to 0 with the lockdown function.
    event Lockdown(address indexed owner, address token, address spender);

    /// @notice The permit data for a token
    struct PermitDetails {
        // ERC20 token address
        address token;
        // the maximum amount allowed to spend
        uint160 amount;
        // timestamp at which a spender's token allowances become invalid
        uint48 expiration;
        // an incrementing value indexed per owner,token,and spender for each signature
        uint48 nonce;
    }

    /// @notice The permit message signed for a single token allownce
    struct PermitSingle {
        // the permit data for a single token alownce
        PermitDetails details;
        // address permissioned on the allowed tokens
        address spender;
        // deadline on the permit signature
        uint256 sigDeadline;
    }

    /// @notice The permit message signed for multiple token allowances
    struct PermitBatch {
        // the permit data for multiple token allowances
        PermitDetails[] details;
        // address permissioned on the allowed tokens
        address spender;
        // deadline on the permit signature
        uint256 sigDeadline;
    }

    /// @notice The saved permissions
    /// @dev This info is saved per owner, per token, per spender and all signed over in the permit message
    /// @dev Setting amount to type(uint160).max sets an unlimited approval
    struct PackedAllowance {
        // amount allowed
        uint160 amount;
        // permission expiry
        uint48 expiration;
        // an incrementing value indexed per owner,token,and spender for each signature
        uint48 nonce;
    }

    /// @notice A token spender pair.
    struct TokenSpenderPair {
        // the token the spender is approved
        address token;
        // the spender address
        address spender;
    }

    /// @notice Details for a token transfer.
    struct AllowanceTransferDetails {
        // the owner of the token
        address from;
        // the recipient of the token
        address to;
        // the amount of the token
        uint160 amount;
        // the token to be transferred
        address token;
    }

    /// @notice A mapping from owner address to token address to spender address to PackedAllowance struct, which contains details and conditions of the approval.
    /// @notice The mapping is indexed in the above order see: allowance[ownerAddress][tokenAddress][spenderAddress]
    /// @dev The packed slot holds the allowed amount, expiration at which the allowed amount is no longer valid, and current nonce thats updated on any signature based approvals.
    function allowance(
        address user,
        address token,
        address spender
    ) external view returns (uint160 amount, uint48 expiration, uint48 nonce);

    /// @notice Approves the spender to use up to amount of the specified token up until the expiration
    /// @param token The token to approve
    /// @param spender The spender address to approve
    /// @param amount The approved amount of the token
    /// @param expiration The timestamp at which the approval is no longer valid
    /// @dev The packed allowance also holds a nonce, which will stay unchanged in approve
    /// @dev Setting amount to type(uint160).max sets an unlimited approval
    function approve(
        address token,
        address spender,
        uint160 amount,
        uint48 expiration
    ) external;

    /// @notice Permit a spender to a given amount of the owners token via the owner's EIP-712 signature
    /// @dev May fail if the owner's nonce was invalidated in-flight by invalidateNonce
    /// @param owner The owner of the tokens being approved
    /// @param permitSingle Data signed over by the owner specifying the terms of approval
    /// @param signature The owner's signature over the permit data
    function permit(
        address owner,
        PermitSingle memory permitSingle,
        bytes calldata signature
    ) external;

    /// @notice Permit a spender to the signed amounts of the owners tokens via the owner's EIP-712 signature
    /// @dev May fail if the owner's nonce was invalidated in-flight by invalidateNonce
    /// @param owner The owner of the tokens being approved
    /// @param permitBatch Data signed over by the owner specifying the terms of approval
    /// @param signature The owner's signature over the permit data
    function permit(
        address owner,
        PermitBatch memory permitBatch,
        bytes calldata signature
    ) external;

    /// @notice Transfer approved tokens from one address to another
    /// @param from The address to transfer from
    /// @param to The address of the recipient
    /// @param amount The amount of the token to transfer
    /// @param token The token address to transfer
    /// @dev Requires the from address to have approved at least the desired amount
    /// of tokens to msg.sender.
    function transferFrom(
        address from,
        address to,
        uint160 amount,
        address token
    ) external;

    /// @notice Transfer approved tokens in a batch
    /// @param transferDetails Array of owners, recipients, amounts, and tokens for the transfers
    /// @dev Requires the from addresses to have approved at least the desired amount
    /// of tokens to msg.sender.
    function transferFrom(
        AllowanceTransferDetails[] calldata transferDetails
    ) external;

    /// @notice Enables performing a "lockdown" of the sender's Permit2 identity
    /// by batch revoking approvals
    /// @param approvals Array of approvals to revoke.
    function lockdown(TokenSpenderPair[] calldata approvals) external;

    /// @notice Invalidate nonces for a given (token, spender) pair
    /// @param token The token to invalidate nonces for
    /// @param spender The spender to invalidate nonces for
    /// @param newNonce The new nonce to set. Invalidates all nonces less than it.
    /// @dev Can't invalidate more than 2**16 nonces per transaction.
    function invalidateNonces(
        address token,
        address spender,
        uint48 newNonce
    ) external;
}

File 3 of 17 : IEIP712.sol
// SPDX-License-Identifier: MIT
pragma solidity ^0.8.17;

interface IEIP712 {
    function DOMAIN_SEPARATOR() external view returns (bytes32);
}

File 4 of 17 : IPermit2.sol
// SPDX-License-Identifier: MIT
pragma solidity ^0.8.0;

import {ISignatureTransfer} from "./ISignatureTransfer.sol";
import {IAllowanceTransfer} from "./IAllowanceTransfer.sol";

/// @notice Permit2 handles signature-based transfers in SignatureTransfer and allowance-based transfers in AllowanceTransfer.
/// @dev Users must approve Permit2 before calling any of the transfer functions.
interface IPermit2 is ISignatureTransfer, IAllowanceTransfer {
    // IPermit2 unifies the two interfaces so users have maximal flexibility with their approval.
}

File 5 of 17 : ISignatureTransfer.sol
// SPDX-License-Identifier: MIT
pragma solidity ^0.8.17;

import {IEIP712} from "./IEIP712.sol";

/// @title SignatureTransfer
/// @notice Handles ERC20 token transfers through signature based actions
/// @dev Requires user's token approval on the Permit2 contract
interface ISignatureTransfer is IEIP712 {
    /// @notice Thrown when the requested amount for a transfer is larger than the permissioned amount
    /// @param maxAmount The maximum amount a spender can request to transfer
    error InvalidAmount(uint256 maxAmount);

    /// @notice Thrown when the number of tokens permissioned to a spender does not match the number of tokens being transferred
    /// @dev If the spender does not need to transfer the number of tokens permitted, the spender can request amount 0 to be transferred
    error LengthMismatch();

    /// @notice Emits an event when the owner successfully invalidates an unordered nonce.
    event UnorderedNonceInvalidation(
        address indexed owner,
        uint256 word,
        uint256 mask
    );

    /// @notice The token and amount details for a transfer signed in the permit transfer signature
    struct TokenPermissions {
        // ERC20 token address
        address token;
        // the maximum amount that can be spent
        uint256 amount;
    }

    /// @notice The signed permit message for a single token transfer
    struct PermitTransferFrom {
        TokenPermissions permitted;
        // a unique value for every token owner's signature to prevent signature replays
        uint256 nonce;
        // deadline on the permit signature
        uint256 deadline;
    }

    /// @notice Specifies the recipient address and amount for batched transfers.
    /// @dev Recipients and amounts correspond to the index of the signed token permissions array.
    /// @dev Reverts if the requested amount is greater than the permitted signed amount.
    struct SignatureTransferDetails {
        // recipient address
        address to;
        // spender requested amount
        uint256 requestedAmount;
    }

    /// @notice Used to reconstruct the signed permit message for multiple token transfers
    /// @dev Do not need to pass in spender address as it is required that it is msg.sender
    /// @dev Note that a user still signs over a spender address
    struct PermitBatchTransferFrom {
        // the tokens and corresponding amounts permitted for a transfer
        TokenPermissions[] permitted;
        // a unique value for every token owner's signature to prevent signature replays
        uint256 nonce;
        // deadline on the permit signature
        uint256 deadline;
    }

    /// @notice A map from token owner address and a caller specified word index to a bitmap. Used to set bits in the bitmap to prevent against signature replay protection
    /// @dev Uses unordered nonces so that permit messages do not need to be spent in a certain order
    /// @dev The mapping is indexed first by the token owner, then by an index specified in the nonce
    /// @dev It returns a uint256 bitmap
    /// @dev The index, or wordPosition is capped at type(uint248).max
    function nonceBitmap(address, uint256) external view returns (uint256);

    /// @notice Transfers a token using a signed permit message
    /// @dev Reverts if the requested amount is greater than the permitted signed amount
    /// @param permit The permit data signed over by the owner
    /// @param owner The owner of the tokens to transfer
    /// @param transferDetails The spender's requested transfer details for the permitted token
    /// @param signature The signature to verify
    function permitTransferFrom(
        PermitTransferFrom memory permit,
        SignatureTransferDetails calldata transferDetails,
        address owner,
        bytes calldata signature
    ) external;

    /// @notice Transfers a token using a signed permit message
    /// @notice Includes extra data provided by the caller to verify signature over
    /// @dev The witness type string must follow EIP712 ordering of nested structs and must include the TokenPermissions type definition
    /// @dev Reverts if the requested amount is greater than the permitted signed amount
    /// @param permit The permit data signed over by the owner
    /// @param owner The owner of the tokens to transfer
    /// @param transferDetails The spender's requested transfer details for the permitted token
    /// @param witness Extra data to include when checking the user signature
    /// @param witnessTypeString The EIP-712 type definition for remaining string stub of the typehash
    /// @param signature The signature to verify
    function permitWitnessTransferFrom(
        PermitTransferFrom memory permit,
        SignatureTransferDetails calldata transferDetails,
        address owner,
        bytes32 witness,
        string calldata witnessTypeString,
        bytes calldata signature
    ) external;

    /// @notice Transfers multiple tokens using a signed permit message
    /// @param permit The permit data signed over by the owner
    /// @param owner The owner of the tokens to transfer
    /// @param transferDetails Specifies the recipient and requested amount for the token transfer
    /// @param signature The signature to verify
    function permitTransferFrom(
        PermitBatchTransferFrom memory permit,
        SignatureTransferDetails[] calldata transferDetails,
        address owner,
        bytes calldata signature
    ) external;

    /// @notice Transfers multiple tokens using a signed permit message
    /// @dev The witness type string must follow EIP712 ordering of nested structs and must include the TokenPermissions type definition
    /// @notice Includes extra data provided by the caller to verify signature over
    /// @param permit The permit data signed over by the owner
    /// @param owner The owner of the tokens to transfer
    /// @param transferDetails Specifies the recipient and requested amount for the token transfer
    /// @param witness Extra data to include when checking the user signature
    /// @param witnessTypeString The EIP-712 type definition for remaining string stub of the typehash
    /// @param signature The signature to verify
    function permitWitnessTransferFrom(
        PermitBatchTransferFrom memory permit,
        SignatureTransferDetails[] calldata transferDetails,
        address owner,
        bytes32 witness,
        string calldata witnessTypeString,
        bytes calldata signature
    ) external;

    /// @notice Invalidates the bits specified in mask for the bitmap at the word position
    /// @dev The wordPos is maxed at type(uint248).max
    /// @param wordPos A number to index the nonceBitmap at
    /// @param mask A bitmap masked against msg.sender's current bitmap at the word position
    function invalidateUnorderedNonces(uint256 wordPos, uint256 mask) external;
}

File 6 of 17 : ERC20.sol
// SPDX-License-Identifier: AGPL-3.0-only
pragma solidity >=0.8.0;

/// @notice Modern and gas efficient ERC20 + EIP-2612 implementation.
/// @author Solmate (https://github.com/transmissions11/solmate/blob/main/src/tokens/ERC20.sol)
/// @author Modified from Uniswap (https://github.com/Uniswap/uniswap-v2-core/blob/master/contracts/UniswapV2ERC20.sol)
/// @dev Do not manually set balances without updating totalSupply, as the sum of all user balances must not exceed it.
abstract contract ERC20 {
    /*//////////////////////////////////////////////////////////////
                                 EVENTS
    //////////////////////////////////////////////////////////////*/

    event Transfer(address indexed from, address indexed to, uint256 amount);

    event Approval(address indexed owner, address indexed spender, uint256 amount);

    /*//////////////////////////////////////////////////////////////
                            METADATA STORAGE
    //////////////////////////////////////////////////////////////*/

    string public name;

    string public symbol;

    uint8 public immutable decimals;

    /*//////////////////////////////////////////////////////////////
                              ERC20 STORAGE
    //////////////////////////////////////////////////////////////*/

    uint256 public totalSupply;

    mapping(address => uint256) public balanceOf;

    mapping(address => mapping(address => uint256)) public allowance;

    /*//////////////////////////////////////////////////////////////
                            EIP-2612 STORAGE
    //////////////////////////////////////////////////////////////*/

    uint256 internal immutable INITIAL_CHAIN_ID;

    bytes32 internal immutable INITIAL_DOMAIN_SEPARATOR;

    mapping(address => uint256) public nonces;

    /*//////////////////////////////////////////////////////////////
                               CONSTRUCTOR
    //////////////////////////////////////////////////////////////*/

    constructor(string memory _name, string memory _symbol, uint8 _decimals) {
        name = _name;
        symbol = _symbol;
        decimals = _decimals;

        INITIAL_CHAIN_ID = block.chainid;
        INITIAL_DOMAIN_SEPARATOR = computeDomainSeparator();
    }

    /*//////////////////////////////////////////////////////////////
                               ERC20 LOGIC
    //////////////////////////////////////////////////////////////*/

    function approve(address spender, uint256 amount) public virtual returns (bool) {
        allowance[msg.sender][spender] = amount;

        emit Approval(msg.sender, spender, amount);

        return true;
    }

    function transfer(address to, uint256 amount) public virtual returns (bool) {
        balanceOf[msg.sender] -= amount;

        // Cannot overflow because the sum of all user
        // balances can't exceed the max uint256 value.
        unchecked {
            balanceOf[to] += amount;
        }

        emit Transfer(msg.sender, to, amount);

        return true;
    }

    function transferFrom(address from, address to, uint256 amount) public virtual returns (bool) {
        uint256 allowed = allowance[from][msg.sender]; // Saves gas for limited approvals.

        if (allowed != type(uint256).max) allowance[from][msg.sender] = allowed - amount;

        balanceOf[from] -= amount;

        // Cannot overflow because the sum of all user
        // balances can't exceed the max uint256 value.
        unchecked {
            balanceOf[to] += amount;
        }

        emit Transfer(from, to, amount);

        return true;
    }

    /*//////////////////////////////////////////////////////////////
                             EIP-2612 LOGIC
    //////////////////////////////////////////////////////////////*/

    function permit(
        address owner,
        address spender,
        uint256 value,
        uint256 deadline,
        uint8 v,
        bytes32 r,
        bytes32 s
    ) public virtual {
        require(deadline >= block.timestamp, "PERMIT_DEADLINE_EXPIRED");

        // Unchecked because the only math done is incrementing
        // the owner's nonce which cannot realistically overflow.
        unchecked {
            address recoveredAddress = ecrecover(
                keccak256(
                    abi.encodePacked(
                        "\x19\x01",
                        DOMAIN_SEPARATOR(),
                        keccak256(
                            abi.encode(
                                keccak256(
                                    "Permit(address owner,address spender,uint256 value,uint256 nonce,uint256 deadline)"
                                ),
                                owner,
                                spender,
                                value,
                                nonces[owner]++,
                                deadline
                            )
                        )
                    )
                ),
                v,
                r,
                s
            );

            require(recoveredAddress != address(0) && recoveredAddress == owner, "INVALID_SIGNER");

            allowance[recoveredAddress][spender] = value;
        }

        emit Approval(owner, spender, value);
    }

    function DOMAIN_SEPARATOR() public view virtual returns (bytes32) {
        return block.chainid == INITIAL_CHAIN_ID ? INITIAL_DOMAIN_SEPARATOR : computeDomainSeparator();
    }

    function computeDomainSeparator() internal view virtual returns (bytes32) {
        return
            keccak256(
                abi.encode(
                    keccak256("EIP712Domain(string name,string version,uint256 chainId,address verifyingContract)"),
                    keccak256(bytes(name)),
                    keccak256("1"),
                    block.chainid,
                    address(this)
                )
            );
    }

    /*//////////////////////////////////////////////////////////////
                        INTERNAL MINT/BURN LOGIC
    //////////////////////////////////////////////////////////////*/

    function _mint(address to, uint256 amount) internal virtual {
        totalSupply += amount;

        // Cannot overflow because the sum of all user
        // balances can't exceed the max uint256 value.
        unchecked {
            balanceOf[to] += amount;
        }

        emit Transfer(address(0), to, amount);
    }

    function _burn(address from, uint256 amount) internal virtual {
        balanceOf[from] -= amount;

        // Cannot underflow because a user's balance
        // will never be larger than the total supply.
        unchecked {
            totalSupply -= amount;
        }

        emit Transfer(from, address(0), amount);
    }
}

File 7 of 17 : WETH.sol
// SPDX-License-Identifier: AGPL-3.0-only
pragma solidity >=0.8.0;

import {ERC20} from "./ERC20.sol";

import {SafeTransferLib} from "../utils/SafeTransferLib.sol";

/// @notice Minimalist and modern Wrapped Ether implementation.
/// @author Solmate (https://github.com/transmissions11/solmate/blob/main/src/tokens/WETH.sol)
/// @author Inspired by WETH9 (https://github.com/dapphub/ds-weth/blob/master/src/weth9.sol)
contract WETH is ERC20("Wrapped Ether", "WETH", 18) {
    using SafeTransferLib for address;

    event Deposit(address indexed from, uint256 amount);

    event Withdrawal(address indexed to, uint256 amount);

    function deposit() public payable virtual {
        _mint(msg.sender, msg.value);

        emit Deposit(msg.sender, msg.value);
    }

    function withdraw(uint256 amount) public virtual {
        _burn(msg.sender, amount);

        emit Withdrawal(msg.sender, amount);

        msg.sender.safeTransferETH(amount);
    }

    receive() external payable virtual {
        deposit();
    }
}

File 8 of 17 : SafeTransferLib.sol
// SPDX-License-Identifier: AGPL-3.0-only
pragma solidity >=0.8.0;

import {ERC20} from "../tokens/ERC20.sol";

/// @notice Safe ETH and ERC20 transfer library that gracefully handles missing return values.
/// @author Solmate (https://github.com/transmissions11/solmate/blob/main/src/utils/SafeTransferLib.sol)
/// @dev Use with caution! Some functions in this library knowingly create dirty bits at the destination of the free memory pointer.
/// @dev Note that none of the functions in this library check that a token has code at all! That responsibility is delegated to the caller.
library SafeTransferLib {
    /*//////////////////////////////////////////////////////////////
                             ETH OPERATIONS
    //////////////////////////////////////////////////////////////*/

    function safeTransferETH(address to, uint256 amount) internal {
        bool success;

        /// @solidity memory-safe-assembly
        assembly {
            // Transfer the ETH and store if it succeeded or not.
            success := call(gas(), to, amount, 0, 0, 0, 0)
        }

        require(success, "ETH_TRANSFER_FAILED");
    }

    /*//////////////////////////////////////////////////////////////
                            ERC20 OPERATIONS
    //////////////////////////////////////////////////////////////*/

    function safeTransferFrom(ERC20 token, address from, address to, uint256 amount) internal {
        bool success;

        /// @solidity memory-safe-assembly
        assembly {
            // Get a pointer to some free memory.
            let freeMemoryPointer := mload(0x40)

            // Write the abi-encoded calldata into memory, beginning with the function selector.
            mstore(freeMemoryPointer, 0x23b872dd00000000000000000000000000000000000000000000000000000000)
            mstore(add(freeMemoryPointer, 4), and(from, 0xffffffffffffffffffffffffffffffffffffffff)) // Append and mask the "from" argument.
            mstore(add(freeMemoryPointer, 36), and(to, 0xffffffffffffffffffffffffffffffffffffffff)) // Append and mask the "to" argument.
            mstore(add(freeMemoryPointer, 68), amount) // Append the "amount" argument. Masking not required as it's a full 32 byte type.

            success := and(
                // Set success to whether the call reverted, if not we check it either
                // returned exactly 1 (can't just be non-zero data), or had no return data.
                or(and(eq(mload(0), 1), gt(returndatasize(), 31)), iszero(returndatasize())),
                // We use 100 because the length of our calldata totals up like so: 4 + 32 * 3.
                // We use 0 and 32 to copy up to 32 bytes of return data into the scratch space.
                // Counterintuitively, this call must be positioned second to the or() call in the
                // surrounding and() call or else returndatasize() will be zero during the computation.
                call(gas(), token, 0, freeMemoryPointer, 100, 0, 32)
            )
        }

        require(success, "TRANSFER_FROM_FAILED");
    }

    function safeTransfer(ERC20 token, address to, uint256 amount) internal {
        bool success;

        /// @solidity memory-safe-assembly
        assembly {
            // Get a pointer to some free memory.
            let freeMemoryPointer := mload(0x40)

            // Write the abi-encoded calldata into memory, beginning with the function selector.
            mstore(freeMemoryPointer, 0xa9059cbb00000000000000000000000000000000000000000000000000000000)
            mstore(add(freeMemoryPointer, 4), and(to, 0xffffffffffffffffffffffffffffffffffffffff)) // Append and mask the "to" argument.
            mstore(add(freeMemoryPointer, 36), amount) // Append the "amount" argument. Masking not required as it's a full 32 byte type.

            success := and(
                // Set success to whether the call reverted, if not we check it either
                // returned exactly 1 (can't just be non-zero data), or had no return data.
                or(and(eq(mload(0), 1), gt(returndatasize(), 31)), iszero(returndatasize())),
                // We use 68 because the length of our calldata totals up like so: 4 + 32 * 2.
                // We use 0 and 32 to copy up to 32 bytes of return data into the scratch space.
                // Counterintuitively, this call must be positioned second to the or() call in the
                // surrounding and() call or else returndatasize() will be zero during the computation.
                call(gas(), token, 0, freeMemoryPointer, 68, 0, 32)
            )
        }

        require(success, "TRANSFER_FAILED");
    }

    function safeApprove(ERC20 token, address to, uint256 amount) internal {
        bool success;

        /// @solidity memory-safe-assembly
        assembly {
            // Get a pointer to some free memory.
            let freeMemoryPointer := mload(0x40)

            // Write the abi-encoded calldata into memory, beginning with the function selector.
            mstore(freeMemoryPointer, 0x095ea7b300000000000000000000000000000000000000000000000000000000)
            mstore(add(freeMemoryPointer, 4), and(to, 0xffffffffffffffffffffffffffffffffffffffff)) // Append and mask the "to" argument.
            mstore(add(freeMemoryPointer, 36), amount) // Append the "amount" argument. Masking not required as it's a full 32 byte type.

            success := and(
                // Set success to whether the call reverted, if not we check it either
                // returned exactly 1 (can't just be non-zero data), or had no return data.
                or(and(eq(mload(0), 1), gt(returndatasize(), 31)), iszero(returndatasize())),
                // We use 68 because the length of our calldata totals up like so: 4 + 32 * 2.
                // We use 0 and 32 to copy up to 32 bytes of return data into the scratch space.
                // Counterintuitively, this call must be positioned second to the or() call in the
                // surrounding and() call or else returndatasize() will be zero during the computation.
                call(gas(), token, 0, freeMemoryPointer, 68, 0, 32)
            )
        }

        require(success, "APPROVE_FAILED");
    }
}

File 9 of 17 : BungeeErrors.sol
// SPDX-License-Identifier: MIT
pragma solidity ^0.8.17;

//

error MofaSignatureInvalid();
error InsufficientNativeAmount();
error InvalidMultipleNativeTokens();

error FulfilmentChainInvalid();
error RequestAlreadyFulfilled();
error RouterNotRegistered();

error TransferFailed();
error CallerNotBungeeGateway();

error NoExecutionCacheFound();
error ExecutionCacheFailed();
error SwapOutputInsufficient();

error UnsupportedDestinationChainId();

error MinOutputNotMet();

error OnlyOwner();
error OnlyNominee();

error InvalidRequest();
error FulfilmentDeadlineNotMet();
error CallerNotDelegate();

error BungeeSiblingDoesNotExist();
error InvalidMsg();

error NotDelegate();
error RequestProcessed();
error RequestNotProcessed();

error InvalidSwitchboard();
error PromisedAmountNotMet();

error MsgReceiveFailed();

error RouterAlreadyWhitelisted();
error InvalidStake();
error RouterAlreadyRegistered();

error InvalidFulfil();

error InsufficientCapacity();

error ReleaseFundsNotImplemented();

File 10 of 17 : InboxErrors.sol
// SPDX-License-Identifier: UNLICENSED
pragma solidity ^0.8.17;

/// @dev reverts with error when chainId does not match block.chainid
error InvalidChainId();

/// @dev reverts with error if the msg.value and Request inputAmount do not match
error InvalidMsgValue();

/// @dev reverts with error if the nonce has already been used
error InvalidNonce();

///@dev reverts with error if native / ERC-20 transfer fails
error TransferFailed();

/// @dev reverts with error if Request does not exist on the Inbox contract
error RequestDoesNotExist();

/// @dev reverts with error if Request has already been withdrawn by the user post-extraction
error RequestAlreadyWithdraw();

/// @dev reverts with error if Request has already been extracted
error RequestAlreadyFulfilled();

File 11 of 17 : InboxEvents.sol
// SPDX-License-Identifier: MIT
pragma solidity ^0.8.17;

contract InboxEvents {
    /// @notice Emitted when the user creates a Request on the Refuel Inbox
    /// @param requestHash Hash of the request
    /// @param ogSender Address of the user who created the request
    /// @param request Encoded Request details
    event RefuelRequestCreated(bytes32 indexed requestHash, address ogSender, bytes request);

    /// @notice Emitted when the user withdraws a Request
    /// @param requestHash Hash of the request
    event RefuelRequestWithdrawn(bytes32 indexed requestHash);
}

contract SwapRequestInboxEvents {
    /// @notice Emitted when the user creates a Swap Request on the Refuel Inbox
    /// @param requestHash Hash of the request
    /// @param ogSender Address of the user who created the request
    /// @param request Encoded Request details
    event SwapRequestCreated(bytes32 indexed requestHash, address ogSender, bytes request);

    /// @notice Emitted when the user withdraws a Swap Request
    /// @param requestHash Hash of the request
    event SwapRequestWithdrawn(bytes32 indexed requestHash);
}

File 12 of 17 : SwapRequestStructs.sol
// SPDX-License-Identifier: UNLICENSED
pragma solidity ^0.8.17;

// Basic details in the request
struct BasicRequest {
    // chain id
    uint256 chainId;
    // deadline of the request
    uint256 deadline;
    // nonce used for uniqueness in signature
    uint256 nonce;
    // address of the user placing the request.
    address sender;
    // address of the receiver on destination chain
    address receiver;
    // address of bungee gateway, this address will have access to pull funds from the sender.
    address bungeeGateway;
    // address of the input token
    address inputToken;
    // amount of the input tokens
    uint256 inputAmount;
    // array of output tokens to be received on the destination.
    address[] outputTokens;
    // array of minimum amounts to be receive on the destination for the output tokens array.
    uint256[] minOutputAmounts;
}

// The Request which user signs
struct Request {
    // basic details in the request.
    BasicRequest basicReq;
    // array of addresses to check if request whitelists only certain transmitters
    address[] exclusiveTransmitters;
    // any sort of metadata to be passed with the request
    bytes32 metadata;
    // fees of the affiliate if any
    bytes affiliateFees;
    // calldata execution parameter. Only to be used when execution is required on destination.
    // minimum dest gas limit to execute calldata on destination
    uint256 minDestGas;
    // calldata to be executed on the destination
    // calldata can only be executed on the receiver in the request.
    bytes destinationPayload;
}

// Transmitter's origin chain execution details for a request with fulfilAmounts.
struct SwapExec {
    // User signed Request
    Request request;
    // array of fulfil amounts for the corresponding output tokens on the destination
    uint256[] fulfilAmounts;
    // swapPayload 0x00 if no swap is involved.
    bytes swapPayload;
    // swapRouterAddress
    address swapRouter;
    // user signature against the request
    bytes userSignature;
    // address of the beneficiary submitted by the transmitter.
    // the beneficiary will be the one receiving locked tokens when a request is settled.
    address beneficiary;
}

File 13 of 17 : IBungeeGateway.sol
// SPDX-License-Identifier: UNLICENSED
pragma solidity ^0.8.17;

interface IBungeeGateway {
    function setWhitelistedReceiver(address receiver, uint256 destinationChainId, address router) external;

    function getWhitelistedReceiver(address router, uint256 destinationChainId) external view returns (address);

    function inboundMsgFromSwitchboard(uint8 msgId, uint32 switchboardId, bytes calldata payload) external;

    function isBungeeRouter(address router) external view returns (bool);

    function withdrawnRequests(bytes32 requestHash) external view returns (bool);

    function withdrawRequestOnOrigin(bytes32 requestHash) external;

    function executeSOR(bytes calldata data) external payable returns (bytes memory);
}

File 14 of 17 : Permit2HashHelper.sol
// SPDX-License-Identifier: UNLICENSED
pragma solidity ^0.8.17;

import {IPermit2} from "permit2/src/interfaces/IPermit2.sol";
import {ISignatureTransfer} from "permit2/src/interfaces/ISignatureTransfer.sol";

/// @title Permit2HashHelper
/// @notice Helper library to generate the WitnessHash used by Permit2
/// @dev We receive hashTypedData from Permit2 in isValidSignature function, used to verify the Commande details.
/// @dev This lib helps generate the witness hash which is used to create the hashTypedData
/// @dev reference : https://github.com/Uniswap/permit2/blob/cc56ad0f3439c502c246fc5cfcc3db92bb8b7219/src/libraries/PermitHash.sol#L85
/// @dev reference : https://github.com/Uniswap/permit2/blob/cc56ad0f3439c502c246fc5cfcc3db92bb8b7219/src/EIP712.sol#L38
library Permit2HashHelper {
    string public constant _PERMIT_TRANSFER_FROM_WITNESS_TYPEHASH_STUB =
        "PermitWitnessTransferFrom(TokenPermissions permitted,address spender,uint256 nonce,uint256 deadline,";

    bytes32 public constant _TOKEN_PERMISSIONS_TYPEHASH = keccak256("TokenPermissions(address token,uint256 amount)");

    function returnHashWithWitness(
        address token,
        uint256 amount,
        uint256 nonce,
        uint256 deadline,
        bytes32 commandHash,
        string memory witnessTypeString,
        address bungeeGateway
    ) internal pure returns (bytes32) {
        ISignatureTransfer.TokenPermissions memory tokenPermissions = ISignatureTransfer.TokenPermissions(
            token,
            amount
        );
        ISignatureTransfer.PermitTransferFrom memory permit = ISignatureTransfer.PermitTransferFrom(
            tokenPermissions,
            nonce,
            deadline
        );
        return hashWithWitness(permit, commandHash, witnessTypeString, bungeeGateway);
    }

    function hashWithWitness(
        ISignatureTransfer.PermitTransferFrom memory permit,
        bytes32 witness,
        string memory witnessTypeString,
        address bungeeGateway
    ) internal pure returns (bytes32) {
        bytes32 typeHash = keccak256(abi.encodePacked(_PERMIT_TRANSFER_FROM_WITNESS_TYPEHASH_STUB, witnessTypeString));

        bytes32 tokenPermissionsHash = _hashTokenPermissions(permit.permitted);
        return
            keccak256(
                abi.encode(typeHash, tokenPermissionsHash, bungeeGateway, permit.nonce, permit.deadline, witness)
            );
    }

    function _hashTokenPermissions(
        ISignatureTransfer.TokenPermissions memory permitted
    ) private pure returns (bytes32) {
        return keccak256(abi.encode(_TOKEN_PERMISSIONS_TYPEHASH, permitted));
    }

    /// @notice Creates an EIP-712 typed data hash
    function _hashTypedData(address permit2, bytes32 dataHash) internal view returns (bytes32) {
        return keccak256(abi.encodePacked("\x19\x01", IPermit2(permit2).DOMAIN_SEPARATOR(), dataHash));
    }
}

File 15 of 17 : Permit2Lib.sol
// SPDX-License-Identifier: UNLICENSED
pragma solidity ^0.8.17;
import {ISignatureTransfer} from "permit2/src/interfaces/ISignatureTransfer.sol";

// Library to get Permit 2 related data.
library Permit2Lib {
    string public constant TOKEN_PERMISSIONS_TYPE = "TokenPermissions(address token,uint256 amount)";

    function toPermit(
        address inputToken,
        uint256 inputAmount,
        uint256 nonce,
        uint256 deadline
    ) internal pure returns (ISignatureTransfer.PermitTransferFrom memory) {
        return
            ISignatureTransfer.PermitTransferFrom({
                permitted: ISignatureTransfer.TokenPermissions({token: inputToken, amount: inputAmount}),
                nonce: nonce,
                deadline: deadline
            });
    }

    function transferDetails(
        uint256 amount,
        address spender
    ) internal pure returns (ISignatureTransfer.SignatureTransferDetails memory) {
        return ISignatureTransfer.SignatureTransferDetails({to: spender, requestedAmount: amount});
    }
}

File 16 of 17 : SwapRequestLib.sol
// SPDX-License-Identifier: GPL-3.0-only
pragma solidity ^0.8.17;

import {BasicRequest, Request, SwapExec} from "../common/SwapRequestStructs.sol";
import {Permit2Lib} from "./Permit2Lib.sol";

/// @notice helpers for handling CommandInfo objects
library BasicRequestLib {
    bytes internal constant BASIC_REQUEST_TYPE =
        abi.encodePacked(
            "BasicRequest(",
            "uint256 chainId,",
            "uint256 deadline,",
            "uint256 nonce,",
            "address sender,",
            "address receiver,",
            "address bungeeGateway,",
            "address inputToken,",
            "uint256 inputAmount,",
            "address[] outputTokens,",
            "uint256[] minOutputAmounts)"
        );
    bytes32 internal constant BASIC_REQUEST_TYPE_HASH = keccak256(BASIC_REQUEST_TYPE);

    /// @notice Hash of BasicRequest struct on the swap chain
    /// @dev enforces chainId to be the current chainId
    /// @dev helps avoid extra checking of chainId in the contract
    /// @param basicReq BasicRequest object to be hashed
    function hash(BasicRequest memory basicReq) internal view returns (bytes32) {
        return
            keccak256(
                abi.encode(
                    BASIC_REQUEST_TYPE_HASH,
                    block.chainid,
                    basicReq.deadline,
                    basicReq.nonce,
                    basicReq.sender,
                    basicReq.receiver,
                    basicReq.bungeeGateway,
                    basicReq.inputToken,
                    basicReq.inputAmount,
                    keccak256(abi.encodePacked(basicReq.outputTokens)),
                    keccak256(abi.encodePacked(basicReq.minOutputAmounts))
                )
            );
    }
}

/// @title Bungee Request Library.
/// @author bungee protocol
/// @notice This library is responsible for all the hashing related to Request object.
library RequestLib {
    using BasicRequestLib for BasicRequest;

    // Permit 2 Witness Order Type.
    string internal constant PERMIT2_ORDER_TYPE =
        string(
            abi.encodePacked(
                "Request witness)",
                abi.encodePacked(BasicRequestLib.BASIC_REQUEST_TYPE, REQUEST_TYPE),
                Permit2Lib.TOKEN_PERMISSIONS_TYPE
            )
        );

    // REQUEST TYPE encode packed
    bytes internal constant REQUEST_TYPE =
        abi.encodePacked(
            "Request(",
            "BasicRequest basicReq,",
            "address[] exclusiveTransmitters,",
            "bytes32 metadata,",
            "bytes affiliateFees,",
            "uint256 minDestGas,",
            "bytes destinationPayload)"
        );

    // SWAP EXEC TYPE.
    // @review this lib again, make sure things are solid
    bytes internal constant SWAP_EXEC_TYPE =
        abi.encodePacked(
            "SwapExec(",
            "Request request,",
            "uint256[] fulfilAmounts,",
            "bytes swapPayload,",
            "address swapRouter,",
            "bytes userSignature,",
            "address beneficiary)"
        );

    // BUNGEE_REQUEST_TYPE
    bytes internal constant BUNGEE_REQUEST_TYPE = abi.encodePacked(REQUEST_TYPE, BasicRequestLib.BASIC_REQUEST_TYPE);

    // Keccak Hash of BUNGEE_REQUEST_TYPE
    bytes32 internal constant BUNGEE_REQUEST_TYPE_HASH = keccak256(BUNGEE_REQUEST_TYPE);

    // Exec Type.
    bytes internal constant EXEC_TYPE = abi.encodePacked(SWAP_EXEC_TYPE, REQUEST_TYPE);

    // Keccak Hash of Exec Type.
    bytes32 internal constant SWAP_EXEC_TYPE_HASH = keccak256(EXEC_TYPE);

    /// @notice Hash of request on the swap chain
    /// @param request request that is signe by the user
    function hashRequest(Request memory request) internal view returns (bytes32) {
        return
            keccak256(
                abi.encode(
                    BUNGEE_REQUEST_TYPE_HASH,
                    request.basicReq.hash(),
                    keccak256(abi.encodePacked(request.exclusiveTransmitters)),
                    request.metadata,
                    keccak256(request.affiliateFees),
                    request.minDestGas,
                    keccak256(request.destinationPayload)
                )
            );
    }

    /// @notice Hash of Swap Exec on the swap chain
    /// @param execution Transmitter submitted swap exec object
    function hashSwapExec(SwapExec memory execution) internal view returns (bytes32) {
        return
            keccak256(
                abi.encode(
                    SWAP_EXEC_TYPE_HASH,
                    hashRequest(execution.request),
                    keccak256(abi.encodePacked(execution.fulfilAmounts)),
                    keccak256(execution.swapPayload),
                    execution.swapRouter,
                    keccak256(execution.userSignature),
                    execution.beneficiary
                )
            );
    }

    /// @notice hash a batch of swap execs
    /// @param swapExecs batch of swap execs to be hashed
    function hashBatch(SwapExec[] memory swapExecs) internal view returns (bytes32) {
        unchecked {
            bytes32 outputHash = keccak256("BUNGEE_SWAP_EXEC");
            // Hash all of the swap execs present in the batch.
            for (uint256 i = 0; i < swapExecs.length; i++) {
                outputHash = keccak256(abi.encode(outputHash, hashSwapExec(swapExecs[i])));
            }

            return outputHash;
        }
    }
}

File 17 of 17 : Ownable.sol
// SPDX-License-Identifier: GPL-3.0-only
pragma solidity ^0.8.17;

import {OnlyOwner, OnlyNominee} from "../common/BungeeErrors.sol";

abstract contract Ownable {
    address private _owner;
    address private _nominee;

    event OwnerNominated(address indexed nominee);
    event OwnerClaimed(address indexed claimer);

    constructor(address owner_) {
        _claimOwner(owner_);
    }

    modifier onlyOwner() {
        if (msg.sender != _owner) {
            revert OnlyOwner();
        }
        _;
    }

    function owner() public view returns (address) {
        return _owner;
    }

    function nominee() public view returns (address) {
        return _nominee;
    }

    function nominateOwner(address nominee_) external {
        if (msg.sender != _owner) {
            revert OnlyOwner();
        }
        _nominee = nominee_;
        emit OwnerNominated(_nominee);
    }

    function claimOwner() external {
        if (msg.sender != _nominee) {
            revert OnlyNominee();
        }
        _claimOwner(msg.sender);
    }

    function _claimOwner(address claimer_) internal {
        _owner = claimer_;
        _nominee = address(0);
        emit OwnerClaimed(claimer_);
    }
}

Settings
{
  "optimizer": {
    "enabled": true,
    "runs": 200
  },
  "outputSelection": {
    "*": {
      "*": [
        "evm.bytecode",
        "evm.deployedBytecode",
        "devdoc",
        "userdoc",
        "metadata",
        "abi"
      ]
    }
  },
  "libraries": {}
}

Contract Security Audit

Contract ABI

[{"inputs":[{"internalType":"address","name":"_owner","type":"address"},{"internalType":"address","name":"_permit2","type":"address"},{"internalType":"address","name":"_bungeeGateway","type":"address"},{"internalType":"address payable","name":"_wrappedNativeToken","type":"address"}],"stateMutability":"nonpayable","type":"constructor"},{"inputs":[],"name":"InvalidChainId","type":"error"},{"inputs":[],"name":"InvalidMsgValue","type":"error"},{"inputs":[],"name":"InvalidNonce","type":"error"},{"inputs":[],"name":"OnlyNominee","type":"error"},{"inputs":[],"name":"OnlyOwner","type":"error"},{"inputs":[],"name":"RequestAlreadyFulfilled","type":"error"},{"inputs":[],"name":"RequestDoesNotExist","type":"error"},{"inputs":[],"name":"TransferFailed","type":"error"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"claimer","type":"address"}],"name":"OwnerClaimed","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"nominee","type":"address"}],"name":"OwnerNominated","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"bytes32","name":"requestHash","type":"bytes32"},{"indexed":false,"internalType":"address","name":"ogSender","type":"address"},{"indexed":false,"internalType":"bytes","name":"request","type":"bytes"}],"name":"SwapRequestCreated","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"bytes32","name":"requestHash","type":"bytes32"}],"name":"SwapRequestWithdrawn","type":"event"},{"inputs":[],"name":"BUNGEE_GATEWAY","outputs":[{"internalType":"contract IBungeeGateway","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"NATIVE_TOKEN_ADDRESS","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"PERMIT2","outputs":[{"internalType":"contract IPermit2","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"WRAPPED_NATIVE_TOKEN","outputs":[{"internalType":"contract WETH","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"claimOwner","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"components":[{"components":[{"internalType":"uint256","name":"chainId","type":"uint256"},{"internalType":"uint256","name":"deadline","type":"uint256"},{"internalType":"uint256","name":"nonce","type":"uint256"},{"internalType":"address","name":"sender","type":"address"},{"internalType":"address","name":"receiver","type":"address"},{"internalType":"address","name":"bungeeGateway","type":"address"},{"internalType":"address","name":"inputToken","type":"address"},{"internalType":"uint256","name":"inputAmount","type":"uint256"},{"internalType":"address[]","name":"outputTokens","type":"address[]"},{"internalType":"uint256[]","name":"minOutputAmounts","type":"uint256[]"}],"internalType":"struct BasicRequest","name":"basicReq","type":"tuple"},{"internalType":"address[]","name":"exclusiveTransmitters","type":"address[]"},{"internalType":"bytes32","name":"metadata","type":"bytes32"},{"internalType":"bytes","name":"affiliateFees","type":"bytes"},{"internalType":"uint256","name":"minDestGas","type":"uint256"},{"internalType":"bytes","name":"destinationPayload","type":"bytes"}],"internalType":"struct Request","name":"swapRequest","type":"tuple"}],"name":"createSwapRequest","outputs":[],"stateMutability":"payable","type":"function"},{"inputs":[{"internalType":"bytes32","name":"_hash","type":"bytes32"},{"internalType":"bytes","name":"_signature","type":"bytes"}],"name":"isValidSignature","outputs":[{"internalType":"bytes4","name":"magicValue","type":"bytes4"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"nominee_","type":"address"}],"name":"nominateOwner","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"nominee","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"owner","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"nonce","type":"uint256"}],"name":"requestInbox","outputs":[{"internalType":"address","name":"ogSender","type":"address"},{"internalType":"bytes32","name":"typedDataHash","type":"bytes32"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"token","type":"address"},{"internalType":"address","name":"to","type":"address"},{"internalType":"uint256","name":"amount","type":"uint256"}],"name":"rescue","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"components":[{"components":[{"internalType":"uint256","name":"chainId","type":"uint256"},{"internalType":"uint256","name":"deadline","type":"uint256"},{"internalType":"uint256","name":"nonce","type":"uint256"},{"internalType":"address","name":"sender","type":"address"},{"internalType":"address","name":"receiver","type":"address"},{"internalType":"address","name":"bungeeGateway","type":"address"},{"internalType":"address","name":"inputToken","type":"address"},{"internalType":"uint256","name":"inputAmount","type":"uint256"},{"internalType":"address[]","name":"outputTokens","type":"address[]"},{"internalType":"uint256[]","name":"minOutputAmounts","type":"uint256[]"}],"internalType":"struct BasicRequest","name":"basicReq","type":"tuple"},{"internalType":"address[]","name":"exclusiveTransmitters","type":"address[]"},{"internalType":"bytes32","name":"metadata","type":"bytes32"},{"internalType":"bytes","name":"affiliateFees","type":"bytes"},{"internalType":"uint256","name":"minDestGas","type":"uint256"},{"internalType":"bytes","name":"destinationPayload","type":"bytes"}],"internalType":"struct Request","name":"swapRequest","type":"tuple"}],"name":"withdrawFunds","outputs":[],"stateMutability":"nonpayable","type":"function"},{"stateMutability":"payable","type":"receive"}]

60e06040523480156200001157600080fd5b5060405162001de538038062001de5833981016040819052620000349162000143565b836200004081620000d7565b506001600160a01b03838116608081905283821660c05290821660a081905260405163095ea7b360e01b8152600481019290925260001960248301529063095ea7b3906044016020604051808303816000875af1158015620000a6573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190620000cc9190620001ab565b5050505050620001d6565b600080546001600160a01b0383166001600160a01b0319918216811783556001805490921690915560405190917ffbe19c9b601f5ee90b44c7390f3fa2319eba01762d34ee372aeafd59b25c7f8791a250565b6001600160a01b03811681146200014057600080fd5b50565b600080600080608085870312156200015a57600080fd5b845162000167816200012a565b60208601519094506200017a816200012a565b60408601519093506200018d816200012a565b6060860151909250620001a0816200012a565b939692955090935050565b600060208284031215620001be57600080fd5b81518015158114620001cf57600080fd5b9392505050565b60805160a05160c051611bc362000222600039600061022a0152600081816101220152818161045a0152610a350152600081816101f60152818161092101526109870152611bc36000f3fe6080604052600436106100c65760003560e01c806366f6a69a1161007f57806382f935e21161005957806382f935e21461024c5780638da5cb5b1461026c578063a4a96bee1461028a578063df2ebdbb146102e957600080fd5b806366f6a69a146101d15780636afdd850146101e457806370372d851461021857600080fd5b80631626ba7e146100d25780631b3f8c5e1461011057806320f99c0a1461015c57806320ff430b1461017a5780633bd1adec1461019c5780635b94db27146101b157600080fd5b366100cd57005b600080fd5b3480156100de57600080fd5b506100f26100ed366004610fa6565b610311565b6040516001600160e01b031990911681526020015b60405180910390f35b34801561011c57600080fd5b506101447f000000000000000000000000000000000000000000000000000000000000000081565b6040516001600160a01b039091168152602001610107565b34801561016857600080fd5b506001546001600160a01b0316610144565b34801561018657600080fd5b5061019a61019536600461103e565b610360565b005b3480156101a857600080fd5b5061019a61039b565b3480156101bd57600080fd5b5061019a6101cc36600461107a565b6103d1565b61019a6101df366004611095565b610446565b3480156101f057600080fd5b506101447f000000000000000000000000000000000000000000000000000000000000000081565b34801561022457600080fd5b506101447f000000000000000000000000000000000000000000000000000000000000000081565b34801561025857600080fd5b5061019a610267366004611095565b61059b565b34801561027857600080fd5b506000546001600160a01b0316610144565b34801561029657600080fd5b506102ca6102a53660046110d0565b600260205260009081526040902080546001909101546001600160a01b039091169082565b604080516001600160a01b039093168352602083019190915201610107565b3480156102f557600080fd5b5061014473eeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeee81565b600080610320838501856110d0565b60008181526002602052604090206001015490915085900361034c5750630b135d3f60e11b9050610359565b506001600160e01b031990505b9392505050565b6000546001600160a01b0316331461038b57604051635fc483c560e01b815260040160405180910390fd5b610396838383610683565b505050565b6001546001600160a01b031633146103c657604051637c91ccdd60e01b815260040160405180910390fd5b6103cf3361073b565b565b6000546001600160a01b031633146103fc57604051635fc483c560e01b815260040160405180910390fd5b600180546001600160a01b0319166001600160a01b0383169081179091556040517f906a1c6bd7e3091ea86693dd029a831c19049ce77f1dce2ce0bab1cacbabce2290600090a250565b61045861045382806110e9565b61078e565b7f00000000000000000000000000000000000000000000000000000000000000006001600160a01b031663d0e30db0346040518263ffffffff1660e01b81526004016000604051808303818588803b1580156104b357600080fd5b505af11580156104c7573d6000803e3d6000fd5b50505050506000806104d883610809565b60408051808201909152338152602081018290529193509150600260006104ff86806110e9565b6040908101358252602080830193909352908101600020835181546001600160a01b0319166001600160a01b0390911617815592820151600190930192909255905183917f5f35e6c87802b75b33eba5733b4ded8a8b8d406d93e2227ed7d74f134b390d6791339161057391889101611323565b60408051601f198184030181529082905261058e929161140e565b60405180910390a2505050565b6000806105a783610809565b909250905080600260006105bb86806110e9565b60400135815260200190815260200160002060010154146105ef57604051637037cbb560e11b815260040160405180910390fd5b6106056105fc84806110e9565b6040013561094e565b1561066a5761062f61061784806110e9565b6040013561062585806110e9565b60e0013584610a1f565b6002600061063d85806110e9565b6040908101358252602082019290925201600090812080546001600160a01b031916815560010155505050565b60405163533d99dd60e01b815260040160405180910390fd5b73eeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeed196001600160a01b03841601610727576000826001600160a01b03168261138890604051600060405180830381858888f193505050503d80600081146106fa576040519150601f19603f3d011682016040523d82523d6000602084013e6106ff565b606091505b5050905080610721576040516312171d8360e31b815260040160405180910390fd5b50505050565b6103966001600160a01b0384168383610b01565b600080546001600160a01b0383166001600160a01b0319918216811783556001805490921690915560405190917ffbe19c9b601f5ee90b44c7390f3fa2319eba01762d34ee372aeafd59b25c7f8791a250565b8060e0013534146107b257604051631841b4e160e01b815260040160405180910390fd5b803546146107d357604051633d23e4d160e11b815260040160405180910390fd5b6040808201356000908152600260205220600101541561080657604051633ab3447f60e11b815260040160405180910390fd5b50565b60008061081d61081884611726565b610b86565b9150600061091a61082e85806110e9565b61083f9060e081019060c00161107a565b61084986806110e9565b60e0013561085787806110e9565b6040013561086588806110e9565b6020013587604051602001610879906117f1565b6040516020818303038152906040526040516020016108979061192a565b60408051601f19818403018152908290526108b592916020016119fc565b60408051601f1981840301815260608301909152602e808352909190611afc60208301396040516020016108ea929190611a2b565b60408051601f198184030181529190526109048b806110e9565b6109159060c081019060a00161107a565b610c9f565b90506109467f000000000000000000000000000000000000000000000000000000000000000082610cf0565b915050915091565b60008080600884901c60ff85166040516313f80ad160e21b81523060048201526024810183905291935091506000906001600160a01b037f00000000000000000000000000000000000000000000000000000000000000001690634fe02b4490604401602060405180830381865afa1580156109ce573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906109f29190611a79565b90506001821b808218808216600003610a12575060009695505050505050565b5060019695505050505050565b604051632e1a7d4d60e01b8152600481018390527f00000000000000000000000000000000000000000000000000000000000000006001600160a01b031690632e1a7d4d90602401600060405180830381600087803b158015610a8157600080fd5b505af1158015610a95573d6000803e3d6000fd5b505050600084815260026020526040902054610ad1915073eeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeee906001600160a01b031684610683565b60405181907fa60380b825a601e4b8fc7abe6ef2b41879098292cfd96335954f819ba271f09f90600090a2505050565b600060405163a9059cbb60e01b81526001600160a01b0384166004820152826024820152602060006044836000895af13d15601f3d11600160005114161716915050806107215760405162461bcd60e51b815260206004820152600f60248201526e1514905394d1915497d19052531151608a1b604482015260640160405180910390fd5b6000604051602001610b979061192a565b604051602081830303815290604052604051602001610bb5906117f1565b60408051601f1981840301815290829052610bd392916020016119fc565b60405160208183030381529060405280519060200120610bf68360000151610d92565b602080850151604051610c099201611a92565b60405160208183030381529060405280519060200120846040015185606001518051906020012086608001518760a0015180519060200120604051602001610c829796959493929190968752602087019590955260408601939093526060850191909152608084015260a083015260c082015260e00190565b604051602081830303815290604052805190602001209050919050565b6040805180820182526001600160a01b038916815260208082018990528251606081018452828152908101889052918201869052600091610ce281878787610e98565b9a9950505050505050505050565b6000826001600160a01b0316633644e5156040518163ffffffff1660e01b8152600401602060405180830381865afa158015610d30573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190610d549190611a79565b60405161190160f01b602082015260228101919091526042810183905260620160405160208183030381529060405280519060200120905092915050565b6000604051602001610da3906117f1565b604051602081830303815290604052805190602001204683602001518460400151856060015186608001518760a001518860c001518960e001518a6101000151604051602001610df39190611a92565b604051602081830303815290604052805190602001208b6101200151604051602001610e1f9190611ad1565b60408051601f1981840301815282825280516020918201209083019c909c52810199909952606089019790975260808801959095526001600160a01b0393841660a088015291831660c0870152821660e08601521661010084015261012083015261014082015261016081019190915261018001610c82565b6000806040518060a0016040528060648152602001611b2a6064913984604051602001610ec69291906119fc565b6040516020818303038152906040528051906020012090506000610eed8760000151610f52565b6020808901516040808b015181519384018790529083018490526001600160a01b0388166060840152608083019190915260a082015260c0810188905290915060e0016040516020818303038152906040528051906020012092505050949350505050565b60007f618358ac3db8dc274f0cd8829da7e234bd48cd73c4a740aede1adec9846d06a182604051602001610c8292919091825280516001600160a01b03166020808401919091520151604082015260600190565b600080600060408486031215610fbb57600080fd5b83359250602084013567ffffffffffffffff80821115610fda57600080fd5b818601915086601f830112610fee57600080fd5b813581811115610ffd57600080fd5b87602082850101111561100f57600080fd5b6020830194508093505050509250925092565b80356001600160a01b038116811461103957600080fd5b919050565b60008060006060848603121561105357600080fd5b61105c84611022565b925061106a60208501611022565b9150604084013590509250925092565b60006020828403121561108c57600080fd5b61035982611022565b6000602082840312156110a757600080fd5b813567ffffffffffffffff8111156110be57600080fd5b820160c0818503121561035957600080fd5b6000602082840312156110e257600080fd5b5035919050565b6000823561013e1983360301811261110057600080fd5b9190910192915050565b6000808335601e1984360301811261112157600080fd5b830160208101925035905067ffffffffffffffff81111561114157600080fd5b8060051b360382131561115357600080fd5b9250929050565b8183526000602080850194508260005b85811015611196576001600160a01b0361118383611022565b168752958201959082019060010161116a565b509495945050505050565b81835260006001600160fb1b038311156111ba57600080fd5b8260051b80836020870137939093016020019392505050565b60006101408235845260208301356020850152604083013560408501526111fc60608401611022565b6001600160a01b0316606085015261121660808401611022565b6001600160a01b0316608085015261123060a08401611022565b6001600160a01b031660a085015261124a60c08401611022565b6001600160a01b031660c085015260e083810135908501526101006112718185018561110a565b8383880152611283848801828461115a565b93505050506101206112978185018561110a565b868403838801526112a98482846111a1565b979650505050505050565b6000808335601e198436030181126112cb57600080fd5b830160208101925035905067ffffffffffffffff8111156112eb57600080fd5b80360382131561115357600080fd5b81835281816020850137506000828201602090810191909152601f909101601f19169091010190565b602081526000823561013e1984360301811261133e57600080fd5b60c0602084015261135460e084018583016111d3565b9050611363602085018561110a565b601f198086850301604087015261137b84838561115a565b93506040870135606087015261139460608801886112b4565b93509150808685030160808701526113ad8484846112fa565b9350608087013560a08701526113c660a08801886112b4565b93509150808685030160c0870152506113e08383836112fa565b9695505050505050565b60005b838110156114055781810151838201526020016113ed565b50506000910152565b60018060a01b0383168152604060208201526000825180604084015261143b8160608501602087016113ea565b601f01601f1916919091016060019392505050565b634e487b7160e01b600052604160045260246000fd5b604051610140810167ffffffffffffffff8111828210171561148a5761148a611450565b60405290565b60405160c0810167ffffffffffffffff8111828210171561148a5761148a611450565b604051601f8201601f1916810167ffffffffffffffff811182821017156114dc576114dc611450565b604052919050565b600067ffffffffffffffff8211156114fe576114fe611450565b5060051b60200190565b600082601f83011261151957600080fd5b8135602061152e611529836114e4565b6114b3565b82815260059290921b8401810191818101908684111561154d57600080fd5b8286015b8481101561156f5761156281611022565b8352918301918301611551565b509695505050505050565b600082601f83011261158b57600080fd5b8135602061159b611529836114e4565b82815260059290921b840181019181810190868411156115ba57600080fd5b8286015b8481101561156f57803583529183019183016115be565b600061014082840312156115e857600080fd5b6115f0611466565b905081358152602082013560208201526040820135604082015261161660608301611022565b606082015261162760808301611022565b608082015261163860a08301611022565b60a082015261164960c08301611022565b60c082015260e082013560e08201526101008083013567ffffffffffffffff8082111561167557600080fd5b61168186838701611508565b8385015261012092508285013591508082111561169d57600080fd5b506116aa8582860161157a565b82840152505092915050565b600082601f8301126116c757600080fd5b813567ffffffffffffffff8111156116e1576116e1611450565b6116f4601f8201601f19166020016114b3565b81815284602083860101111561170957600080fd5b816020850160208301376000918101602001919091529392505050565b600060c0823603121561173857600080fd5b611740611490565b823567ffffffffffffffff8082111561175857600080fd5b611764368387016115d5565b8352602085013591508082111561177a57600080fd5b61178636838701611508565b60208401526040850135604084015260608501359150808211156117a957600080fd5b6117b5368387016116b6565b60608401526080850135608084015260a08501359150808211156117d857600080fd5b506117e5368286016116b6565b60a08301525092915050565b6c084c2e6d2c6a4cae2eacae6e85609b1b81526f1d5a5b9d0c8d4d8818da185a5b92590b60821b600d820152701d5a5b9d0c8d4d88191958591b1a5b994b607a1b601d8201526d1d5a5b9d0c8d4d881b9bdb98d94b60921b602e8201526e1859191c995cdcc81cd95b99195c8b608a1b603c820152701859191c995cdcc81c9958d95a5d995c8b607a1b604b820152751859191c995cdcc8189d5b99d95951d85d195dd85e4b60521b605c820152721859191c995cdcc81a5b9c1d5d151bdad95b8b606a1b6072820152731d5a5b9d0c8d4d881a5b9c1d5d105b5bdd5b9d0b60621b60858201527f616464726573735b5d206f7574707574546f6b656e732c00000000000000000060998201527f75696e743235365b5d206d696e4f7574707574416d6f756e747329000000000060b082015260cb0190565b670a4cae2eacae6e8560c31b81527510985cda58d4995c5d595cdd0818985cda58d4995c4b60521b60088201527f616464726573735b5d206578636c75736976655472616e736d6974746572732c601e82015270189e5d195ccccc881b595d1859185d184b607a1b603e82015273189e5d195cc81859999a5b1a585d195199595ccb60621b604f820152721d5a5b9d0c8d4d881b5a5b91195cdd11d85ccb606a1b60638201527f62797465732064657374696e6174696f6e5061796c6f616429000000000000006076820152608f0190565b60008351611a0e8184602088016113ea565b835190830190611a228183602088016113ea565b01949350505050565b6f52657175657374207769746e6573732960801b815260008351611a568160108501602088016113ea565b835190830190611a6d8160108401602088016113ea565b01601001949350505050565b600060208284031215611a8b57600080fd5b5051919050565b815160009082906020808601845b83811015611ac55781516001600160a01b031685529382019390820190600101611aa0565b50929695505050505050565b815160009082906020808601845b83811015611ac557815185529382019390820190600101611adf56fe546f6b656e5065726d697373696f6e73286164647265737320746f6b656e2c75696e7432353620616d6f756e74295065726d69745769746e6573735472616e7366657246726f6d28546f6b656e5065726d697373696f6e73207065726d69747465642c61646472657373207370656e6465722c75696e74323536206e6f6e63652c75696e7432353620646561646c696e652ca264697066735822122014480a44b67d5fe8e744c0db7d308db792e07df8ef282349b46909f519b68bfa64736f6c63430008130033000000000000000000000000a5acba07788f16b4790fcbb09ca3b7fc8dd053a2000000000000000000000000000000000022d473030f116ddee9f6b43ac78ba3000000000000000000000000d26410401cc61a24205a01cc620a73c010fba290000000000000000000000000039e2fb66102314ce7b64ce5ce3e5183bc94ad38

Deployed Bytecode

0x6080604052600436106100c65760003560e01c806366f6a69a1161007f57806382f935e21161005957806382f935e21461024c5780638da5cb5b1461026c578063a4a96bee1461028a578063df2ebdbb146102e957600080fd5b806366f6a69a146101d15780636afdd850146101e457806370372d851461021857600080fd5b80631626ba7e146100d25780631b3f8c5e1461011057806320f99c0a1461015c57806320ff430b1461017a5780633bd1adec1461019c5780635b94db27146101b157600080fd5b366100cd57005b600080fd5b3480156100de57600080fd5b506100f26100ed366004610fa6565b610311565b6040516001600160e01b031990911681526020015b60405180910390f35b34801561011c57600080fd5b506101447f000000000000000000000000039e2fb66102314ce7b64ce5ce3e5183bc94ad3881565b6040516001600160a01b039091168152602001610107565b34801561016857600080fd5b506001546001600160a01b0316610144565b34801561018657600080fd5b5061019a61019536600461103e565b610360565b005b3480156101a857600080fd5b5061019a61039b565b3480156101bd57600080fd5b5061019a6101cc36600461107a565b6103d1565b61019a6101df366004611095565b610446565b3480156101f057600080fd5b506101447f000000000000000000000000000000000022d473030f116ddee9f6b43ac78ba381565b34801561022457600080fd5b506101447f000000000000000000000000d26410401cc61a24205a01cc620a73c010fba29081565b34801561025857600080fd5b5061019a610267366004611095565b61059b565b34801561027857600080fd5b506000546001600160a01b0316610144565b34801561029657600080fd5b506102ca6102a53660046110d0565b600260205260009081526040902080546001909101546001600160a01b039091169082565b604080516001600160a01b039093168352602083019190915201610107565b3480156102f557600080fd5b5061014473eeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeee81565b600080610320838501856110d0565b60008181526002602052604090206001015490915085900361034c5750630b135d3f60e11b9050610359565b506001600160e01b031990505b9392505050565b6000546001600160a01b0316331461038b57604051635fc483c560e01b815260040160405180910390fd5b610396838383610683565b505050565b6001546001600160a01b031633146103c657604051637c91ccdd60e01b815260040160405180910390fd5b6103cf3361073b565b565b6000546001600160a01b031633146103fc57604051635fc483c560e01b815260040160405180910390fd5b600180546001600160a01b0319166001600160a01b0383169081179091556040517f906a1c6bd7e3091ea86693dd029a831c19049ce77f1dce2ce0bab1cacbabce2290600090a250565b61045861045382806110e9565b61078e565b7f000000000000000000000000039e2fb66102314ce7b64ce5ce3e5183bc94ad386001600160a01b031663d0e30db0346040518263ffffffff1660e01b81526004016000604051808303818588803b1580156104b357600080fd5b505af11580156104c7573d6000803e3d6000fd5b50505050506000806104d883610809565b60408051808201909152338152602081018290529193509150600260006104ff86806110e9565b6040908101358252602080830193909352908101600020835181546001600160a01b0319166001600160a01b0390911617815592820151600190930192909255905183917f5f35e6c87802b75b33eba5733b4ded8a8b8d406d93e2227ed7d74f134b390d6791339161057391889101611323565b60408051601f198184030181529082905261058e929161140e565b60405180910390a2505050565b6000806105a783610809565b909250905080600260006105bb86806110e9565b60400135815260200190815260200160002060010154146105ef57604051637037cbb560e11b815260040160405180910390fd5b6106056105fc84806110e9565b6040013561094e565b1561066a5761062f61061784806110e9565b6040013561062585806110e9565b60e0013584610a1f565b6002600061063d85806110e9565b6040908101358252602082019290925201600090812080546001600160a01b031916815560010155505050565b60405163533d99dd60e01b815260040160405180910390fd5b73eeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeed196001600160a01b03841601610727576000826001600160a01b03168261138890604051600060405180830381858888f193505050503d80600081146106fa576040519150601f19603f3d011682016040523d82523d6000602084013e6106ff565b606091505b5050905080610721576040516312171d8360e31b815260040160405180910390fd5b50505050565b6103966001600160a01b0384168383610b01565b600080546001600160a01b0383166001600160a01b0319918216811783556001805490921690915560405190917ffbe19c9b601f5ee90b44c7390f3fa2319eba01762d34ee372aeafd59b25c7f8791a250565b8060e0013534146107b257604051631841b4e160e01b815260040160405180910390fd5b803546146107d357604051633d23e4d160e11b815260040160405180910390fd5b6040808201356000908152600260205220600101541561080657604051633ab3447f60e11b815260040160405180910390fd5b50565b60008061081d61081884611726565b610b86565b9150600061091a61082e85806110e9565b61083f9060e081019060c00161107a565b61084986806110e9565b60e0013561085787806110e9565b6040013561086588806110e9565b6020013587604051602001610879906117f1565b6040516020818303038152906040526040516020016108979061192a565b60408051601f19818403018152908290526108b592916020016119fc565b60408051601f1981840301815260608301909152602e808352909190611afc60208301396040516020016108ea929190611a2b565b60408051601f198184030181529190526109048b806110e9565b6109159060c081019060a00161107a565b610c9f565b90506109467f000000000000000000000000000000000022d473030f116ddee9f6b43ac78ba382610cf0565b915050915091565b60008080600884901c60ff85166040516313f80ad160e21b81523060048201526024810183905291935091506000906001600160a01b037f000000000000000000000000000000000022d473030f116ddee9f6b43ac78ba31690634fe02b4490604401602060405180830381865afa1580156109ce573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906109f29190611a79565b90506001821b808218808216600003610a12575060009695505050505050565b5060019695505050505050565b604051632e1a7d4d60e01b8152600481018390527f000000000000000000000000039e2fb66102314ce7b64ce5ce3e5183bc94ad386001600160a01b031690632e1a7d4d90602401600060405180830381600087803b158015610a8157600080fd5b505af1158015610a95573d6000803e3d6000fd5b505050600084815260026020526040902054610ad1915073eeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeee906001600160a01b031684610683565b60405181907fa60380b825a601e4b8fc7abe6ef2b41879098292cfd96335954f819ba271f09f90600090a2505050565b600060405163a9059cbb60e01b81526001600160a01b0384166004820152826024820152602060006044836000895af13d15601f3d11600160005114161716915050806107215760405162461bcd60e51b815260206004820152600f60248201526e1514905394d1915497d19052531151608a1b604482015260640160405180910390fd5b6000604051602001610b979061192a565b604051602081830303815290604052604051602001610bb5906117f1565b60408051601f1981840301815290829052610bd392916020016119fc565b60405160208183030381529060405280519060200120610bf68360000151610d92565b602080850151604051610c099201611a92565b60405160208183030381529060405280519060200120846040015185606001518051906020012086608001518760a0015180519060200120604051602001610c829796959493929190968752602087019590955260408601939093526060850191909152608084015260a083015260c082015260e00190565b604051602081830303815290604052805190602001209050919050565b6040805180820182526001600160a01b038916815260208082018990528251606081018452828152908101889052918201869052600091610ce281878787610e98565b9a9950505050505050505050565b6000826001600160a01b0316633644e5156040518163ffffffff1660e01b8152600401602060405180830381865afa158015610d30573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190610d549190611a79565b60405161190160f01b602082015260228101919091526042810183905260620160405160208183030381529060405280519060200120905092915050565b6000604051602001610da3906117f1565b604051602081830303815290604052805190602001204683602001518460400151856060015186608001518760a001518860c001518960e001518a6101000151604051602001610df39190611a92565b604051602081830303815290604052805190602001208b6101200151604051602001610e1f9190611ad1565b60408051601f1981840301815282825280516020918201209083019c909c52810199909952606089019790975260808801959095526001600160a01b0393841660a088015291831660c0870152821660e08601521661010084015261012083015261014082015261016081019190915261018001610c82565b6000806040518060a0016040528060648152602001611b2a6064913984604051602001610ec69291906119fc565b6040516020818303038152906040528051906020012090506000610eed8760000151610f52565b6020808901516040808b015181519384018790529083018490526001600160a01b0388166060840152608083019190915260a082015260c0810188905290915060e0016040516020818303038152906040528051906020012092505050949350505050565b60007f618358ac3db8dc274f0cd8829da7e234bd48cd73c4a740aede1adec9846d06a182604051602001610c8292919091825280516001600160a01b03166020808401919091520151604082015260600190565b600080600060408486031215610fbb57600080fd5b83359250602084013567ffffffffffffffff80821115610fda57600080fd5b818601915086601f830112610fee57600080fd5b813581811115610ffd57600080fd5b87602082850101111561100f57600080fd5b6020830194508093505050509250925092565b80356001600160a01b038116811461103957600080fd5b919050565b60008060006060848603121561105357600080fd5b61105c84611022565b925061106a60208501611022565b9150604084013590509250925092565b60006020828403121561108c57600080fd5b61035982611022565b6000602082840312156110a757600080fd5b813567ffffffffffffffff8111156110be57600080fd5b820160c0818503121561035957600080fd5b6000602082840312156110e257600080fd5b5035919050565b6000823561013e1983360301811261110057600080fd5b9190910192915050565b6000808335601e1984360301811261112157600080fd5b830160208101925035905067ffffffffffffffff81111561114157600080fd5b8060051b360382131561115357600080fd5b9250929050565b8183526000602080850194508260005b85811015611196576001600160a01b0361118383611022565b168752958201959082019060010161116a565b509495945050505050565b81835260006001600160fb1b038311156111ba57600080fd5b8260051b80836020870137939093016020019392505050565b60006101408235845260208301356020850152604083013560408501526111fc60608401611022565b6001600160a01b0316606085015261121660808401611022565b6001600160a01b0316608085015261123060a08401611022565b6001600160a01b031660a085015261124a60c08401611022565b6001600160a01b031660c085015260e083810135908501526101006112718185018561110a565b8383880152611283848801828461115a565b93505050506101206112978185018561110a565b868403838801526112a98482846111a1565b979650505050505050565b6000808335601e198436030181126112cb57600080fd5b830160208101925035905067ffffffffffffffff8111156112eb57600080fd5b80360382131561115357600080fd5b81835281816020850137506000828201602090810191909152601f909101601f19169091010190565b602081526000823561013e1984360301811261133e57600080fd5b60c0602084015261135460e084018583016111d3565b9050611363602085018561110a565b601f198086850301604087015261137b84838561115a565b93506040870135606087015261139460608801886112b4565b93509150808685030160808701526113ad8484846112fa565b9350608087013560a08701526113c660a08801886112b4565b93509150808685030160c0870152506113e08383836112fa565b9695505050505050565b60005b838110156114055781810151838201526020016113ed565b50506000910152565b60018060a01b0383168152604060208201526000825180604084015261143b8160608501602087016113ea565b601f01601f1916919091016060019392505050565b634e487b7160e01b600052604160045260246000fd5b604051610140810167ffffffffffffffff8111828210171561148a5761148a611450565b60405290565b60405160c0810167ffffffffffffffff8111828210171561148a5761148a611450565b604051601f8201601f1916810167ffffffffffffffff811182821017156114dc576114dc611450565b604052919050565b600067ffffffffffffffff8211156114fe576114fe611450565b5060051b60200190565b600082601f83011261151957600080fd5b8135602061152e611529836114e4565b6114b3565b82815260059290921b8401810191818101908684111561154d57600080fd5b8286015b8481101561156f5761156281611022565b8352918301918301611551565b509695505050505050565b600082601f83011261158b57600080fd5b8135602061159b611529836114e4565b82815260059290921b840181019181810190868411156115ba57600080fd5b8286015b8481101561156f57803583529183019183016115be565b600061014082840312156115e857600080fd5b6115f0611466565b905081358152602082013560208201526040820135604082015261161660608301611022565b606082015261162760808301611022565b608082015261163860a08301611022565b60a082015261164960c08301611022565b60c082015260e082013560e08201526101008083013567ffffffffffffffff8082111561167557600080fd5b61168186838701611508565b8385015261012092508285013591508082111561169d57600080fd5b506116aa8582860161157a565b82840152505092915050565b600082601f8301126116c757600080fd5b813567ffffffffffffffff8111156116e1576116e1611450565b6116f4601f8201601f19166020016114b3565b81815284602083860101111561170957600080fd5b816020850160208301376000918101602001919091529392505050565b600060c0823603121561173857600080fd5b611740611490565b823567ffffffffffffffff8082111561175857600080fd5b611764368387016115d5565b8352602085013591508082111561177a57600080fd5b61178636838701611508565b60208401526040850135604084015260608501359150808211156117a957600080fd5b6117b5368387016116b6565b60608401526080850135608084015260a08501359150808211156117d857600080fd5b506117e5368286016116b6565b60a08301525092915050565b6c084c2e6d2c6a4cae2eacae6e85609b1b81526f1d5a5b9d0c8d4d8818da185a5b92590b60821b600d820152701d5a5b9d0c8d4d88191958591b1a5b994b607a1b601d8201526d1d5a5b9d0c8d4d881b9bdb98d94b60921b602e8201526e1859191c995cdcc81cd95b99195c8b608a1b603c820152701859191c995cdcc81c9958d95a5d995c8b607a1b604b820152751859191c995cdcc8189d5b99d95951d85d195dd85e4b60521b605c820152721859191c995cdcc81a5b9c1d5d151bdad95b8b606a1b6072820152731d5a5b9d0c8d4d881a5b9c1d5d105b5bdd5b9d0b60621b60858201527f616464726573735b5d206f7574707574546f6b656e732c00000000000000000060998201527f75696e743235365b5d206d696e4f7574707574416d6f756e747329000000000060b082015260cb0190565b670a4cae2eacae6e8560c31b81527510985cda58d4995c5d595cdd0818985cda58d4995c4b60521b60088201527f616464726573735b5d206578636c75736976655472616e736d6974746572732c601e82015270189e5d195ccccc881b595d1859185d184b607a1b603e82015273189e5d195cc81859999a5b1a585d195199595ccb60621b604f820152721d5a5b9d0c8d4d881b5a5b91195cdd11d85ccb606a1b60638201527f62797465732064657374696e6174696f6e5061796c6f616429000000000000006076820152608f0190565b60008351611a0e8184602088016113ea565b835190830190611a228183602088016113ea565b01949350505050565b6f52657175657374207769746e6573732960801b815260008351611a568160108501602088016113ea565b835190830190611a6d8160108401602088016113ea565b01601001949350505050565b600060208284031215611a8b57600080fd5b5051919050565b815160009082906020808601845b83811015611ac55781516001600160a01b031685529382019390820190600101611aa0565b50929695505050505050565b815160009082906020808601845b83811015611ac557815185529382019390820190600101611adf56fe546f6b656e5065726d697373696f6e73286164647265737320746f6b656e2c75696e7432353620616d6f756e74295065726d69745769746e6573735472616e7366657246726f6d28546f6b656e5065726d697373696f6e73207065726d69747465642c61646472657373207370656e6465722c75696e74323536206e6f6e63652c75696e7432353620646561646c696e652ca264697066735822122014480a44b67d5fe8e744c0db7d308db792e07df8ef282349b46909f519b68bfa64736f6c63430008130033

Constructor Arguments (ABI-Encoded and is the last bytes of the Contract Creation Code above)

000000000000000000000000a5acba07788f16b4790fcbb09ca3b7fc8dd053a2000000000000000000000000000000000022d473030f116ddee9f6b43ac78ba3000000000000000000000000d26410401cc61a24205a01cc620a73c010fba290000000000000000000000000039e2fb66102314ce7b64ce5ce3e5183bc94ad38

-----Decoded View---------------
Arg [0] : _owner (address): 0xa5acBA07788f16B4790FCBb09cA3b7Fc8dd053A2
Arg [1] : _permit2 (address): 0x000000000022D473030F116dDEE9F6B43aC78BA3
Arg [2] : _bungeeGateway (address): 0xD26410401cC61a24205A01CC620A73c010FBA290
Arg [3] : _wrappedNativeToken (address): 0x039e2fB66102314Ce7b64Ce5Ce3E5183bc94aD38

-----Encoded View---------------
4 Constructor Arguments found :
Arg [0] : 000000000000000000000000a5acba07788f16b4790fcbb09ca3b7fc8dd053a2
Arg [1] : 000000000000000000000000000000000022d473030f116ddee9f6b43ac78ba3
Arg [2] : 000000000000000000000000d26410401cc61a24205a01cc620a73c010fba290
Arg [3] : 000000000000000000000000039e2fb66102314ce7b64ce5ce3e5183bc94ad38


Block Transaction Gas Used Reward
view all blocks produced

Block Uncle Number Difficulty Gas Used Reward
View All Uncles
Loading...
Loading
Loading...
Loading

Validator Index Block Amount
View All Withdrawals

Transaction Hash Block Value Eth2 PubKey Valid
View All Deposits
[ Download: CSV Export  ]
[ Download: CSV Export  ]

A contract address hosts a smart contract, which is a set of code stored on the blockchain that runs when predetermined conditions are met. Learn more about addresses in our Knowledge Base.