Feature Tip: Add private address tag to any address under My Name Tag !
Overview
ETH Balance
0 ETH
Eth Value
$0.00More Info
Private Name Tags
ContractCreator
Latest 1 from a total of 1 transactions
Transaction Hash |
Method
|
Block
|
From
|
To
|
|||||
---|---|---|---|---|---|---|---|---|---|
0x61012060 | 20114834 | 153 days ago | IN | 0 ETH | 0.0144041 |
View more zero value Internal Transactions in Advanced View mode
Advanced mode:
Loading...
Loading
Contract Name:
HypXERC20Lockbox
Compiler Version
v0.8.19+commit.7dd6d404
Contract Source Code (Solidity Standard Json-Input format)
// SPDX-License-Identifier: MIT OR Apache-2.0 pragma solidity >=0.8.0; import {IXERC20Lockbox} from "../interfaces/IXERC20Lockbox.sol"; import {IXERC20, IERC20} from "../interfaces/IXERC20.sol"; import {HypERC20Collateral} from "../HypERC20Collateral.sol"; contract HypXERC20Lockbox is HypERC20Collateral { uint256 constant MAX_INT = 2 ** 256 - 1; IXERC20Lockbox public immutable lockbox; IXERC20 public immutable xERC20; constructor( address _lockbox, address _mailbox ) HypERC20Collateral(address(IXERC20Lockbox(_lockbox).ERC20()), _mailbox) { lockbox = IXERC20Lockbox(_lockbox); xERC20 = lockbox.XERC20(); approveLockbox(); _disableInitializers(); } /** * @notice Approve the lockbox to spend the wrapped token and xERC20 * @dev This function is idempotent and need not be access controlled */ function approveLockbox() public { require( IERC20(wrappedToken).approve(address(lockbox), MAX_INT), "erc20 lockbox approve failed" ); require( xERC20.approve(address(lockbox), MAX_INT), "xerc20 lockbox approve failed" ); } /** * @notice Initialize the contract * @param _hook The address of the hook contract * @param _ism The address of the interchain security module * @param _owner The address of the owner */ function initialize( address _hook, address _ism, address _owner ) public override initializer { approveLockbox(); _MailboxClient_initialize(_hook, _ism, _owner); } function _transferFromSender( uint256 _amount ) internal override returns (bytes memory) { // transfer erc20 from sender super._transferFromSender(_amount); // convert erc20 to xERC20 lockbox.deposit(_amount); // burn xERC20 xERC20.burn(address(this), _amount); return bytes(""); } function _transferTo( address _recipient, uint256 _amount, bytes calldata /*metadata*/ ) internal override { // mint xERC20 xERC20.mint(address(this), _amount); // convert xERC20 to erc20 lockbox.withdrawTo(_recipient, _amount); } }
// SPDX-License-Identifier: UNLICENSED pragma solidity >=0.8.4 <0.9.0; // adapted from https://github.com/defi-wonderland/xERC20 import {IXERC20} from "./IXERC20.sol"; import {IERC20} from "@openzeppelin/contracts/token/ERC20/IERC20.sol"; interface IXERC20Lockbox { /** * @notice The XERC20 token of this contract */ function XERC20() external returns (IXERC20); /** * @notice The ERC20 token of this contract */ function ERC20() external returns (IERC20); /** * @notice Deposit ERC20 tokens into the lockbox * * @param _amount The amount of tokens to deposit */ function deposit(uint256 _amount) external; /** * @notice Deposit ERC20 tokens into the lockbox, and send the XERC20 to a user * * @param _user The user to send the XERC20 to * @param _amount The amount of tokens to deposit */ function depositTo(address _user, uint256 _amount) external; /** * @notice Deposit the native asset into the lockbox, and send the XERC20 to a user * * @param _user The user to send the XERC20 to */ function depositNativeTo(address _user) external payable; /** * @notice Withdraw ERC20 tokens from the lockbox * * @param _amount The amount of tokens to withdraw */ function withdraw(uint256 _amount) external; /** * @notice Withdraw ERC20 tokens from the lockbox * * @param _user The user to withdraw to * @param _amount The amount of tokens to withdraw */ function withdrawTo(address _user, uint256 _amount) external; }
// SPDX-License-Identifier: UNLICENSED pragma solidity >=0.8.0; // adapted from https://github.com/defi-wonderland/xERC20 import {IERC20} from "@openzeppelin/contracts/token/ERC20/IERC20.sol"; interface IXERC20 is IERC20 { /** * @notice Mints tokens for a user * @dev Can only be called by a minter * @param _user The address of the user who needs tokens minted * @param _amount The amount of tokens being minted */ function mint(address _user, uint256 _amount) external; /** * @notice Burns tokens for a user * @dev Can only be called by a minter * @param _user The address of the user who needs tokens burned * @param _amount The amount of tokens being burned */ function burn(address _user, uint256 _amount) external; /** * @notice Updates the limits of any bridge * @dev Can only be called by the owner * @param _mintingLimit The updated minting limit we are setting to the bridge * @param _burningLimit The updated burning limit we are setting to the bridge * @param _bridge The address of the bridge we are setting the limits too */ function setLimits( address _bridge, uint256 _mintingLimit, uint256 _burningLimit ) external; function owner() external returns (address); /** * @notice Returns the current limit of a bridge * @param _bridge the bridge we are viewing the limits of * @return _limit The limit the bridge has */ function burningCurrentLimitOf( address _bridge ) external view returns (uint256 _limit); /** * @notice Returns the current limit of a bridge * @param _bridge the bridge we are viewing the limits of * @return _limit The limit the bridge has */ function mintingCurrentLimitOf( address _bridge ) external view returns (uint256 _limit); }
// SPDX-License-Identifier: Apache-2.0 pragma solidity >=0.8.0; import {TokenRouter} from "./libs/TokenRouter.sol"; import {TokenMessage} from "./libs/TokenMessage.sol"; import {MailboxClient} from "../client/MailboxClient.sol"; import {IERC20} from "@openzeppelin/contracts/token/ERC20/IERC20.sol"; import {SafeERC20} from "@openzeppelin/contracts/token/ERC20/utils/SafeERC20.sol"; /** * @title Hyperlane ERC20 Token Collateral that wraps an existing ERC20 with remote transfer functionality. * @author Abacus Works */ contract HypERC20Collateral is TokenRouter { using SafeERC20 for IERC20; IERC20 public immutable wrappedToken; /** * @notice Constructor * @param erc20 Address of the token to keep as collateral */ constructor(address erc20, address _mailbox) TokenRouter(_mailbox) { wrappedToken = IERC20(erc20); } function initialize( address _hook, address _interchainSecurityModule, address _owner ) public virtual initializer { _MailboxClient_initialize(_hook, _interchainSecurityModule, _owner); } function balanceOf( address _account ) external view override returns (uint256) { return wrappedToken.balanceOf(_account); } /** * @dev Transfers `_amount` of `wrappedToken` from `msg.sender` to this contract. * @inheritdoc TokenRouter */ function _transferFromSender( uint256 _amount ) internal virtual override returns (bytes memory) { wrappedToken.safeTransferFrom(msg.sender, address(this), _amount); return bytes(""); // no metadata } /** * @dev Transfers `_amount` of `wrappedToken` from this contract to `_recipient`. * @inheritdoc TokenRouter */ function _transferTo( address _recipient, uint256 _amount, bytes calldata // no metadata ) internal virtual override { wrappedToken.safeTransfer(_recipient, _amount); } }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v4.9.0) (token/ERC20/IERC20.sol) pragma solidity ^0.8.0; /** * @dev Interface of the ERC20 standard as defined in the EIP. */ interface IERC20 { /** * @dev Emitted when `value` tokens are moved from one account (`from`) to * another (`to`). * * Note that `value` may be zero. */ event Transfer(address indexed from, address indexed to, uint256 value); /** * @dev Emitted when the allowance of a `spender` for an `owner` is set by * a call to {approve}. `value` is the new allowance. */ event Approval(address indexed owner, address indexed spender, uint256 value); /** * @dev Returns the amount of tokens in existence. */ function totalSupply() external view returns (uint256); /** * @dev Returns the amount of tokens owned by `account`. */ function balanceOf(address account) external view returns (uint256); /** * @dev Moves `amount` tokens from the caller's account to `to`. * * Returns a boolean value indicating whether the operation succeeded. * * Emits a {Transfer} event. */ function transfer(address to, uint256 amount) external returns (bool); /** * @dev Returns the remaining number of tokens that `spender` will be * allowed to spend on behalf of `owner` through {transferFrom}. This is * zero by default. * * This value changes when {approve} or {transferFrom} are called. */ function allowance(address owner, address spender) external view returns (uint256); /** * @dev Sets `amount` as the allowance of `spender` over the caller's tokens. * * Returns a boolean value indicating whether the operation succeeded. * * IMPORTANT: Beware that changing an allowance with this method brings the risk * that someone may use both the old and the new allowance by unfortunate * transaction ordering. One possible solution to mitigate this race * condition is to first reduce the spender's allowance to 0 and set the * desired value afterwards: * https://github.com/ethereum/EIPs/issues/20#issuecomment-263524729 * * Emits an {Approval} event. */ function approve(address spender, uint256 amount) external returns (bool); /** * @dev Moves `amount` tokens from `from` to `to` using the * allowance mechanism. `amount` is then deducted from the caller's * allowance. * * Returns a boolean value indicating whether the operation succeeded. * * Emits a {Transfer} event. */ function transferFrom(address from, address to, uint256 amount) external returns (bool); }
// SPDX-License-Identifier: Apache-2.0 pragma solidity >=0.8.0; import {IPostDispatchHook} from "../../interfaces/hooks/IPostDispatchHook.sol"; import {GasRouter} from "../../client/GasRouter.sol"; import {MailboxClient} from "../../client/MailboxClient.sol"; import {TypeCasts} from "../../libs/TypeCasts.sol"; import {TokenMessage} from "./TokenMessage.sol"; /** * @title Hyperlane Token Router that extends Router with abstract token (ERC20/ERC721) remote transfer functionality. * @author Abacus Works */ abstract contract TokenRouter is GasRouter { using TypeCasts for bytes32; using TypeCasts for address; using TokenMessage for bytes; /** * @dev Emitted on `transferRemote` when a transfer message is dispatched. * @param destination The identifier of the destination chain. * @param recipient The address of the recipient on the destination chain. * @param amount The amount of tokens burnt on the origin chain. */ event SentTransferRemote( uint32 indexed destination, bytes32 indexed recipient, uint256 amount ); /** * @dev Emitted on `_handle` when a transfer message is processed. * @param origin The identifier of the origin chain. * @param recipient The address of the recipient on the destination chain. * @param amount The amount of tokens minted on the destination chain. */ event ReceivedTransferRemote( uint32 indexed origin, bytes32 indexed recipient, uint256 amount ); constructor(address _mailbox) GasRouter(_mailbox) {} /** * @notice Transfers `_amountOrId` token to `_recipient` on `_destination` domain. * @dev Delegates transfer logic to `_transferFromSender` implementation. * @dev Emits `SentTransferRemote` event on the origin chain. * @param _destination The identifier of the destination chain. * @param _recipient The address of the recipient on the destination chain. * @param _amountOrId The amount or identifier of tokens to be sent to the remote recipient. * @return messageId The identifier of the dispatched message. */ function transferRemote( uint32 _destination, bytes32 _recipient, uint256 _amountOrId ) external payable virtual returns (bytes32 messageId) { return _transferRemote(_destination, _recipient, _amountOrId, msg.value); } /** * @notice Transfers `_amountOrId` token to `_recipient` on `_destination` domain with a specified hook * @dev Delegates transfer logic to `_transferFromSender` implementation. * @dev The metadata is the token metadata, and is DIFFERENT than the hook metadata. * @dev Emits `SentTransferRemote` event on the origin chain. * @param _destination The identifier of the destination chain. * @param _recipient The address of the recipient on the destination chain. * @param _amountOrId The amount or identifier of tokens to be sent to the remote recipient. * @param _hookMetadata The metadata passed into the hook * @param _hook The post dispatch hook to be called by the Mailbox * @return messageId The identifier of the dispatched message. */ function transferRemote( uint32 _destination, bytes32 _recipient, uint256 _amountOrId, bytes calldata _hookMetadata, address _hook ) external payable virtual returns (bytes32 messageId) { return _transferRemote( _destination, _recipient, _amountOrId, msg.value, _hookMetadata, _hook ); } function _transferRemote( uint32 _destination, bytes32 _recipient, uint256 _amountOrId, uint256 _value ) internal returns (bytes32 messageId) { return _transferRemote( _destination, _recipient, _amountOrId, _value, _GasRouter_hookMetadata(_destination), address(hook) ); } function _transferRemote( uint32 _destination, bytes32 _recipient, uint256 _amountOrId, uint256 _value, bytes memory _hookMetadata, address _hook ) internal virtual returns (bytes32 messageId) { bytes memory _tokenMetadata = _transferFromSender(_amountOrId); bytes memory _tokenMessage = TokenMessage.format( _recipient, _amountOrId, _tokenMetadata ); messageId = _Router_dispatch( _destination, _value, _tokenMessage, _hookMetadata, _hook ); emit SentTransferRemote(_destination, _recipient, _amountOrId); } /** * @dev Should transfer `_amountOrId` of tokens from `msg.sender` to this token router. * @dev Called by `transferRemote` before message dispatch. * @dev Optionally returns `metadata` associated with the transfer to be passed in message. */ function _transferFromSender( uint256 _amountOrId ) internal virtual returns (bytes memory metadata); /** * @notice Returns the balance of `account` on this token router. * @param account The address to query the balance of. * @return The balance of `account`. */ function balanceOf(address account) external virtual returns (uint256); /** * @dev Mints tokens to recipient when router receives transfer message. * @dev Emits `ReceivedTransferRemote` event on the destination chain. * @param _origin The identifier of the origin chain. * @param _message The encoded remote transfer message containing the recipient address and amount. */ function _handle( uint32 _origin, bytes32, bytes calldata _message ) internal virtual override { bytes32 recipient = _message.recipient(); uint256 amount = _message.amount(); bytes calldata metadata = _message.metadata(); _transferTo(recipient.bytes32ToAddress(), amount, metadata); emit ReceivedTransferRemote(_origin, recipient, amount); } /** * @dev Should transfer `_amountOrId` of tokens from this token router to `_recipient`. * @dev Called by `handle` after message decoding. * @dev Optionally handles `metadata` associated with transfer passed in message. */ function _transferTo( address _recipient, uint256 _amountOrId, bytes calldata metadata ) internal virtual; }
// SPDX-License-Identifier: MIT OR Apache-2.0 pragma solidity >=0.8.0; library TokenMessage { function format( bytes32 _recipient, uint256 _amount, bytes memory _metadata ) internal pure returns (bytes memory) { return abi.encodePacked(_recipient, _amount, _metadata); } function recipient(bytes calldata message) internal pure returns (bytes32) { return bytes32(message[0:32]); } function amount(bytes calldata message) internal pure returns (uint256) { return uint256(bytes32(message[32:64])); } // alias for ERC721 function tokenId(bytes calldata message) internal pure returns (uint256) { return amount(message); } function metadata( bytes calldata message ) internal pure returns (bytes calldata) { return message[64:]; } }
// SPDX-License-Identifier: MIT OR Apache-2.0 pragma solidity >=0.6.11; // ============ Internal Imports ============ import {IMailbox} from "../interfaces/IMailbox.sol"; import {IPostDispatchHook} from "../interfaces/hooks/IPostDispatchHook.sol"; import {IInterchainSecurityModule} from "../interfaces/IInterchainSecurityModule.sol"; import {Message} from "../libs/Message.sol"; // ============ External Imports ============ import {Address} from "@openzeppelin/contracts/utils/Address.sol"; import {OwnableUpgradeable} from "@openzeppelin/contracts-upgradeable/access/OwnableUpgradeable.sol"; abstract contract MailboxClient is OwnableUpgradeable { using Message for bytes; IMailbox public immutable mailbox; uint32 public immutable localDomain; IPostDispatchHook public hook; IInterchainSecurityModule public interchainSecurityModule; uint256[48] private __GAP; // gap for upgrade safety // ============ Modifiers ============ modifier onlyContract(address _contract) { require( Address.isContract(_contract), "MailboxClient: invalid mailbox" ); _; } modifier onlyContractOrNull(address _contract) { require( Address.isContract(_contract) || _contract == address(0), "MailboxClient: invalid contract setting" ); _; } /** * @notice Only accept messages from an Hyperlane Mailbox contract */ modifier onlyMailbox() { require( msg.sender == address(mailbox), "MailboxClient: sender not mailbox" ); _; } constructor(address _mailbox) onlyContract(_mailbox) { mailbox = IMailbox(_mailbox); localDomain = mailbox.localDomain(); _transferOwnership(msg.sender); } /** * @notice Sets the address of the application's custom hook. * @param _hook The address of the hook contract. */ function setHook(address _hook) public onlyContractOrNull(_hook) onlyOwner { hook = IPostDispatchHook(_hook); } /** * @notice Sets the address of the application's custom interchain security module. * @param _module The address of the interchain security module contract. */ function setInterchainSecurityModule( address _module ) public onlyContractOrNull(_module) onlyOwner { interchainSecurityModule = IInterchainSecurityModule(_module); } // ======== Initializer ========= function _MailboxClient_initialize( address _hook, address _interchainSecurityModule, address _owner ) internal onlyInitializing { __Ownable_init(); setHook(_hook); setInterchainSecurityModule(_interchainSecurityModule); _transferOwnership(_owner); } function _isLatestDispatched(bytes32 id) internal view returns (bool) { return mailbox.latestDispatchedId() == id; } function _isDelivered(bytes32 id) internal view returns (bool) { return mailbox.delivered(id); } }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v4.9.3) (token/ERC20/utils/SafeERC20.sol) pragma solidity ^0.8.0; import "../IERC20.sol"; import "../extensions/IERC20Permit.sol"; import "../../../utils/Address.sol"; /** * @title SafeERC20 * @dev Wrappers around ERC20 operations that throw on failure (when the token * contract returns false). Tokens that return no value (and instead revert or * throw on failure) are also supported, non-reverting calls are assumed to be * successful. * To use this library you can add a `using SafeERC20 for IERC20;` statement to your contract, * which allows you to call the safe operations as `token.safeTransfer(...)`, etc. */ library SafeERC20 { using Address for address; /** * @dev Transfer `value` amount of `token` from the calling contract to `to`. If `token` returns no value, * non-reverting calls are assumed to be successful. */ function safeTransfer(IERC20 token, address to, uint256 value) internal { _callOptionalReturn(token, abi.encodeWithSelector(token.transfer.selector, to, value)); } /** * @dev Transfer `value` amount of `token` from `from` to `to`, spending the approval given by `from` to the * calling contract. If `token` returns no value, non-reverting calls are assumed to be successful. */ function safeTransferFrom(IERC20 token, address from, address to, uint256 value) internal { _callOptionalReturn(token, abi.encodeWithSelector(token.transferFrom.selector, from, to, value)); } /** * @dev Deprecated. This function has issues similar to the ones found in * {IERC20-approve}, and its usage is discouraged. * * Whenever possible, use {safeIncreaseAllowance} and * {safeDecreaseAllowance} instead. */ function safeApprove(IERC20 token, address spender, uint256 value) internal { // safeApprove should only be called when setting an initial allowance, // or when resetting it to zero. To increase and decrease it, use // 'safeIncreaseAllowance' and 'safeDecreaseAllowance' require( (value == 0) || (token.allowance(address(this), spender) == 0), "SafeERC20: approve from non-zero to non-zero allowance" ); _callOptionalReturn(token, abi.encodeWithSelector(token.approve.selector, spender, value)); } /** * @dev Increase the calling contract's allowance toward `spender` by `value`. If `token` returns no value, * non-reverting calls are assumed to be successful. */ function safeIncreaseAllowance(IERC20 token, address spender, uint256 value) internal { uint256 oldAllowance = token.allowance(address(this), spender); _callOptionalReturn(token, abi.encodeWithSelector(token.approve.selector, spender, oldAllowance + value)); } /** * @dev Decrease the calling contract's allowance toward `spender` by `value`. If `token` returns no value, * non-reverting calls are assumed to be successful. */ function safeDecreaseAllowance(IERC20 token, address spender, uint256 value) internal { unchecked { uint256 oldAllowance = token.allowance(address(this), spender); require(oldAllowance >= value, "SafeERC20: decreased allowance below zero"); _callOptionalReturn(token, abi.encodeWithSelector(token.approve.selector, spender, oldAllowance - value)); } } /** * @dev Set the calling contract's allowance toward `spender` to `value`. If `token` returns no value, * non-reverting calls are assumed to be successful. Meant to be used with tokens that require the approval * to be set to zero before setting it to a non-zero value, such as USDT. */ function forceApprove(IERC20 token, address spender, uint256 value) internal { bytes memory approvalCall = abi.encodeWithSelector(token.approve.selector, spender, value); if (!_callOptionalReturnBool(token, approvalCall)) { _callOptionalReturn(token, abi.encodeWithSelector(token.approve.selector, spender, 0)); _callOptionalReturn(token, approvalCall); } } /** * @dev Use a ERC-2612 signature to set the `owner` approval toward `spender` on `token`. * Revert on invalid signature. */ function safePermit( IERC20Permit token, address owner, address spender, uint256 value, uint256 deadline, uint8 v, bytes32 r, bytes32 s ) internal { uint256 nonceBefore = token.nonces(owner); token.permit(owner, spender, value, deadline, v, r, s); uint256 nonceAfter = token.nonces(owner); require(nonceAfter == nonceBefore + 1, "SafeERC20: permit did not succeed"); } /** * @dev Imitates a Solidity high-level call (i.e. a regular function call to a contract), relaxing the requirement * on the return value: the return value is optional (but if data is returned, it must not be false). * @param token The token targeted by the call. * @param data The call data (encoded using abi.encode or one of its variants). */ function _callOptionalReturn(IERC20 token, bytes memory data) private { // We need to perform a low level call here, to bypass Solidity's return data size checking mechanism, since // we're implementing it ourselves. We use {Address-functionCall} to perform this call, which verifies that // the target address contains contract code and also asserts for success in the low-level call. bytes memory returndata = address(token).functionCall(data, "SafeERC20: low-level call failed"); require(returndata.length == 0 || abi.decode(returndata, (bool)), "SafeERC20: ERC20 operation did not succeed"); } /** * @dev Imitates a Solidity high-level call (i.e. a regular function call to a contract), relaxing the requirement * on the return value: the return value is optional (but if data is returned, it must not be false). * @param token The token targeted by the call. * @param data The call data (encoded using abi.encode or one of its variants). * * This is a variant of {_callOptionalReturn} that silents catches all reverts and returns a bool instead. */ function _callOptionalReturnBool(IERC20 token, bytes memory data) private returns (bool) { // We need to perform a low level call here, to bypass Solidity's return data size checking mechanism, since // we're implementing it ourselves. We cannot use {Address-functionCall} here since this should return false // and not revert is the subcall reverts. (bool success, bytes memory returndata) = address(token).call(data); return success && (returndata.length == 0 || abi.decode(returndata, (bool))) && Address.isContract(address(token)); } }
// SPDX-License-Identifier: MIT OR Apache-2.0 pragma solidity >=0.8.0; /*@@@@@@@ @@@@@@@@@ @@@@@@@@@ @@@@@@@@@ @@@@@@@@@ @@@@@@@@@ @@@@@@@@@ @@@@@@@@@ @@@@@@@@@@@@@@@@@@@@@@@@@ @@@@@ HYPERLANE @@@@@@@ @@@@@@@@@@@@@@@@@@@@@@@@@ @@@@@@@@@ @@@@@@@@@ @@@@@@@@@ @@@@@@@@@ @@@@@@@@@ @@@@@@@@@ @@@@@@@@@ @@@@@@@@*/ interface IPostDispatchHook { enum Types { UNUSED, ROUTING, AGGREGATION, MERKLE_TREE, INTERCHAIN_GAS_PAYMASTER, FALLBACK_ROUTING, ID_AUTH_ISM, PAUSABLE, PROTOCOL_FEE, LAYER_ZERO_V1, Rate_Limited_Hook } /** * @notice Returns an enum that represents the type of hook */ function hookType() external view returns (uint8); /** * @notice Returns whether the hook supports metadata * @param metadata metadata * @return Whether the hook supports metadata */ function supportsMetadata( bytes calldata metadata ) external view returns (bool); /** * @notice Post action after a message is dispatched via the Mailbox * @param metadata The metadata required for the hook * @param message The message passed from the Mailbox.dispatch() call */ function postDispatch( bytes calldata metadata, bytes calldata message ) external payable; /** * @notice Compute the payment required by the postDispatch call * @param metadata The metadata required for the hook * @param message The message passed from the Mailbox.dispatch() call * @return Quoted payment for the postDispatch call */ function quoteDispatch( bytes calldata metadata, bytes calldata message ) external view returns (uint256); }
// SPDX-License-Identifier: MIT OR Apache-2.0 pragma solidity >=0.6.11; import {Router} from "./Router.sol"; import {StandardHookMetadata} from "../hooks/libs/StandardHookMetadata.sol"; abstract contract GasRouter is Router { // ============ Mutable Storage ============ mapping(uint32 => uint256) public destinationGas; struct GasRouterConfig { uint32 domain; uint256 gas; } constructor(address _mailbox) Router(_mailbox) {} /** * @notice Sets the gas amount dispatched for each configured domain. * @param gasConfigs The array of GasRouterConfig structs */ function setDestinationGas( GasRouterConfig[] calldata gasConfigs ) external onlyOwner { for (uint256 i = 0; i < gasConfigs.length; i += 1) { _setDestinationGas(gasConfigs[i].domain, gasConfigs[i].gas); } } /** * @notice Sets the gas amount dispatched for each configured domain. * @param domain The destination domain ID * @param gas The gas limit */ function setDestinationGas(uint32 domain, uint256 gas) external onlyOwner { _setDestinationGas(domain, gas); } /** * @notice Returns the gas payment required to dispatch a message to the given domain's router. * @param _destinationDomain The domain of the router. * @return _gasPayment Payment computed by the registered InterchainGasPaymaster. */ function quoteGasPayment( uint32 _destinationDomain ) external view returns (uint256) { return _GasRouter_quoteDispatch(_destinationDomain, "", address(hook)); } function _GasRouter_hookMetadata( uint32 _destination ) internal view returns (bytes memory) { return StandardHookMetadata.overrideGasLimit(destinationGas[_destination]); } function _setDestinationGas(uint32 domain, uint256 gas) internal { destinationGas[domain] = gas; } function _GasRouter_dispatch( uint32 _destination, uint256 _value, bytes memory _messageBody, address _hook ) internal returns (bytes32) { return _Router_dispatch( _destination, _value, _messageBody, _GasRouter_hookMetadata(_destination), _hook ); } function _GasRouter_quoteDispatch( uint32 _destination, bytes memory _messageBody, address _hook ) internal view returns (uint256) { return _Router_quoteDispatch( _destination, _messageBody, _GasRouter_hookMetadata(_destination), _hook ); } }
// SPDX-License-Identifier: MIT OR Apache-2.0 pragma solidity >=0.6.11; library TypeCasts { // alignment preserving cast function addressToBytes32(address _addr) internal pure returns (bytes32) { return bytes32(uint256(uint160(_addr))); } // alignment preserving cast function bytes32ToAddress(bytes32 _buf) internal pure returns (address) { return address(uint160(uint256(_buf))); } }
// SPDX-License-Identifier: MIT OR Apache-2.0 pragma solidity >=0.8.0; import {IInterchainSecurityModule} from "./IInterchainSecurityModule.sol"; import {IPostDispatchHook} from "./hooks/IPostDispatchHook.sol"; interface IMailbox { // ============ Events ============ /** * @notice Emitted when a new message is dispatched via Hyperlane * @param sender The address that dispatched the message * @param destination The destination domain of the message * @param recipient The message recipient address on `destination` * @param message Raw bytes of message */ event Dispatch( address indexed sender, uint32 indexed destination, bytes32 indexed recipient, bytes message ); /** * @notice Emitted when a new message is dispatched via Hyperlane * @param messageId The unique message identifier */ event DispatchId(bytes32 indexed messageId); /** * @notice Emitted when a Hyperlane message is processed * @param messageId The unique message identifier */ event ProcessId(bytes32 indexed messageId); /** * @notice Emitted when a Hyperlane message is delivered * @param origin The origin domain of the message * @param sender The message sender address on `origin` * @param recipient The address that handled the message */ event Process( uint32 indexed origin, bytes32 indexed sender, address indexed recipient ); function localDomain() external view returns (uint32); function delivered(bytes32 messageId) external view returns (bool); function defaultIsm() external view returns (IInterchainSecurityModule); function defaultHook() external view returns (IPostDispatchHook); function requiredHook() external view returns (IPostDispatchHook); function latestDispatchedId() external view returns (bytes32); function dispatch( uint32 destinationDomain, bytes32 recipientAddress, bytes calldata messageBody ) external payable returns (bytes32 messageId); function quoteDispatch( uint32 destinationDomain, bytes32 recipientAddress, bytes calldata messageBody ) external view returns (uint256 fee); function dispatch( uint32 destinationDomain, bytes32 recipientAddress, bytes calldata body, bytes calldata defaultHookMetadata ) external payable returns (bytes32 messageId); function quoteDispatch( uint32 destinationDomain, bytes32 recipientAddress, bytes calldata messageBody, bytes calldata defaultHookMetadata ) external view returns (uint256 fee); function dispatch( uint32 destinationDomain, bytes32 recipientAddress, bytes calldata body, bytes calldata customHookMetadata, IPostDispatchHook customHook ) external payable returns (bytes32 messageId); function quoteDispatch( uint32 destinationDomain, bytes32 recipientAddress, bytes calldata messageBody, bytes calldata customHookMetadata, IPostDispatchHook customHook ) external view returns (uint256 fee); function process( bytes calldata metadata, bytes calldata message ) external payable; function recipientIsm( address recipient ) external view returns (IInterchainSecurityModule module); }
// SPDX-License-Identifier: MIT OR Apache-2.0 pragma solidity >=0.6.11; interface IInterchainSecurityModule { enum Types { UNUSED, ROUTING, AGGREGATION, LEGACY_MULTISIG, MERKLE_ROOT_MULTISIG, MESSAGE_ID_MULTISIG, NULL, // used with relayer carrying no metadata CCIP_READ } /** * @notice Returns an enum that represents the type of security model * encoded by this ISM. * @dev Relayers infer how to fetch and format metadata. */ function moduleType() external view returns (uint8); /** * @notice Defines a security model responsible for verifying interchain * messages based on the provided metadata. * @param _metadata Off-chain metadata provided by a relayer, specific to * the security model encoded by the module (e.g. validator signatures) * @param _message Hyperlane encoded interchain message * @return True if the message was verified */ function verify( bytes calldata _metadata, bytes calldata _message ) external returns (bool); } interface ISpecifiesInterchainSecurityModule { function interchainSecurityModule() external view returns (IInterchainSecurityModule); }
// SPDX-License-Identifier: MIT OR Apache-2.0 pragma solidity >=0.8.0; import {TypeCasts} from "./TypeCasts.sol"; /** * @title Hyperlane Message Library * @notice Library for formatted messages used by Mailbox **/ library Message { using TypeCasts for bytes32; uint256 private constant VERSION_OFFSET = 0; uint256 private constant NONCE_OFFSET = 1; uint256 private constant ORIGIN_OFFSET = 5; uint256 private constant SENDER_OFFSET = 9; uint256 private constant DESTINATION_OFFSET = 41; uint256 private constant RECIPIENT_OFFSET = 45; uint256 private constant BODY_OFFSET = 77; /** * @notice Returns formatted (packed) Hyperlane message with provided fields * @dev This function should only be used in memory message construction. * @param _version The version of the origin and destination Mailboxes * @param _nonce A nonce to uniquely identify the message on its origin chain * @param _originDomain Domain of origin chain * @param _sender Address of sender as bytes32 * @param _destinationDomain Domain of destination chain * @param _recipient Address of recipient on destination chain as bytes32 * @param _messageBody Raw bytes of message body * @return Formatted message */ function formatMessage( uint8 _version, uint32 _nonce, uint32 _originDomain, bytes32 _sender, uint32 _destinationDomain, bytes32 _recipient, bytes calldata _messageBody ) internal pure returns (bytes memory) { return abi.encodePacked( _version, _nonce, _originDomain, _sender, _destinationDomain, _recipient, _messageBody ); } /** * @notice Returns the message ID. * @param _message ABI encoded Hyperlane message. * @return ID of `_message` */ function id(bytes memory _message) internal pure returns (bytes32) { return keccak256(_message); } /** * @notice Returns the message version. * @param _message ABI encoded Hyperlane message. * @return Version of `_message` */ function version(bytes calldata _message) internal pure returns (uint8) { return uint8(bytes1(_message[VERSION_OFFSET:NONCE_OFFSET])); } /** * @notice Returns the message nonce. * @param _message ABI encoded Hyperlane message. * @return Nonce of `_message` */ function nonce(bytes calldata _message) internal pure returns (uint32) { return uint32(bytes4(_message[NONCE_OFFSET:ORIGIN_OFFSET])); } /** * @notice Returns the message origin domain. * @param _message ABI encoded Hyperlane message. * @return Origin domain of `_message` */ function origin(bytes calldata _message) internal pure returns (uint32) { return uint32(bytes4(_message[ORIGIN_OFFSET:SENDER_OFFSET])); } /** * @notice Returns the message sender as bytes32. * @param _message ABI encoded Hyperlane message. * @return Sender of `_message` as bytes32 */ function sender(bytes calldata _message) internal pure returns (bytes32) { return bytes32(_message[SENDER_OFFSET:DESTINATION_OFFSET]); } /** * @notice Returns the message sender as address. * @param _message ABI encoded Hyperlane message. * @return Sender of `_message` as address */ function senderAddress( bytes calldata _message ) internal pure returns (address) { return sender(_message).bytes32ToAddress(); } /** * @notice Returns the message destination domain. * @param _message ABI encoded Hyperlane message. * @return Destination domain of `_message` */ function destination( bytes calldata _message ) internal pure returns (uint32) { return uint32(bytes4(_message[DESTINATION_OFFSET:RECIPIENT_OFFSET])); } /** * @notice Returns the message recipient as bytes32. * @param _message ABI encoded Hyperlane message. * @return Recipient of `_message` as bytes32 */ function recipient( bytes calldata _message ) internal pure returns (bytes32) { return bytes32(_message[RECIPIENT_OFFSET:BODY_OFFSET]); } /** * @notice Returns the message recipient as address. * @param _message ABI encoded Hyperlane message. * @return Recipient of `_message` as address */ function recipientAddress( bytes calldata _message ) internal pure returns (address) { return recipient(_message).bytes32ToAddress(); } /** * @notice Returns the message body. * @param _message ABI encoded Hyperlane message. * @return Body of `_message` */ function body( bytes calldata _message ) internal pure returns (bytes calldata) { return bytes(_message[BODY_OFFSET:]); } }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v4.9.0) (utils/Address.sol) pragma solidity ^0.8.1; /** * @dev Collection of functions related to the address type */ library Address { /** * @dev Returns true if `account` is a contract. * * [IMPORTANT] * ==== * It is unsafe to assume that an address for which this function returns * false is an externally-owned account (EOA) and not a contract. * * Among others, `isContract` will return false for the following * types of addresses: * * - an externally-owned account * - a contract in construction * - an address where a contract will be created * - an address where a contract lived, but was destroyed * * Furthermore, `isContract` will also return true if the target contract within * the same transaction is already scheduled for destruction by `SELFDESTRUCT`, * which only has an effect at the end of a transaction. * ==== * * [IMPORTANT] * ==== * You shouldn't rely on `isContract` to protect against flash loan attacks! * * Preventing calls from contracts is highly discouraged. It breaks composability, breaks support for smart wallets * like Gnosis Safe, and does not provide security since it can be circumvented by calling from a contract * constructor. * ==== */ function isContract(address account) internal view returns (bool) { // This method relies on extcodesize/address.code.length, which returns 0 // for contracts in construction, since the code is only stored at the end // of the constructor execution. return account.code.length > 0; } /** * @dev Replacement for Solidity's `transfer`: sends `amount` wei to * `recipient`, forwarding all available gas and reverting on errors. * * https://eips.ethereum.org/EIPS/eip-1884[EIP1884] increases the gas cost * of certain opcodes, possibly making contracts go over the 2300 gas limit * imposed by `transfer`, making them unable to receive funds via * `transfer`. {sendValue} removes this limitation. * * https://consensys.net/diligence/blog/2019/09/stop-using-soliditys-transfer-now/[Learn more]. * * IMPORTANT: because control is transferred to `recipient`, care must be * taken to not create reentrancy vulnerabilities. Consider using * {ReentrancyGuard} or the * https://solidity.readthedocs.io/en/v0.8.0/security-considerations.html#use-the-checks-effects-interactions-pattern[checks-effects-interactions pattern]. */ function sendValue(address payable recipient, uint256 amount) internal { require(address(this).balance >= amount, "Address: insufficient balance"); (bool success, ) = recipient.call{value: amount}(""); require(success, "Address: unable to send value, recipient may have reverted"); } /** * @dev Performs a Solidity function call using a low level `call`. A * plain `call` is an unsafe replacement for a function call: use this * function instead. * * If `target` reverts with a revert reason, it is bubbled up by this * function (like regular Solidity function calls). * * Returns the raw returned data. To convert to the expected return value, * use https://solidity.readthedocs.io/en/latest/units-and-global-variables.html?highlight=abi.decode#abi-encoding-and-decoding-functions[`abi.decode`]. * * Requirements: * * - `target` must be a contract. * - calling `target` with `data` must not revert. * * _Available since v3.1._ */ function functionCall(address target, bytes memory data) internal returns (bytes memory) { return functionCallWithValue(target, data, 0, "Address: low-level call failed"); } /** * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`], but with * `errorMessage` as a fallback revert reason when `target` reverts. * * _Available since v3.1._ */ function functionCall( address target, bytes memory data, string memory errorMessage ) internal returns (bytes memory) { return functionCallWithValue(target, data, 0, errorMessage); } /** * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`], * but also transferring `value` wei to `target`. * * Requirements: * * - the calling contract must have an ETH balance of at least `value`. * - the called Solidity function must be `payable`. * * _Available since v3.1._ */ function functionCallWithValue(address target, bytes memory data, uint256 value) internal returns (bytes memory) { return functionCallWithValue(target, data, value, "Address: low-level call with value failed"); } /** * @dev Same as {xref-Address-functionCallWithValue-address-bytes-uint256-}[`functionCallWithValue`], but * with `errorMessage` as a fallback revert reason when `target` reverts. * * _Available since v3.1._ */ function functionCallWithValue( address target, bytes memory data, uint256 value, string memory errorMessage ) internal returns (bytes memory) { require(address(this).balance >= value, "Address: insufficient balance for call"); (bool success, bytes memory returndata) = target.call{value: value}(data); return verifyCallResultFromTarget(target, success, returndata, errorMessage); } /** * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`], * but performing a static call. * * _Available since v3.3._ */ function functionStaticCall(address target, bytes memory data) internal view returns (bytes memory) { return functionStaticCall(target, data, "Address: low-level static call failed"); } /** * @dev Same as {xref-Address-functionCall-address-bytes-string-}[`functionCall`], * but performing a static call. * * _Available since v3.3._ */ function functionStaticCall( address target, bytes memory data, string memory errorMessage ) internal view returns (bytes memory) { (bool success, bytes memory returndata) = target.staticcall(data); return verifyCallResultFromTarget(target, success, returndata, errorMessage); } /** * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`], * but performing a delegate call. * * _Available since v3.4._ */ function functionDelegateCall(address target, bytes memory data) internal returns (bytes memory) { return functionDelegateCall(target, data, "Address: low-level delegate call failed"); } /** * @dev Same as {xref-Address-functionCall-address-bytes-string-}[`functionCall`], * but performing a delegate call. * * _Available since v3.4._ */ function functionDelegateCall( address target, bytes memory data, string memory errorMessage ) internal returns (bytes memory) { (bool success, bytes memory returndata) = target.delegatecall(data); return verifyCallResultFromTarget(target, success, returndata, errorMessage); } /** * @dev Tool to verify that a low level call to smart-contract was successful, and revert (either by bubbling * the revert reason or using the provided one) in case of unsuccessful call or if target was not a contract. * * _Available since v4.8._ */ function verifyCallResultFromTarget( address target, bool success, bytes memory returndata, string memory errorMessage ) internal view returns (bytes memory) { if (success) { if (returndata.length == 0) { // only check isContract if the call was successful and the return data is empty // otherwise we already know that it was a contract require(isContract(target), "Address: call to non-contract"); } return returndata; } else { _revert(returndata, errorMessage); } } /** * @dev Tool to verify that a low level call was successful, and revert if it wasn't, either by bubbling the * revert reason or using the provided one. * * _Available since v4.3._ */ function verifyCallResult( bool success, bytes memory returndata, string memory errorMessage ) internal pure returns (bytes memory) { if (success) { return returndata; } else { _revert(returndata, errorMessage); } } function _revert(bytes memory returndata, string memory errorMessage) private pure { // Look for revert reason and bubble it up if present if (returndata.length > 0) { // The easiest way to bubble the revert reason is using memory via assembly /// @solidity memory-safe-assembly assembly { let returndata_size := mload(returndata) revert(add(32, returndata), returndata_size) } } else { revert(errorMessage); } } }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v4.9.0) (access/Ownable.sol) pragma solidity ^0.8.0; import "../utils/ContextUpgradeable.sol"; import "../proxy/utils/Initializable.sol"; /** * @dev Contract module which provides a basic access control mechanism, where * there is an account (an owner) that can be granted exclusive access to * specific functions. * * By default, the owner account will be the one that deploys the contract. This * can later be changed with {transferOwnership}. * * This module is used through inheritance. It will make available the modifier * `onlyOwner`, which can be applied to your functions to restrict their use to * the owner. */ abstract contract OwnableUpgradeable is Initializable, ContextUpgradeable { address private _owner; event OwnershipTransferred(address indexed previousOwner, address indexed newOwner); /** * @dev Initializes the contract setting the deployer as the initial owner. */ function __Ownable_init() internal onlyInitializing { __Ownable_init_unchained(); } function __Ownable_init_unchained() internal onlyInitializing { _transferOwnership(_msgSender()); } /** * @dev Throws if called by any account other than the owner. */ modifier onlyOwner() { _checkOwner(); _; } /** * @dev Returns the address of the current owner. */ function owner() public view virtual returns (address) { return _owner; } /** * @dev Throws if the sender is not the owner. */ function _checkOwner() internal view virtual { require(owner() == _msgSender(), "Ownable: caller is not the owner"); } /** * @dev Leaves the contract without owner. It will not be possible to call * `onlyOwner` functions. Can only be called by the current owner. * * NOTE: Renouncing ownership will leave the contract without an owner, * thereby disabling any functionality that is only available to the owner. */ function renounceOwnership() public virtual onlyOwner { _transferOwnership(address(0)); } /** * @dev Transfers ownership of the contract to a new account (`newOwner`). * Can only be called by the current owner. */ function transferOwnership(address newOwner) public virtual onlyOwner { require(newOwner != address(0), "Ownable: new owner is the zero address"); _transferOwnership(newOwner); } /** * @dev Transfers ownership of the contract to a new account (`newOwner`). * Internal function without access restriction. */ function _transferOwnership(address newOwner) internal virtual { address oldOwner = _owner; _owner = newOwner; emit OwnershipTransferred(oldOwner, newOwner); } /** * @dev This empty reserved space is put in place to allow future versions to add new * variables without shifting down storage in the inheritance chain. * See https://docs.openzeppelin.com/contracts/4.x/upgradeable#storage_gaps */ uint256[49] private __gap; }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v4.9.0) (token/ERC20/extensions/IERC20Permit.sol) pragma solidity ^0.8.0; /** * @dev Interface of the ERC20 Permit extension allowing approvals to be made via signatures, as defined in * https://eips.ethereum.org/EIPS/eip-2612[EIP-2612]. * * Adds the {permit} method, which can be used to change an account's ERC20 allowance (see {IERC20-allowance}) by * presenting a message signed by the account. By not relying on {IERC20-approve}, the token holder account doesn't * need to send a transaction, and thus is not required to hold Ether at all. */ interface IERC20Permit { /** * @dev Sets `value` as the allowance of `spender` over ``owner``'s tokens, * given ``owner``'s signed approval. * * IMPORTANT: The same issues {IERC20-approve} has related to transaction * ordering also apply here. * * Emits an {Approval} event. * * Requirements: * * - `spender` cannot be the zero address. * - `deadline` must be a timestamp in the future. * - `v`, `r` and `s` must be a valid `secp256k1` signature from `owner` * over the EIP712-formatted function arguments. * - the signature must use ``owner``'s current nonce (see {nonces}). * * For more information on the signature format, see the * https://eips.ethereum.org/EIPS/eip-2612#specification[relevant EIP * section]. */ function permit( address owner, address spender, uint256 value, uint256 deadline, uint8 v, bytes32 r, bytes32 s ) external; /** * @dev Returns the current nonce for `owner`. This value must be * included whenever a signature is generated for {permit}. * * Every successful call to {permit} increases ``owner``'s nonce by one. This * prevents a signature from being used multiple times. */ function nonces(address owner) external view returns (uint256); /** * @dev Returns the domain separator used in the encoding of the signature for {permit}, as defined by {EIP712}. */ // solhint-disable-next-line func-name-mixedcase function DOMAIN_SEPARATOR() external view returns (bytes32); }
// SPDX-License-Identifier: MIT OR Apache-2.0 pragma solidity >=0.6.11; // ============ Internal Imports ============ import {IMessageRecipient} from "../interfaces/IMessageRecipient.sol"; import {IPostDispatchHook} from "../interfaces/hooks/IPostDispatchHook.sol"; import {IInterchainSecurityModule} from "../interfaces/IInterchainSecurityModule.sol"; import {MailboxClient} from "./MailboxClient.sol"; import {EnumerableMapExtended} from "../libs/EnumerableMapExtended.sol"; // ============ External Imports ============ import {Strings} from "@openzeppelin/contracts/utils/Strings.sol"; abstract contract Router is MailboxClient, IMessageRecipient { using EnumerableMapExtended for EnumerableMapExtended.UintToBytes32Map; using Strings for uint32; // ============ Mutable Storage ============ EnumerableMapExtended.UintToBytes32Map internal _routers; uint256[48] private __GAP; // gap for upgrade safety constructor(address _mailbox) MailboxClient(_mailbox) {} // ============ External functions ============ function domains() external view returns (uint32[] memory) { return _routers.uint32Keys(); } /** * @notice Returns the address of the Router contract for the given domain * @param _domain The remote domain ID. * @dev Returns 0 address if no router is enrolled for the given domain * @return router The address of the Router contract for the given domain */ function routers(uint32 _domain) public view virtual returns (bytes32) { (, bytes32 _router) = _routers.tryGet(_domain); return _router; } /** * @notice Unregister the domain * @param _domain The domain of the remote Application Router */ function unenrollRemoteRouter(uint32 _domain) external virtual onlyOwner { _unenrollRemoteRouter(_domain); } /** * @notice Register the address of a Router contract for the same Application on a remote chain * @param _domain The domain of the remote Application Router * @param _router The address of the remote Application Router */ function enrollRemoteRouter( uint32 _domain, bytes32 _router ) external virtual onlyOwner { _enrollRemoteRouter(_domain, _router); } /** * @notice Batch version of `enrollRemoteRouter` * @param _domains The domains of the remote Application Routers * @param _addresses The addresses of the remote Application Routers */ function enrollRemoteRouters( uint32[] calldata _domains, bytes32[] calldata _addresses ) external virtual onlyOwner { require(_domains.length == _addresses.length, "!length"); uint256 length = _domains.length; for (uint256 i = 0; i < length; i += 1) { _enrollRemoteRouter(_domains[i], _addresses[i]); } } /** * @notice Batch version of `unenrollRemoteRouter` * @param _domains The domains of the remote Application Routers */ function unenrollRemoteRouters( uint32[] calldata _domains ) external virtual onlyOwner { uint256 length = _domains.length; for (uint256 i = 0; i < length; i += 1) { _unenrollRemoteRouter(_domains[i]); } } /** * @notice Handles an incoming message * @param _origin The origin domain * @param _sender The sender address * @param _message The message */ function handle( uint32 _origin, bytes32 _sender, bytes calldata _message ) external payable virtual override onlyMailbox { bytes32 _router = _mustHaveRemoteRouter(_origin); require(_router == _sender, "Enrolled router does not match sender"); _handle(_origin, _sender, _message); } // ============ Virtual functions ============ function _handle( uint32 _origin, bytes32 _sender, bytes calldata _message ) internal virtual; // ============ Internal functions ============ /** * @notice Set the router for a given domain * @param _domain The domain * @param _address The new router */ function _enrollRemoteRouter( uint32 _domain, bytes32 _address ) internal virtual { _routers.set(_domain, _address); } /** * @notice Remove the router for a given domain * @param _domain The domain */ function _unenrollRemoteRouter(uint32 _domain) internal virtual { require(_routers.remove(_domain), _domainNotFoundError(_domain)); } /** * @notice Return true if the given domain / router is the address of a remote Application Router * @param _domain The domain of the potential remote Application Router * @param _address The address of the potential remote Application Router */ function _isRemoteRouter( uint32 _domain, bytes32 _address ) internal view returns (bool) { return routers(_domain) == _address; } /** * @notice Assert that the given domain has a Application Router registered and return its address * @param _domain The domain of the chain for which to get the Application Router * @return _router The address of the remote Application Router on _domain */ function _mustHaveRemoteRouter( uint32 _domain ) internal view returns (bytes32) { (bool contained, bytes32 _router) = _routers.tryGet(_domain); require(contained, _domainNotFoundError(_domain)); return _router; } function _domainNotFoundError( uint32 _domain ) internal pure returns (string memory) { return string.concat( "No router enrolled for domain: ", _domain.toString() ); } function _Router_dispatch( uint32 _destinationDomain, uint256 _value, bytes memory _messageBody, bytes memory _hookMetadata, address _hook ) internal returns (bytes32) { bytes32 _router = _mustHaveRemoteRouter(_destinationDomain); return mailbox.dispatch{value: _value}( _destinationDomain, _router, _messageBody, _hookMetadata, IPostDispatchHook(_hook) ); } /** * DEPRECATED: Use `_Router_dispatch` instead * @dev For backward compatibility with v2 client contracts */ function _dispatch( uint32 _destinationDomain, bytes memory _messageBody ) internal returns (bytes32) { return _Router_dispatch( _destinationDomain, msg.value, _messageBody, "", address(hook) ); } function _Router_quoteDispatch( uint32 _destinationDomain, bytes memory _messageBody, bytes memory _hookMetadata, address _hook ) internal view returns (uint256) { bytes32 _router = _mustHaveRemoteRouter(_destinationDomain); return mailbox.quoteDispatch( _destinationDomain, _router, _messageBody, _hookMetadata, IPostDispatchHook(_hook) ); } /** * DEPRECATED: Use `_Router_quoteDispatch` instead * @dev For backward compatibility with v2 client contracts */ function _quoteDispatch( uint32 _destinationDomain, bytes memory _messageBody ) internal view returns (uint256) { return _Router_quoteDispatch( _destinationDomain, _messageBody, "", address(hook) ); } }
// SPDX-License-Identifier: MIT OR Apache-2.0 pragma solidity >=0.8.0; /*@@@@@@@ @@@@@@@@@ @@@@@@@@@ @@@@@@@@@ @@@@@@@@@ @@@@@@@@@ @@@@@@@@@ @@@@@@@@@ @@@@@@@@@@@@@@@@@@@@@@@@@ @@@@@ HYPERLANE @@@@@@@ @@@@@@@@@@@@@@@@@@@@@@@@@ @@@@@@@@@ @@@@@@@@@ @@@@@@@@@ @@@@@@@@@ @@@@@@@@@ @@@@@@@@@ @@@@@@@@@ @@@@@@@@*/ /** * Format of metadata: * * [0:2] variant * [2:34] msg.value * [34:66] Gas limit for message (IGP) * [66:86] Refund address for message (IGP) * [86:] Custom metadata */ library StandardHookMetadata { struct Metadata { uint16 variant; uint256 msgValue; uint256 gasLimit; address refundAddress; } uint8 private constant VARIANT_OFFSET = 0; uint8 private constant MSG_VALUE_OFFSET = 2; uint8 private constant GAS_LIMIT_OFFSET = 34; uint8 private constant REFUND_ADDRESS_OFFSET = 66; uint256 private constant MIN_METADATA_LENGTH = 86; uint16 public constant VARIANT = 1; /** * @notice Returns the variant of the metadata. * @param _metadata ABI encoded standard hook metadata. * @return variant of the metadata as uint8. */ function variant(bytes calldata _metadata) internal pure returns (uint16) { if (_metadata.length < VARIANT_OFFSET + 2) return 0; return uint16(bytes2(_metadata[VARIANT_OFFSET:VARIANT_OFFSET + 2])); } /** * @notice Returns the specified value for the message. * @param _metadata ABI encoded standard hook metadata. * @param _default Default fallback value. * @return Value for the message as uint256. */ function msgValue( bytes calldata _metadata, uint256 _default ) internal pure returns (uint256) { if (_metadata.length < MSG_VALUE_OFFSET + 32) return _default; return uint256(bytes32(_metadata[MSG_VALUE_OFFSET:MSG_VALUE_OFFSET + 32])); } /** * @notice Returns the specified gas limit for the message. * @param _metadata ABI encoded standard hook metadata. * @param _default Default fallback gas limit. * @return Gas limit for the message as uint256. */ function gasLimit( bytes calldata _metadata, uint256 _default ) internal pure returns (uint256) { if (_metadata.length < GAS_LIMIT_OFFSET + 32) return _default; return uint256(bytes32(_metadata[GAS_LIMIT_OFFSET:GAS_LIMIT_OFFSET + 32])); } /** * @notice Returns the specified refund address for the message. * @param _metadata ABI encoded standard hook metadata. * @param _default Default fallback refund address. * @return Refund address for the message as address. */ function refundAddress( bytes calldata _metadata, address _default ) internal pure returns (address) { if (_metadata.length < REFUND_ADDRESS_OFFSET + 20) return _default; return address( bytes20( _metadata[REFUND_ADDRESS_OFFSET:REFUND_ADDRESS_OFFSET + 20] ) ); } /** * @notice Returns any custom metadata. * @param _metadata ABI encoded standard hook metadata. * @return Custom metadata. */ function getCustomMetadata( bytes calldata _metadata ) internal pure returns (bytes calldata) { if (_metadata.length < MIN_METADATA_LENGTH) return _metadata[0:0]; return _metadata[MIN_METADATA_LENGTH:]; } /** * @notice Formats the specified gas limit and refund address into standard hook metadata. * @param _msgValue msg.value for the message. * @param _gasLimit Gas limit for the message. * @param _refundAddress Refund address for the message. * @param _customMetadata Additional metadata to include in the standard hook metadata. * @return ABI encoded standard hook metadata. */ function formatMetadata( uint256 _msgValue, uint256 _gasLimit, address _refundAddress, bytes memory _customMetadata ) internal pure returns (bytes memory) { return abi.encodePacked( VARIANT, _msgValue, _gasLimit, _refundAddress, _customMetadata ); } /** * @notice Formats the specified gas limit and refund address into standard hook metadata. * @param _msgValue msg.value for the message. * @return ABI encoded standard hook metadata. */ function overrideMsgValue( uint256 _msgValue ) internal view returns (bytes memory) { return formatMetadata(_msgValue, uint256(0), msg.sender, ""); } /** * @notice Formats the specified gas limit and refund address into standard hook metadata. * @param _gasLimit Gas limit for the message. * @return ABI encoded standard hook metadata. */ function overrideGasLimit( uint256 _gasLimit ) internal view returns (bytes memory) { return formatMetadata(uint256(0), _gasLimit, msg.sender, ""); } /** * @notice Formats the specified refund address into standard hook metadata. * @param _refundAddress Refund address for the message. * @return ABI encoded standard hook metadata. */ function overrideRefundAddress( address _refundAddress ) internal pure returns (bytes memory) { return formatMetadata(uint256(0), uint256(0), _refundAddress, ""); } }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts v4.4.1 (utils/Context.sol) pragma solidity ^0.8.0; import "../proxy/utils/Initializable.sol"; /** * @dev Provides information about the current execution context, including the * sender of the transaction and its data. While these are generally available * via msg.sender and msg.data, they should not be accessed in such a direct * manner, since when dealing with meta-transactions the account sending and * paying for execution may not be the actual sender (as far as an application * is concerned). * * This contract is only required for intermediate, library-like contracts. */ abstract contract ContextUpgradeable is Initializable { function __Context_init() internal onlyInitializing { } function __Context_init_unchained() internal onlyInitializing { } function _msgSender() internal view virtual returns (address) { return msg.sender; } function _msgData() internal view virtual returns (bytes calldata) { return msg.data; } /** * @dev This empty reserved space is put in place to allow future versions to add new * variables without shifting down storage in the inheritance chain. * See https://docs.openzeppelin.com/contracts/4.x/upgradeable#storage_gaps */ uint256[50] private __gap; }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v4.9.0) (proxy/utils/Initializable.sol) pragma solidity ^0.8.2; import "../../utils/AddressUpgradeable.sol"; /** * @dev This is a base contract to aid in writing upgradeable contracts, or any kind of contract that will be deployed * behind a proxy. Since proxied contracts do not make use of a constructor, it's common to move constructor logic to an * external initializer function, usually called `initialize`. It then becomes necessary to protect this initializer * function so it can only be called once. The {initializer} modifier provided by this contract will have this effect. * * The initialization functions use a version number. Once a version number is used, it is consumed and cannot be * reused. This mechanism prevents re-execution of each "step" but allows the creation of new initialization steps in * case an upgrade adds a module that needs to be initialized. * * For example: * * [.hljs-theme-light.nopadding] * ```solidity * contract MyToken is ERC20Upgradeable { * function initialize() initializer public { * __ERC20_init("MyToken", "MTK"); * } * } * * contract MyTokenV2 is MyToken, ERC20PermitUpgradeable { * function initializeV2() reinitializer(2) public { * __ERC20Permit_init("MyToken"); * } * } * ``` * * TIP: To avoid leaving the proxy in an uninitialized state, the initializer function should be called as early as * possible by providing the encoded function call as the `_data` argument to {ERC1967Proxy-constructor}. * * CAUTION: When used with inheritance, manual care must be taken to not invoke a parent initializer twice, or to ensure * that all initializers are idempotent. This is not verified automatically as constructors are by Solidity. * * [CAUTION] * ==== * Avoid leaving a contract uninitialized. * * An uninitialized contract can be taken over by an attacker. This applies to both a proxy and its implementation * contract, which may impact the proxy. To prevent the implementation contract from being used, you should invoke * the {_disableInitializers} function in the constructor to automatically lock it when it is deployed: * * [.hljs-theme-light.nopadding] * ``` * /// @custom:oz-upgrades-unsafe-allow constructor * constructor() { * _disableInitializers(); * } * ``` * ==== */ abstract contract Initializable { /** * @dev Indicates that the contract has been initialized. * @custom:oz-retyped-from bool */ uint8 private _initialized; /** * @dev Indicates that the contract is in the process of being initialized. */ bool private _initializing; /** * @dev Triggered when the contract has been initialized or reinitialized. */ event Initialized(uint8 version); /** * @dev A modifier that defines a protected initializer function that can be invoked at most once. In its scope, * `onlyInitializing` functions can be used to initialize parent contracts. * * Similar to `reinitializer(1)`, except that functions marked with `initializer` can be nested in the context of a * constructor. * * Emits an {Initialized} event. */ modifier initializer() { bool isTopLevelCall = !_initializing; require( (isTopLevelCall && _initialized < 1) || (!AddressUpgradeable.isContract(address(this)) && _initialized == 1), "Initializable: contract is already initialized" ); _initialized = 1; if (isTopLevelCall) { _initializing = true; } _; if (isTopLevelCall) { _initializing = false; emit Initialized(1); } } /** * @dev A modifier that defines a protected reinitializer function that can be invoked at most once, and only if the * contract hasn't been initialized to a greater version before. In its scope, `onlyInitializing` functions can be * used to initialize parent contracts. * * A reinitializer may be used after the original initialization step. This is essential to configure modules that * are added through upgrades and that require initialization. * * When `version` is 1, this modifier is similar to `initializer`, except that functions marked with `reinitializer` * cannot be nested. If one is invoked in the context of another, execution will revert. * * Note that versions can jump in increments greater than 1; this implies that if multiple reinitializers coexist in * a contract, executing them in the right order is up to the developer or operator. * * WARNING: setting the version to 255 will prevent any future reinitialization. * * Emits an {Initialized} event. */ modifier reinitializer(uint8 version) { require(!_initializing && _initialized < version, "Initializable: contract is already initialized"); _initialized = version; _initializing = true; _; _initializing = false; emit Initialized(version); } /** * @dev Modifier to protect an initialization function so that it can only be invoked by functions with the * {initializer} and {reinitializer} modifiers, directly or indirectly. */ modifier onlyInitializing() { require(_initializing, "Initializable: contract is not initializing"); _; } /** * @dev Locks the contract, preventing any future reinitialization. This cannot be part of an initializer call. * Calling this in the constructor of a contract will prevent that contract from being initialized or reinitialized * to any version. It is recommended to use this to lock implementation contracts that are designed to be called * through proxies. * * Emits an {Initialized} event the first time it is successfully executed. */ function _disableInitializers() internal virtual { require(!_initializing, "Initializable: contract is initializing"); if (_initialized != type(uint8).max) { _initialized = type(uint8).max; emit Initialized(type(uint8).max); } } /** * @dev Returns the highest version that has been initialized. See {reinitializer}. */ function _getInitializedVersion() internal view returns (uint8) { return _initialized; } /** * @dev Returns `true` if the contract is currently initializing. See {onlyInitializing}. */ function _isInitializing() internal view returns (bool) { return _initializing; } }
// SPDX-License-Identifier: MIT OR Apache-2.0 pragma solidity >=0.6.11; interface IMessageRecipient { function handle( uint32 _origin, bytes32 _sender, bytes calldata _message ) external payable; }
// SPDX-License-Identifier: MIT OR Apache-2.0 pragma solidity >=0.6.11; // ============ External Imports ============ import "@openzeppelin/contracts/utils/structs/EnumerableMap.sol"; import "@openzeppelin/contracts/utils/structs/EnumerableSet.sol"; // extends EnumerableMap with uint256 => bytes32 type // modelled after https://github.com/OpenZeppelin/openzeppelin-contracts/blob/v4.8.0/contracts/utils/structs/EnumerableMap.sol library EnumerableMapExtended { using EnumerableMap for EnumerableMap.Bytes32ToBytes32Map; using EnumerableSet for EnumerableSet.Bytes32Set; struct UintToBytes32Map { EnumerableMap.Bytes32ToBytes32Map _inner; } // ============ Library Functions ============ function keys( UintToBytes32Map storage map ) internal view returns (uint256[] memory _keys) { uint256 _length = map._inner.length(); _keys = new uint256[](_length); for (uint256 i = 0; i < _length; i++) { _keys[i] = uint256(map._inner._keys.at(i)); } } function uint32Keys( UintToBytes32Map storage map ) internal view returns (uint32[] memory _keys) { uint256[] memory uint256keys = keys(map); _keys = new uint32[](uint256keys.length); for (uint256 i = 0; i < uint256keys.length; i++) { _keys[i] = uint32(uint256keys[i]); } } function set( UintToBytes32Map storage map, uint256 key, bytes32 value ) internal { map._inner.set(bytes32(key), value); } function get( UintToBytes32Map storage map, uint256 key ) internal view returns (bytes32) { return map._inner.get(bytes32(key)); } function tryGet( UintToBytes32Map storage map, uint256 key ) internal view returns (bool, bytes32) { return map._inner.tryGet(bytes32(key)); } function remove( UintToBytes32Map storage map, uint256 key ) internal returns (bool) { return map._inner.remove(bytes32(key)); } function contains( UintToBytes32Map storage map, uint256 key ) internal view returns (bool) { return map._inner.contains(bytes32(key)); } function length( UintToBytes32Map storage map ) internal view returns (uint256) { return map._inner.length(); } function at( UintToBytes32Map storage map, uint256 index ) internal view returns (uint256, bytes32) { (bytes32 key, bytes32 value) = map._inner.at(index); return (uint256(key), value); } }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v4.9.0) (utils/Strings.sol) pragma solidity ^0.8.0; import "./math/Math.sol"; import "./math/SignedMath.sol"; /** * @dev String operations. */ library Strings { bytes16 private constant _SYMBOLS = "0123456789abcdef"; uint8 private constant _ADDRESS_LENGTH = 20; /** * @dev Converts a `uint256` to its ASCII `string` decimal representation. */ function toString(uint256 value) internal pure returns (string memory) { unchecked { uint256 length = Math.log10(value) + 1; string memory buffer = new string(length); uint256 ptr; /// @solidity memory-safe-assembly assembly { ptr := add(buffer, add(32, length)) } while (true) { ptr--; /// @solidity memory-safe-assembly assembly { mstore8(ptr, byte(mod(value, 10), _SYMBOLS)) } value /= 10; if (value == 0) break; } return buffer; } } /** * @dev Converts a `int256` to its ASCII `string` decimal representation. */ function toString(int256 value) internal pure returns (string memory) { return string(abi.encodePacked(value < 0 ? "-" : "", toString(SignedMath.abs(value)))); } /** * @dev Converts a `uint256` to its ASCII `string` hexadecimal representation. */ function toHexString(uint256 value) internal pure returns (string memory) { unchecked { return toHexString(value, Math.log256(value) + 1); } } /** * @dev Converts a `uint256` to its ASCII `string` hexadecimal representation with fixed length. */ function toHexString(uint256 value, uint256 length) internal pure returns (string memory) { bytes memory buffer = new bytes(2 * length + 2); buffer[0] = "0"; buffer[1] = "x"; for (uint256 i = 2 * length + 1; i > 1; --i) { buffer[i] = _SYMBOLS[value & 0xf]; value >>= 4; } require(value == 0, "Strings: hex length insufficient"); return string(buffer); } /** * @dev Converts an `address` with fixed length of 20 bytes to its not checksummed ASCII `string` hexadecimal representation. */ function toHexString(address addr) internal pure returns (string memory) { return toHexString(uint256(uint160(addr)), _ADDRESS_LENGTH); } /** * @dev Returns true if the two strings are equal. */ function equal(string memory a, string memory b) internal pure returns (bool) { return keccak256(bytes(a)) == keccak256(bytes(b)); } }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v4.9.0) (utils/Address.sol) pragma solidity ^0.8.1; /** * @dev Collection of functions related to the address type */ library AddressUpgradeable { /** * @dev Returns true if `account` is a contract. * * [IMPORTANT] * ==== * It is unsafe to assume that an address for which this function returns * false is an externally-owned account (EOA) and not a contract. * * Among others, `isContract` will return false for the following * types of addresses: * * - an externally-owned account * - a contract in construction * - an address where a contract will be created * - an address where a contract lived, but was destroyed * * Furthermore, `isContract` will also return true if the target contract within * the same transaction is already scheduled for destruction by `SELFDESTRUCT`, * which only has an effect at the end of a transaction. * ==== * * [IMPORTANT] * ==== * You shouldn't rely on `isContract` to protect against flash loan attacks! * * Preventing calls from contracts is highly discouraged. It breaks composability, breaks support for smart wallets * like Gnosis Safe, and does not provide security since it can be circumvented by calling from a contract * constructor. * ==== */ function isContract(address account) internal view returns (bool) { // This method relies on extcodesize/address.code.length, which returns 0 // for contracts in construction, since the code is only stored at the end // of the constructor execution. return account.code.length > 0; } /** * @dev Replacement for Solidity's `transfer`: sends `amount` wei to * `recipient`, forwarding all available gas and reverting on errors. * * https://eips.ethereum.org/EIPS/eip-1884[EIP1884] increases the gas cost * of certain opcodes, possibly making contracts go over the 2300 gas limit * imposed by `transfer`, making them unable to receive funds via * `transfer`. {sendValue} removes this limitation. * * https://consensys.net/diligence/blog/2019/09/stop-using-soliditys-transfer-now/[Learn more]. * * IMPORTANT: because control is transferred to `recipient`, care must be * taken to not create reentrancy vulnerabilities. Consider using * {ReentrancyGuard} or the * https://solidity.readthedocs.io/en/v0.8.0/security-considerations.html#use-the-checks-effects-interactions-pattern[checks-effects-interactions pattern]. */ function sendValue(address payable recipient, uint256 amount) internal { require(address(this).balance >= amount, "Address: insufficient balance"); (bool success, ) = recipient.call{value: amount}(""); require(success, "Address: unable to send value, recipient may have reverted"); } /** * @dev Performs a Solidity function call using a low level `call`. A * plain `call` is an unsafe replacement for a function call: use this * function instead. * * If `target` reverts with a revert reason, it is bubbled up by this * function (like regular Solidity function calls). * * Returns the raw returned data. To convert to the expected return value, * use https://solidity.readthedocs.io/en/latest/units-and-global-variables.html?highlight=abi.decode#abi-encoding-and-decoding-functions[`abi.decode`]. * * Requirements: * * - `target` must be a contract. * - calling `target` with `data` must not revert. * * _Available since v3.1._ */ function functionCall(address target, bytes memory data) internal returns (bytes memory) { return functionCallWithValue(target, data, 0, "Address: low-level call failed"); } /** * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`], but with * `errorMessage` as a fallback revert reason when `target` reverts. * * _Available since v3.1._ */ function functionCall( address target, bytes memory data, string memory errorMessage ) internal returns (bytes memory) { return functionCallWithValue(target, data, 0, errorMessage); } /** * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`], * but also transferring `value` wei to `target`. * * Requirements: * * - the calling contract must have an ETH balance of at least `value`. * - the called Solidity function must be `payable`. * * _Available since v3.1._ */ function functionCallWithValue(address target, bytes memory data, uint256 value) internal returns (bytes memory) { return functionCallWithValue(target, data, value, "Address: low-level call with value failed"); } /** * @dev Same as {xref-Address-functionCallWithValue-address-bytes-uint256-}[`functionCallWithValue`], but * with `errorMessage` as a fallback revert reason when `target` reverts. * * _Available since v3.1._ */ function functionCallWithValue( address target, bytes memory data, uint256 value, string memory errorMessage ) internal returns (bytes memory) { require(address(this).balance >= value, "Address: insufficient balance for call"); (bool success, bytes memory returndata) = target.call{value: value}(data); return verifyCallResultFromTarget(target, success, returndata, errorMessage); } /** * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`], * but performing a static call. * * _Available since v3.3._ */ function functionStaticCall(address target, bytes memory data) internal view returns (bytes memory) { return functionStaticCall(target, data, "Address: low-level static call failed"); } /** * @dev Same as {xref-Address-functionCall-address-bytes-string-}[`functionCall`], * but performing a static call. * * _Available since v3.3._ */ function functionStaticCall( address target, bytes memory data, string memory errorMessage ) internal view returns (bytes memory) { (bool success, bytes memory returndata) = target.staticcall(data); return verifyCallResultFromTarget(target, success, returndata, errorMessage); } /** * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`], * but performing a delegate call. * * _Available since v3.4._ */ function functionDelegateCall(address target, bytes memory data) internal returns (bytes memory) { return functionDelegateCall(target, data, "Address: low-level delegate call failed"); } /** * @dev Same as {xref-Address-functionCall-address-bytes-string-}[`functionCall`], * but performing a delegate call. * * _Available since v3.4._ */ function functionDelegateCall( address target, bytes memory data, string memory errorMessage ) internal returns (bytes memory) { (bool success, bytes memory returndata) = target.delegatecall(data); return verifyCallResultFromTarget(target, success, returndata, errorMessage); } /** * @dev Tool to verify that a low level call to smart-contract was successful, and revert (either by bubbling * the revert reason or using the provided one) in case of unsuccessful call or if target was not a contract. * * _Available since v4.8._ */ function verifyCallResultFromTarget( address target, bool success, bytes memory returndata, string memory errorMessage ) internal view returns (bytes memory) { if (success) { if (returndata.length == 0) { // only check isContract if the call was successful and the return data is empty // otherwise we already know that it was a contract require(isContract(target), "Address: call to non-contract"); } return returndata; } else { _revert(returndata, errorMessage); } } /** * @dev Tool to verify that a low level call was successful, and revert if it wasn't, either by bubbling the * revert reason or using the provided one. * * _Available since v4.3._ */ function verifyCallResult( bool success, bytes memory returndata, string memory errorMessage ) internal pure returns (bytes memory) { if (success) { return returndata; } else { _revert(returndata, errorMessage); } } function _revert(bytes memory returndata, string memory errorMessage) private pure { // Look for revert reason and bubble it up if present if (returndata.length > 0) { // The easiest way to bubble the revert reason is using memory via assembly /// @solidity memory-safe-assembly assembly { let returndata_size := mload(returndata) revert(add(32, returndata), returndata_size) } } else { revert(errorMessage); } } }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v4.9.0) (utils/structs/EnumerableMap.sol) // This file was procedurally generated from scripts/generate/templates/EnumerableMap.js. pragma solidity ^0.8.0; import "./EnumerableSet.sol"; /** * @dev Library for managing an enumerable variant of Solidity's * https://solidity.readthedocs.io/en/latest/types.html#mapping-types[`mapping`] * type. * * Maps have the following properties: * * - Entries are added, removed, and checked for existence in constant time * (O(1)). * - Entries are enumerated in O(n). No guarantees are made on the ordering. * * ```solidity * contract Example { * // Add the library methods * using EnumerableMap for EnumerableMap.UintToAddressMap; * * // Declare a set state variable * EnumerableMap.UintToAddressMap private myMap; * } * ``` * * The following map types are supported: * * - `uint256 -> address` (`UintToAddressMap`) since v3.0.0 * - `address -> uint256` (`AddressToUintMap`) since v4.6.0 * - `bytes32 -> bytes32` (`Bytes32ToBytes32Map`) since v4.6.0 * - `uint256 -> uint256` (`UintToUintMap`) since v4.7.0 * - `bytes32 -> uint256` (`Bytes32ToUintMap`) since v4.7.0 * * [WARNING] * ==== * Trying to delete such a structure from storage will likely result in data corruption, rendering the structure * unusable. * See https://github.com/ethereum/solidity/pull/11843[ethereum/solidity#11843] for more info. * * In order to clean an EnumerableMap, you can either remove all elements one by one or create a fresh instance using an * array of EnumerableMap. * ==== */ library EnumerableMap { using EnumerableSet for EnumerableSet.Bytes32Set; // To implement this library for multiple types with as little code // repetition as possible, we write it in terms of a generic Map type with // bytes32 keys and values. // The Map implementation uses private functions, and user-facing // implementations (such as Uint256ToAddressMap) are just wrappers around // the underlying Map. // This means that we can only create new EnumerableMaps for types that fit // in bytes32. struct Bytes32ToBytes32Map { // Storage of keys EnumerableSet.Bytes32Set _keys; mapping(bytes32 => bytes32) _values; } /** * @dev Adds a key-value pair to a map, or updates the value for an existing * key. O(1). * * Returns true if the key was added to the map, that is if it was not * already present. */ function set(Bytes32ToBytes32Map storage map, bytes32 key, bytes32 value) internal returns (bool) { map._values[key] = value; return map._keys.add(key); } /** * @dev Removes a key-value pair from a map. O(1). * * Returns true if the key was removed from the map, that is if it was present. */ function remove(Bytes32ToBytes32Map storage map, bytes32 key) internal returns (bool) { delete map._values[key]; return map._keys.remove(key); } /** * @dev Returns true if the key is in the map. O(1). */ function contains(Bytes32ToBytes32Map storage map, bytes32 key) internal view returns (bool) { return map._keys.contains(key); } /** * @dev Returns the number of key-value pairs in the map. O(1). */ function length(Bytes32ToBytes32Map storage map) internal view returns (uint256) { return map._keys.length(); } /** * @dev Returns the key-value pair stored at position `index` in the map. O(1). * * Note that there are no guarantees on the ordering of entries inside the * array, and it may change when more entries are added or removed. * * Requirements: * * - `index` must be strictly less than {length}. */ function at(Bytes32ToBytes32Map storage map, uint256 index) internal view returns (bytes32, bytes32) { bytes32 key = map._keys.at(index); return (key, map._values[key]); } /** * @dev Tries to returns the value associated with `key`. O(1). * Does not revert if `key` is not in the map. */ function tryGet(Bytes32ToBytes32Map storage map, bytes32 key) internal view returns (bool, bytes32) { bytes32 value = map._values[key]; if (value == bytes32(0)) { return (contains(map, key), bytes32(0)); } else { return (true, value); } } /** * @dev Returns the value associated with `key`. O(1). * * Requirements: * * - `key` must be in the map. */ function get(Bytes32ToBytes32Map storage map, bytes32 key) internal view returns (bytes32) { bytes32 value = map._values[key]; require(value != 0 || contains(map, key), "EnumerableMap: nonexistent key"); return value; } /** * @dev Same as {get}, with a custom error message when `key` is not in the map. * * CAUTION: This function is deprecated because it requires allocating memory for the error * message unnecessarily. For custom revert reasons use {tryGet}. */ function get( Bytes32ToBytes32Map storage map, bytes32 key, string memory errorMessage ) internal view returns (bytes32) { bytes32 value = map._values[key]; require(value != 0 || contains(map, key), errorMessage); return value; } /** * @dev Return the an array containing all the keys * * WARNING: This operation will copy the entire storage to memory, which can be quite expensive. This is designed * to mostly be used by view accessors that are queried without any gas fees. Developers should keep in mind that * this function has an unbounded cost, and using it as part of a state-changing function may render the function * uncallable if the map grows to a point where copying to memory consumes too much gas to fit in a block. */ function keys(Bytes32ToBytes32Map storage map) internal view returns (bytes32[] memory) { return map._keys.values(); } // UintToUintMap struct UintToUintMap { Bytes32ToBytes32Map _inner; } /** * @dev Adds a key-value pair to a map, or updates the value for an existing * key. O(1). * * Returns true if the key was added to the map, that is if it was not * already present. */ function set(UintToUintMap storage map, uint256 key, uint256 value) internal returns (bool) { return set(map._inner, bytes32(key), bytes32(value)); } /** * @dev Removes a value from a map. O(1). * * Returns true if the key was removed from the map, that is if it was present. */ function remove(UintToUintMap storage map, uint256 key) internal returns (bool) { return remove(map._inner, bytes32(key)); } /** * @dev Returns true if the key is in the map. O(1). */ function contains(UintToUintMap storage map, uint256 key) internal view returns (bool) { return contains(map._inner, bytes32(key)); } /** * @dev Returns the number of elements in the map. O(1). */ function length(UintToUintMap storage map) internal view returns (uint256) { return length(map._inner); } /** * @dev Returns the element stored at position `index` in the map. O(1). * Note that there are no guarantees on the ordering of values inside the * array, and it may change when more values are added or removed. * * Requirements: * * - `index` must be strictly less than {length}. */ function at(UintToUintMap storage map, uint256 index) internal view returns (uint256, uint256) { (bytes32 key, bytes32 value) = at(map._inner, index); return (uint256(key), uint256(value)); } /** * @dev Tries to returns the value associated with `key`. O(1). * Does not revert if `key` is not in the map. */ function tryGet(UintToUintMap storage map, uint256 key) internal view returns (bool, uint256) { (bool success, bytes32 value) = tryGet(map._inner, bytes32(key)); return (success, uint256(value)); } /** * @dev Returns the value associated with `key`. O(1). * * Requirements: * * - `key` must be in the map. */ function get(UintToUintMap storage map, uint256 key) internal view returns (uint256) { return uint256(get(map._inner, bytes32(key))); } /** * @dev Same as {get}, with a custom error message when `key` is not in the map. * * CAUTION: This function is deprecated because it requires allocating memory for the error * message unnecessarily. For custom revert reasons use {tryGet}. */ function get(UintToUintMap storage map, uint256 key, string memory errorMessage) internal view returns (uint256) { return uint256(get(map._inner, bytes32(key), errorMessage)); } /** * @dev Return the an array containing all the keys * * WARNING: This operation will copy the entire storage to memory, which can be quite expensive. This is designed * to mostly be used by view accessors that are queried without any gas fees. Developers should keep in mind that * this function has an unbounded cost, and using it as part of a state-changing function may render the function * uncallable if the map grows to a point where copying to memory consumes too much gas to fit in a block. */ function keys(UintToUintMap storage map) internal view returns (uint256[] memory) { bytes32[] memory store = keys(map._inner); uint256[] memory result; /// @solidity memory-safe-assembly assembly { result := store } return result; } // UintToAddressMap struct UintToAddressMap { Bytes32ToBytes32Map _inner; } /** * @dev Adds a key-value pair to a map, or updates the value for an existing * key. O(1). * * Returns true if the key was added to the map, that is if it was not * already present. */ function set(UintToAddressMap storage map, uint256 key, address value) internal returns (bool) { return set(map._inner, bytes32(key), bytes32(uint256(uint160(value)))); } /** * @dev Removes a value from a map. O(1). * * Returns true if the key was removed from the map, that is if it was present. */ function remove(UintToAddressMap storage map, uint256 key) internal returns (bool) { return remove(map._inner, bytes32(key)); } /** * @dev Returns true if the key is in the map. O(1). */ function contains(UintToAddressMap storage map, uint256 key) internal view returns (bool) { return contains(map._inner, bytes32(key)); } /** * @dev Returns the number of elements in the map. O(1). */ function length(UintToAddressMap storage map) internal view returns (uint256) { return length(map._inner); } /** * @dev Returns the element stored at position `index` in the map. O(1). * Note that there are no guarantees on the ordering of values inside the * array, and it may change when more values are added or removed. * * Requirements: * * - `index` must be strictly less than {length}. */ function at(UintToAddressMap storage map, uint256 index) internal view returns (uint256, address) { (bytes32 key, bytes32 value) = at(map._inner, index); return (uint256(key), address(uint160(uint256(value)))); } /** * @dev Tries to returns the value associated with `key`. O(1). * Does not revert if `key` is not in the map. */ function tryGet(UintToAddressMap storage map, uint256 key) internal view returns (bool, address) { (bool success, bytes32 value) = tryGet(map._inner, bytes32(key)); return (success, address(uint160(uint256(value)))); } /** * @dev Returns the value associated with `key`. O(1). * * Requirements: * * - `key` must be in the map. */ function get(UintToAddressMap storage map, uint256 key) internal view returns (address) { return address(uint160(uint256(get(map._inner, bytes32(key))))); } /** * @dev Same as {get}, with a custom error message when `key` is not in the map. * * CAUTION: This function is deprecated because it requires allocating memory for the error * message unnecessarily. For custom revert reasons use {tryGet}. */ function get( UintToAddressMap storage map, uint256 key, string memory errorMessage ) internal view returns (address) { return address(uint160(uint256(get(map._inner, bytes32(key), errorMessage)))); } /** * @dev Return the an array containing all the keys * * WARNING: This operation will copy the entire storage to memory, which can be quite expensive. This is designed * to mostly be used by view accessors that are queried without any gas fees. Developers should keep in mind that * this function has an unbounded cost, and using it as part of a state-changing function may render the function * uncallable if the map grows to a point where copying to memory consumes too much gas to fit in a block. */ function keys(UintToAddressMap storage map) internal view returns (uint256[] memory) { bytes32[] memory store = keys(map._inner); uint256[] memory result; /// @solidity memory-safe-assembly assembly { result := store } return result; } // AddressToUintMap struct AddressToUintMap { Bytes32ToBytes32Map _inner; } /** * @dev Adds a key-value pair to a map, or updates the value for an existing * key. O(1). * * Returns true if the key was added to the map, that is if it was not * already present. */ function set(AddressToUintMap storage map, address key, uint256 value) internal returns (bool) { return set(map._inner, bytes32(uint256(uint160(key))), bytes32(value)); } /** * @dev Removes a value from a map. O(1). * * Returns true if the key was removed from the map, that is if it was present. */ function remove(AddressToUintMap storage map, address key) internal returns (bool) { return remove(map._inner, bytes32(uint256(uint160(key)))); } /** * @dev Returns true if the key is in the map. O(1). */ function contains(AddressToUintMap storage map, address key) internal view returns (bool) { return contains(map._inner, bytes32(uint256(uint160(key)))); } /** * @dev Returns the number of elements in the map. O(1). */ function length(AddressToUintMap storage map) internal view returns (uint256) { return length(map._inner); } /** * @dev Returns the element stored at position `index` in the map. O(1). * Note that there are no guarantees on the ordering of values inside the * array, and it may change when more values are added or removed. * * Requirements: * * - `index` must be strictly less than {length}. */ function at(AddressToUintMap storage map, uint256 index) internal view returns (address, uint256) { (bytes32 key, bytes32 value) = at(map._inner, index); return (address(uint160(uint256(key))), uint256(value)); } /** * @dev Tries to returns the value associated with `key`. O(1). * Does not revert if `key` is not in the map. */ function tryGet(AddressToUintMap storage map, address key) internal view returns (bool, uint256) { (bool success, bytes32 value) = tryGet(map._inner, bytes32(uint256(uint160(key)))); return (success, uint256(value)); } /** * @dev Returns the value associated with `key`. O(1). * * Requirements: * * - `key` must be in the map. */ function get(AddressToUintMap storage map, address key) internal view returns (uint256) { return uint256(get(map._inner, bytes32(uint256(uint160(key))))); } /** * @dev Same as {get}, with a custom error message when `key` is not in the map. * * CAUTION: This function is deprecated because it requires allocating memory for the error * message unnecessarily. For custom revert reasons use {tryGet}. */ function get( AddressToUintMap storage map, address key, string memory errorMessage ) internal view returns (uint256) { return uint256(get(map._inner, bytes32(uint256(uint160(key))), errorMessage)); } /** * @dev Return the an array containing all the keys * * WARNING: This operation will copy the entire storage to memory, which can be quite expensive. This is designed * to mostly be used by view accessors that are queried without any gas fees. Developers should keep in mind that * this function has an unbounded cost, and using it as part of a state-changing function may render the function * uncallable if the map grows to a point where copying to memory consumes too much gas to fit in a block. */ function keys(AddressToUintMap storage map) internal view returns (address[] memory) { bytes32[] memory store = keys(map._inner); address[] memory result; /// @solidity memory-safe-assembly assembly { result := store } return result; } // Bytes32ToUintMap struct Bytes32ToUintMap { Bytes32ToBytes32Map _inner; } /** * @dev Adds a key-value pair to a map, or updates the value for an existing * key. O(1). * * Returns true if the key was added to the map, that is if it was not * already present. */ function set(Bytes32ToUintMap storage map, bytes32 key, uint256 value) internal returns (bool) { return set(map._inner, key, bytes32(value)); } /** * @dev Removes a value from a map. O(1). * * Returns true if the key was removed from the map, that is if it was present. */ function remove(Bytes32ToUintMap storage map, bytes32 key) internal returns (bool) { return remove(map._inner, key); } /** * @dev Returns true if the key is in the map. O(1). */ function contains(Bytes32ToUintMap storage map, bytes32 key) internal view returns (bool) { return contains(map._inner, key); } /** * @dev Returns the number of elements in the map. O(1). */ function length(Bytes32ToUintMap storage map) internal view returns (uint256) { return length(map._inner); } /** * @dev Returns the element stored at position `index` in the map. O(1). * Note that there are no guarantees on the ordering of values inside the * array, and it may change when more values are added or removed. * * Requirements: * * - `index` must be strictly less than {length}. */ function at(Bytes32ToUintMap storage map, uint256 index) internal view returns (bytes32, uint256) { (bytes32 key, bytes32 value) = at(map._inner, index); return (key, uint256(value)); } /** * @dev Tries to returns the value associated with `key`. O(1). * Does not revert if `key` is not in the map. */ function tryGet(Bytes32ToUintMap storage map, bytes32 key) internal view returns (bool, uint256) { (bool success, bytes32 value) = tryGet(map._inner, key); return (success, uint256(value)); } /** * @dev Returns the value associated with `key`. O(1). * * Requirements: * * - `key` must be in the map. */ function get(Bytes32ToUintMap storage map, bytes32 key) internal view returns (uint256) { return uint256(get(map._inner, key)); } /** * @dev Same as {get}, with a custom error message when `key` is not in the map. * * CAUTION: This function is deprecated because it requires allocating memory for the error * message unnecessarily. For custom revert reasons use {tryGet}. */ function get( Bytes32ToUintMap storage map, bytes32 key, string memory errorMessage ) internal view returns (uint256) { return uint256(get(map._inner, key, errorMessage)); } /** * @dev Return the an array containing all the keys * * WARNING: This operation will copy the entire storage to memory, which can be quite expensive. This is designed * to mostly be used by view accessors that are queried without any gas fees. Developers should keep in mind that * this function has an unbounded cost, and using it as part of a state-changing function may render the function * uncallable if the map grows to a point where copying to memory consumes too much gas to fit in a block. */ function keys(Bytes32ToUintMap storage map) internal view returns (bytes32[] memory) { bytes32[] memory store = keys(map._inner); bytes32[] memory result; /// @solidity memory-safe-assembly assembly { result := store } return result; } }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v4.9.0) (utils/structs/EnumerableSet.sol) // This file was procedurally generated from scripts/generate/templates/EnumerableSet.js. pragma solidity ^0.8.0; /** * @dev Library for managing * https://en.wikipedia.org/wiki/Set_(abstract_data_type)[sets] of primitive * types. * * Sets have the following properties: * * - Elements are added, removed, and checked for existence in constant time * (O(1)). * - Elements are enumerated in O(n). No guarantees are made on the ordering. * * ```solidity * contract Example { * // Add the library methods * using EnumerableSet for EnumerableSet.AddressSet; * * // Declare a set state variable * EnumerableSet.AddressSet private mySet; * } * ``` * * As of v3.3.0, sets of type `bytes32` (`Bytes32Set`), `address` (`AddressSet`) * and `uint256` (`UintSet`) are supported. * * [WARNING] * ==== * Trying to delete such a structure from storage will likely result in data corruption, rendering the structure * unusable. * See https://github.com/ethereum/solidity/pull/11843[ethereum/solidity#11843] for more info. * * In order to clean an EnumerableSet, you can either remove all elements one by one or create a fresh instance using an * array of EnumerableSet. * ==== */ library EnumerableSet { // To implement this library for multiple types with as little code // repetition as possible, we write it in terms of a generic Set type with // bytes32 values. // The Set implementation uses private functions, and user-facing // implementations (such as AddressSet) are just wrappers around the // underlying Set. // This means that we can only create new EnumerableSets for types that fit // in bytes32. struct Set { // Storage of set values bytes32[] _values; // Position of the value in the `values` array, plus 1 because index 0 // means a value is not in the set. mapping(bytes32 => uint256) _indexes; } /** * @dev Add a value to a set. O(1). * * Returns true if the value was added to the set, that is if it was not * already present. */ function _add(Set storage set, bytes32 value) private returns (bool) { if (!_contains(set, value)) { set._values.push(value); // The value is stored at length-1, but we add 1 to all indexes // and use 0 as a sentinel value set._indexes[value] = set._values.length; return true; } else { return false; } } /** * @dev Removes a value from a set. O(1). * * Returns true if the value was removed from the set, that is if it was * present. */ function _remove(Set storage set, bytes32 value) private returns (bool) { // We read and store the value's index to prevent multiple reads from the same storage slot uint256 valueIndex = set._indexes[value]; if (valueIndex != 0) { // Equivalent to contains(set, value) // To delete an element from the _values array in O(1), we swap the element to delete with the last one in // the array, and then remove the last element (sometimes called as 'swap and pop'). // This modifies the order of the array, as noted in {at}. uint256 toDeleteIndex = valueIndex - 1; uint256 lastIndex = set._values.length - 1; if (lastIndex != toDeleteIndex) { bytes32 lastValue = set._values[lastIndex]; // Move the last value to the index where the value to delete is set._values[toDeleteIndex] = lastValue; // Update the index for the moved value set._indexes[lastValue] = valueIndex; // Replace lastValue's index to valueIndex } // Delete the slot where the moved value was stored set._values.pop(); // Delete the index for the deleted slot delete set._indexes[value]; return true; } else { return false; } } /** * @dev Returns true if the value is in the set. O(1). */ function _contains(Set storage set, bytes32 value) private view returns (bool) { return set._indexes[value] != 0; } /** * @dev Returns the number of values on the set. O(1). */ function _length(Set storage set) private view returns (uint256) { return set._values.length; } /** * @dev Returns the value stored at position `index` in the set. O(1). * * Note that there are no guarantees on the ordering of values inside the * array, and it may change when more values are added or removed. * * Requirements: * * - `index` must be strictly less than {length}. */ function _at(Set storage set, uint256 index) private view returns (bytes32) { return set._values[index]; } /** * @dev Return the entire set in an array * * WARNING: This operation will copy the entire storage to memory, which can be quite expensive. This is designed * to mostly be used by view accessors that are queried without any gas fees. Developers should keep in mind that * this function has an unbounded cost, and using it as part of a state-changing function may render the function * uncallable if the set grows to a point where copying to memory consumes too much gas to fit in a block. */ function _values(Set storage set) private view returns (bytes32[] memory) { return set._values; } // Bytes32Set struct Bytes32Set { Set _inner; } /** * @dev Add a value to a set. O(1). * * Returns true if the value was added to the set, that is if it was not * already present. */ function add(Bytes32Set storage set, bytes32 value) internal returns (bool) { return _add(set._inner, value); } /** * @dev Removes a value from a set. O(1). * * Returns true if the value was removed from the set, that is if it was * present. */ function remove(Bytes32Set storage set, bytes32 value) internal returns (bool) { return _remove(set._inner, value); } /** * @dev Returns true if the value is in the set. O(1). */ function contains(Bytes32Set storage set, bytes32 value) internal view returns (bool) { return _contains(set._inner, value); } /** * @dev Returns the number of values in the set. O(1). */ function length(Bytes32Set storage set) internal view returns (uint256) { return _length(set._inner); } /** * @dev Returns the value stored at position `index` in the set. O(1). * * Note that there are no guarantees on the ordering of values inside the * array, and it may change when more values are added or removed. * * Requirements: * * - `index` must be strictly less than {length}. */ function at(Bytes32Set storage set, uint256 index) internal view returns (bytes32) { return _at(set._inner, index); } /** * @dev Return the entire set in an array * * WARNING: This operation will copy the entire storage to memory, which can be quite expensive. This is designed * to mostly be used by view accessors that are queried without any gas fees. Developers should keep in mind that * this function has an unbounded cost, and using it as part of a state-changing function may render the function * uncallable if the set grows to a point where copying to memory consumes too much gas to fit in a block. */ function values(Bytes32Set storage set) internal view returns (bytes32[] memory) { bytes32[] memory store = _values(set._inner); bytes32[] memory result; /// @solidity memory-safe-assembly assembly { result := store } return result; } // AddressSet struct AddressSet { Set _inner; } /** * @dev Add a value to a set. O(1). * * Returns true if the value was added to the set, that is if it was not * already present. */ function add(AddressSet storage set, address value) internal returns (bool) { return _add(set._inner, bytes32(uint256(uint160(value)))); } /** * @dev Removes a value from a set. O(1). * * Returns true if the value was removed from the set, that is if it was * present. */ function remove(AddressSet storage set, address value) internal returns (bool) { return _remove(set._inner, bytes32(uint256(uint160(value)))); } /** * @dev Returns true if the value is in the set. O(1). */ function contains(AddressSet storage set, address value) internal view returns (bool) { return _contains(set._inner, bytes32(uint256(uint160(value)))); } /** * @dev Returns the number of values in the set. O(1). */ function length(AddressSet storage set) internal view returns (uint256) { return _length(set._inner); } /** * @dev Returns the value stored at position `index` in the set. O(1). * * Note that there are no guarantees on the ordering of values inside the * array, and it may change when more values are added or removed. * * Requirements: * * - `index` must be strictly less than {length}. */ function at(AddressSet storage set, uint256 index) internal view returns (address) { return address(uint160(uint256(_at(set._inner, index)))); } /** * @dev Return the entire set in an array * * WARNING: This operation will copy the entire storage to memory, which can be quite expensive. This is designed * to mostly be used by view accessors that are queried without any gas fees. Developers should keep in mind that * this function has an unbounded cost, and using it as part of a state-changing function may render the function * uncallable if the set grows to a point where copying to memory consumes too much gas to fit in a block. */ function values(AddressSet storage set) internal view returns (address[] memory) { bytes32[] memory store = _values(set._inner); address[] memory result; /// @solidity memory-safe-assembly assembly { result := store } return result; } // UintSet struct UintSet { Set _inner; } /** * @dev Add a value to a set. O(1). * * Returns true if the value was added to the set, that is if it was not * already present. */ function add(UintSet storage set, uint256 value) internal returns (bool) { return _add(set._inner, bytes32(value)); } /** * @dev Removes a value from a set. O(1). * * Returns true if the value was removed from the set, that is if it was * present. */ function remove(UintSet storage set, uint256 value) internal returns (bool) { return _remove(set._inner, bytes32(value)); } /** * @dev Returns true if the value is in the set. O(1). */ function contains(UintSet storage set, uint256 value) internal view returns (bool) { return _contains(set._inner, bytes32(value)); } /** * @dev Returns the number of values in the set. O(1). */ function length(UintSet storage set) internal view returns (uint256) { return _length(set._inner); } /** * @dev Returns the value stored at position `index` in the set. O(1). * * Note that there are no guarantees on the ordering of values inside the * array, and it may change when more values are added or removed. * * Requirements: * * - `index` must be strictly less than {length}. */ function at(UintSet storage set, uint256 index) internal view returns (uint256) { return uint256(_at(set._inner, index)); } /** * @dev Return the entire set in an array * * WARNING: This operation will copy the entire storage to memory, which can be quite expensive. This is designed * to mostly be used by view accessors that are queried without any gas fees. Developers should keep in mind that * this function has an unbounded cost, and using it as part of a state-changing function may render the function * uncallable if the set grows to a point where copying to memory consumes too much gas to fit in a block. */ function values(UintSet storage set) internal view returns (uint256[] memory) { bytes32[] memory store = _values(set._inner); uint256[] memory result; /// @solidity memory-safe-assembly assembly { result := store } return result; } }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v4.9.0) (utils/math/Math.sol) pragma solidity ^0.8.0; /** * @dev Standard math utilities missing in the Solidity language. */ library Math { enum Rounding { Down, // Toward negative infinity Up, // Toward infinity Zero // Toward zero } /** * @dev Returns the largest of two numbers. */ function max(uint256 a, uint256 b) internal pure returns (uint256) { return a > b ? a : b; } /** * @dev Returns the smallest of two numbers. */ function min(uint256 a, uint256 b) internal pure returns (uint256) { return a < b ? a : b; } /** * @dev Returns the average of two numbers. The result is rounded towards * zero. */ function average(uint256 a, uint256 b) internal pure returns (uint256) { // (a + b) / 2 can overflow. return (a & b) + (a ^ b) / 2; } /** * @dev Returns the ceiling of the division of two numbers. * * This differs from standard division with `/` in that it rounds up instead * of rounding down. */ function ceilDiv(uint256 a, uint256 b) internal pure returns (uint256) { // (a + b - 1) / b can overflow on addition, so we distribute. return a == 0 ? 0 : (a - 1) / b + 1; } /** * @notice Calculates floor(x * y / denominator) with full precision. Throws if result overflows a uint256 or denominator == 0 * @dev Original credit to Remco Bloemen under MIT license (https://xn--2-umb.com/21/muldiv) * with further edits by Uniswap Labs also under MIT license. */ function mulDiv(uint256 x, uint256 y, uint256 denominator) internal pure returns (uint256 result) { unchecked { // 512-bit multiply [prod1 prod0] = x * y. Compute the product mod 2^256 and mod 2^256 - 1, then use // use the Chinese Remainder Theorem to reconstruct the 512 bit result. The result is stored in two 256 // variables such that product = prod1 * 2^256 + prod0. uint256 prod0; // Least significant 256 bits of the product uint256 prod1; // Most significant 256 bits of the product assembly { let mm := mulmod(x, y, not(0)) prod0 := mul(x, y) prod1 := sub(sub(mm, prod0), lt(mm, prod0)) } // Handle non-overflow cases, 256 by 256 division. if (prod1 == 0) { // Solidity will revert if denominator == 0, unlike the div opcode on its own. // The surrounding unchecked block does not change this fact. // See https://docs.soliditylang.org/en/latest/control-structures.html#checked-or-unchecked-arithmetic. return prod0 / denominator; } // Make sure the result is less than 2^256. Also prevents denominator == 0. require(denominator > prod1, "Math: mulDiv overflow"); /////////////////////////////////////////////// // 512 by 256 division. /////////////////////////////////////////////// // Make division exact by subtracting the remainder from [prod1 prod0]. uint256 remainder; assembly { // Compute remainder using mulmod. remainder := mulmod(x, y, denominator) // Subtract 256 bit number from 512 bit number. prod1 := sub(prod1, gt(remainder, prod0)) prod0 := sub(prod0, remainder) } // Factor powers of two out of denominator and compute largest power of two divisor of denominator. Always >= 1. // See https://cs.stackexchange.com/q/138556/92363. // Does not overflow because the denominator cannot be zero at this stage in the function. uint256 twos = denominator & (~denominator + 1); assembly { // Divide denominator by twos. denominator := div(denominator, twos) // Divide [prod1 prod0] by twos. prod0 := div(prod0, twos) // Flip twos such that it is 2^256 / twos. If twos is zero, then it becomes one. twos := add(div(sub(0, twos), twos), 1) } // Shift in bits from prod1 into prod0. prod0 |= prod1 * twos; // Invert denominator mod 2^256. Now that denominator is an odd number, it has an inverse modulo 2^256 such // that denominator * inv = 1 mod 2^256. Compute the inverse by starting with a seed that is correct for // four bits. That is, denominator * inv = 1 mod 2^4. uint256 inverse = (3 * denominator) ^ 2; // Use the Newton-Raphson iteration to improve the precision. Thanks to Hensel's lifting lemma, this also works // in modular arithmetic, doubling the correct bits in each step. inverse *= 2 - denominator * inverse; // inverse mod 2^8 inverse *= 2 - denominator * inverse; // inverse mod 2^16 inverse *= 2 - denominator * inverse; // inverse mod 2^32 inverse *= 2 - denominator * inverse; // inverse mod 2^64 inverse *= 2 - denominator * inverse; // inverse mod 2^128 inverse *= 2 - denominator * inverse; // inverse mod 2^256 // Because the division is now exact we can divide by multiplying with the modular inverse of denominator. // This will give us the correct result modulo 2^256. Since the preconditions guarantee that the outcome is // less than 2^256, this is the final result. We don't need to compute the high bits of the result and prod1 // is no longer required. result = prod0 * inverse; return result; } } /** * @notice Calculates x * y / denominator with full precision, following the selected rounding direction. */ function mulDiv(uint256 x, uint256 y, uint256 denominator, Rounding rounding) internal pure returns (uint256) { uint256 result = mulDiv(x, y, denominator); if (rounding == Rounding.Up && mulmod(x, y, denominator) > 0) { result += 1; } return result; } /** * @dev Returns the square root of a number. If the number is not a perfect square, the value is rounded down. * * Inspired by Henry S. Warren, Jr.'s "Hacker's Delight" (Chapter 11). */ function sqrt(uint256 a) internal pure returns (uint256) { if (a == 0) { return 0; } // For our first guess, we get the biggest power of 2 which is smaller than the square root of the target. // // We know that the "msb" (most significant bit) of our target number `a` is a power of 2 such that we have // `msb(a) <= a < 2*msb(a)`. This value can be written `msb(a)=2**k` with `k=log2(a)`. // // This can be rewritten `2**log2(a) <= a < 2**(log2(a) + 1)` // → `sqrt(2**k) <= sqrt(a) < sqrt(2**(k+1))` // → `2**(k/2) <= sqrt(a) < 2**((k+1)/2) <= 2**(k/2 + 1)` // // Consequently, `2**(log2(a) / 2)` is a good first approximation of `sqrt(a)` with at least 1 correct bit. uint256 result = 1 << (log2(a) >> 1); // At this point `result` is an estimation with one bit of precision. We know the true value is a uint128, // since it is the square root of a uint256. Newton's method converges quadratically (precision doubles at // every iteration). We thus need at most 7 iteration to turn our partial result with one bit of precision // into the expected uint128 result. unchecked { result = (result + a / result) >> 1; result = (result + a / result) >> 1; result = (result + a / result) >> 1; result = (result + a / result) >> 1; result = (result + a / result) >> 1; result = (result + a / result) >> 1; result = (result + a / result) >> 1; return min(result, a / result); } } /** * @notice Calculates sqrt(a), following the selected rounding direction. */ function sqrt(uint256 a, Rounding rounding) internal pure returns (uint256) { unchecked { uint256 result = sqrt(a); return result + (rounding == Rounding.Up && result * result < a ? 1 : 0); } } /** * @dev Return the log in base 2, rounded down, of a positive value. * Returns 0 if given 0. */ function log2(uint256 value) internal pure returns (uint256) { uint256 result = 0; unchecked { if (value >> 128 > 0) { value >>= 128; result += 128; } if (value >> 64 > 0) { value >>= 64; result += 64; } if (value >> 32 > 0) { value >>= 32; result += 32; } if (value >> 16 > 0) { value >>= 16; result += 16; } if (value >> 8 > 0) { value >>= 8; result += 8; } if (value >> 4 > 0) { value >>= 4; result += 4; } if (value >> 2 > 0) { value >>= 2; result += 2; } if (value >> 1 > 0) { result += 1; } } return result; } /** * @dev Return the log in base 2, following the selected rounding direction, of a positive value. * Returns 0 if given 0. */ function log2(uint256 value, Rounding rounding) internal pure returns (uint256) { unchecked { uint256 result = log2(value); return result + (rounding == Rounding.Up && 1 << result < value ? 1 : 0); } } /** * @dev Return the log in base 10, rounded down, of a positive value. * Returns 0 if given 0. */ function log10(uint256 value) internal pure returns (uint256) { uint256 result = 0; unchecked { if (value >= 10 ** 64) { value /= 10 ** 64; result += 64; } if (value >= 10 ** 32) { value /= 10 ** 32; result += 32; } if (value >= 10 ** 16) { value /= 10 ** 16; result += 16; } if (value >= 10 ** 8) { value /= 10 ** 8; result += 8; } if (value >= 10 ** 4) { value /= 10 ** 4; result += 4; } if (value >= 10 ** 2) { value /= 10 ** 2; result += 2; } if (value >= 10 ** 1) { result += 1; } } return result; } /** * @dev Return the log in base 10, following the selected rounding direction, of a positive value. * Returns 0 if given 0. */ function log10(uint256 value, Rounding rounding) internal pure returns (uint256) { unchecked { uint256 result = log10(value); return result + (rounding == Rounding.Up && 10 ** result < value ? 1 : 0); } } /** * @dev Return the log in base 256, rounded down, of a positive value. * Returns 0 if given 0. * * Adding one to the result gives the number of pairs of hex symbols needed to represent `value` as a hex string. */ function log256(uint256 value) internal pure returns (uint256) { uint256 result = 0; unchecked { if (value >> 128 > 0) { value >>= 128; result += 16; } if (value >> 64 > 0) { value >>= 64; result += 8; } if (value >> 32 > 0) { value >>= 32; result += 4; } if (value >> 16 > 0) { value >>= 16; result += 2; } if (value >> 8 > 0) { result += 1; } } return result; } /** * @dev Return the log in base 256, following the selected rounding direction, of a positive value. * Returns 0 if given 0. */ function log256(uint256 value, Rounding rounding) internal pure returns (uint256) { unchecked { uint256 result = log256(value); return result + (rounding == Rounding.Up && 1 << (result << 3) < value ? 1 : 0); } } }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v4.8.0) (utils/math/SignedMath.sol) pragma solidity ^0.8.0; /** * @dev Standard signed math utilities missing in the Solidity language. */ library SignedMath { /** * @dev Returns the largest of two signed numbers. */ function max(int256 a, int256 b) internal pure returns (int256) { return a > b ? a : b; } /** * @dev Returns the smallest of two signed numbers. */ function min(int256 a, int256 b) internal pure returns (int256) { return a < b ? a : b; } /** * @dev Returns the average of two signed numbers without overflow. * The result is rounded towards zero. */ function average(int256 a, int256 b) internal pure returns (int256) { // Formula from the book "Hacker's Delight" int256 x = (a & b) + ((a ^ b) >> 1); return x + (int256(uint256(x) >> 255) & (a ^ b)); } /** * @dev Returns the absolute unsigned value of a signed value. */ function abs(int256 n) internal pure returns (uint256) { unchecked { // must be unchecked in order to support `n = type(int256).min` return uint256(n >= 0 ? n : -n); } } }
{ "remappings": [ "@openzeppelin/=../node_modules/@openzeppelin/", "@layerzerolabs/=../node_modules/@layerzerolabs/", "@eth-optimism/=../node_modules/@eth-optimism/", "ds-test/=lib/forge-std/lib/ds-test/src/", "forge-std/=lib/forge-std/src/", "fx-portal/=lib/fx-portal/", "hardhat/=node_modules/hardhat/" ], "optimizer": { "enabled": true, "runs": 999999 }, "metadata": { "useLiteralContent": false, "bytecodeHash": "ipfs", "appendCBOR": true }, "outputSelection": { "*": { "*": [ "evm.bytecode", "evm.deployedBytecode", "abi" ] } }, "evmVersion": "paris", "viaIR": false, "libraries": {} }
Contract Security Audit
- No Contract Security Audit Submitted- Submit Audit Here
[{"inputs":[{"internalType":"address","name":"_lockbox","type":"address"},{"internalType":"address","name":"_mailbox","type":"address"}],"stateMutability":"nonpayable","type":"constructor"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"uint8","name":"version","type":"uint8"}],"name":"Initialized","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"previousOwner","type":"address"},{"indexed":true,"internalType":"address","name":"newOwner","type":"address"}],"name":"OwnershipTransferred","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"uint32","name":"origin","type":"uint32"},{"indexed":true,"internalType":"bytes32","name":"recipient","type":"bytes32"},{"indexed":false,"internalType":"uint256","name":"amount","type":"uint256"}],"name":"ReceivedTransferRemote","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"uint32","name":"destination","type":"uint32"},{"indexed":true,"internalType":"bytes32","name":"recipient","type":"bytes32"},{"indexed":false,"internalType":"uint256","name":"amount","type":"uint256"}],"name":"SentTransferRemote","type":"event"},{"inputs":[],"name":"approveLockbox","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"_account","type":"address"}],"name":"balanceOf","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint32","name":"","type":"uint32"}],"name":"destinationGas","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"domains","outputs":[{"internalType":"uint32[]","name":"","type":"uint32[]"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint32","name":"_domain","type":"uint32"},{"internalType":"bytes32","name":"_router","type":"bytes32"}],"name":"enrollRemoteRouter","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint32[]","name":"_domains","type":"uint32[]"},{"internalType":"bytes32[]","name":"_addresses","type":"bytes32[]"}],"name":"enrollRemoteRouters","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint32","name":"_origin","type":"uint32"},{"internalType":"bytes32","name":"_sender","type":"bytes32"},{"internalType":"bytes","name":"_message","type":"bytes"}],"name":"handle","outputs":[],"stateMutability":"payable","type":"function"},{"inputs":[],"name":"hook","outputs":[{"internalType":"contract IPostDispatchHook","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"_hook","type":"address"},{"internalType":"address","name":"_ism","type":"address"},{"internalType":"address","name":"_owner","type":"address"}],"name":"initialize","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"interchainSecurityModule","outputs":[{"internalType":"contract IInterchainSecurityModule","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"localDomain","outputs":[{"internalType":"uint32","name":"","type":"uint32"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"lockbox","outputs":[{"internalType":"contract IXERC20Lockbox","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"mailbox","outputs":[{"internalType":"contract IMailbox","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"owner","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint32","name":"_destinationDomain","type":"uint32"}],"name":"quoteGasPayment","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"renounceOwnership","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint32","name":"_domain","type":"uint32"}],"name":"routers","outputs":[{"internalType":"bytes32","name":"","type":"bytes32"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint32","name":"domain","type":"uint32"},{"internalType":"uint256","name":"gas","type":"uint256"}],"name":"setDestinationGas","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"components":[{"internalType":"uint32","name":"domain","type":"uint32"},{"internalType":"uint256","name":"gas","type":"uint256"}],"internalType":"struct GasRouter.GasRouterConfig[]","name":"gasConfigs","type":"tuple[]"}],"name":"setDestinationGas","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"_hook","type":"address"}],"name":"setHook","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"_module","type":"address"}],"name":"setInterchainSecurityModule","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"newOwner","type":"address"}],"name":"transferOwnership","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint32","name":"_destination","type":"uint32"},{"internalType":"bytes32","name":"_recipient","type":"bytes32"},{"internalType":"uint256","name":"_amountOrId","type":"uint256"},{"internalType":"bytes","name":"_hookMetadata","type":"bytes"},{"internalType":"address","name":"_hook","type":"address"}],"name":"transferRemote","outputs":[{"internalType":"bytes32","name":"messageId","type":"bytes32"}],"stateMutability":"payable","type":"function"},{"inputs":[{"internalType":"uint32","name":"_destination","type":"uint32"},{"internalType":"bytes32","name":"_recipient","type":"bytes32"},{"internalType":"uint256","name":"_amountOrId","type":"uint256"}],"name":"transferRemote","outputs":[{"internalType":"bytes32","name":"messageId","type":"bytes32"}],"stateMutability":"payable","type":"function"},{"inputs":[{"internalType":"uint32","name":"_domain","type":"uint32"}],"name":"unenrollRemoteRouter","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint32[]","name":"_domains","type":"uint32[]"}],"name":"unenrollRemoteRouters","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"wrappedToken","outputs":[{"internalType":"contract IERC20","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"xERC20","outputs":[{"internalType":"contract IXERC20","name":"","type":"address"}],"stateMutability":"view","type":"function"}]
Contract Creation Code
6101206040523480156200001257600080fd5b5060405162003360380380620033608339810160408190526200003591620004ed565b816001600160a01b031663cc4aa2046040518163ffffffff1660e01b81526004016020604051808303816000875af115801562000076573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906200009c91906200052c565b8180808080806001600160a01b0381163b620000ff5760405162461bcd60e51b815260206004820152601e60248201527f4d61696c626f78436c69656e743a20696e76616c6964206d61696c626f78000060448201526064015b60405180910390fd5b6001600160a01b03821660808190526040805163234d8e3d60e21b81529051638d3638f4916004808201926020929091908290030181865afa1580156200014a573d6000803e3d6000fd5b505050506040513d601f19601f8201168201806040525081019062000170919062000553565b63ffffffff1660a05262000184336200022c565b5050506001600160a01b0393841660c052505050821660e08190526040805163b20a0fb960e01b8152905163b20a0fb99160048082019260209290919082900301816000875af1158015620001dd573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906200020391906200052c565b6001600160a01b0316610100526200021a6200027e565b6200022462000417565b50506200059f565b603380546001600160a01b038381166001600160a01b0319831681179093556040519116919082907f8be0079c531659141344cd1fd0a4f28419497f9722a3daafe3b4186f6b6457e090600090a35050565b60c05160e05160405163095ea7b360e01b81526001600160a01b039182166004820152600019602482015291169063095ea7b3906044016020604051808303816000875af1158015620002d5573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190620002fb91906200057b565b620003495760405162461bcd60e51b815260206004820152601c60248201527f6572633230206c6f636b626f7820617070726f7665206661696c6564000000006044820152606401620000f6565b6101005160e05160405163095ea7b360e01b81526001600160a01b039182166004820152600019602482015291169063095ea7b3906044016020604051808303816000875af1158015620003a1573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190620003c791906200057b565b620004155760405162461bcd60e51b815260206004820152601d60248201527f786572633230206c6f636b626f7820617070726f7665206661696c65640000006044820152606401620000f6565b565b600054610100900460ff1615620004815760405162461bcd60e51b815260206004820152602760248201527f496e697469616c697a61626c653a20636f6e747261637420697320696e697469604482015266616c697a696e6760c81b6064820152608401620000f6565b60005460ff9081161462000415576000805460ff191660ff9081179091556040519081527f7f26b83ff96e1f2b6a682f133852f6798a09c465da95921460cefb38474024989060200160405180910390a1565b6001600160a01b0381168114620004ea57600080fd5b50565b600080604083850312156200050157600080fd5b82516200050e81620004d4565b60208401519092506200052181620004d4565b809150509250929050565b6000602082840312156200053f57600080fd5b81516200054c81620004d4565b9392505050565b6000602082840312156200056657600080fd5b815163ffffffff811681146200054c57600080fd5b6000602082840312156200058e57600080fd5b815180151581146200054c57600080fd5b60805160a05160c05160e05161010051612d216200063f6000396000818161022301528181610c1c0152818161189d0152611ab201526000818161031901528181610a7001528181610bce015281816117f90152611b6d01526000818161048301528181610abe01528181610d390152611dbc0152600061040f015260008181610517015281816108d7015281816119640152611cf50152612d216000f3fe6080604052600436106101b75760003560e01c80637f5a7c7b116100ec578063c0c53b8b1161008a578063e9198bf911610064578063e9198bf914610566578063efae508a14610586578063f2ed8c53146105a6578063f2fde38b146105c657600080fd5b8063c0c53b8b146104e5578063d5438eae14610505578063de523cf31461053957600080fd5b80638da5cb5b116100c65780638da5cb5b14610446578063996c6cc314610471578063b1bd6436146104a5578063b49c53a7146104c557600080fd5b80637f5a7c7b146103bd57806381b4e8b4146103ea5780638d3638f4146103fd57600080fd5b806356d5d4751161015957806370a082311161013357806370a082311461033b578063715018a61461035b57806371a15b3814610370578063775313a11461039057600080fd5b806356d5d475146102df57806358f58c5d146102f257806366cc57021461030757600080fd5b80633dfd3873116101955780633dfd38731461026a578063440df4f41461028a57806349d462ef146102ac57806351debffc146102cc57600080fd5b80630e72cc06146101bc5780632ead72f6146101de578063366a596e14610211575b600080fd5b3480156101c857600080fd5b506101dc6101d7366004612599565b6105e6565b005b3480156101ea57600080fd5b506101fe6101f93660046125c8565b610701565b6040519081526020015b60405180910390f35b34801561021d57600080fd5b506102457f000000000000000000000000000000000000000000000000000000000000000081565b60405173ffffffffffffffffffffffffffffffffffffffff9091168152602001610208565b34801561027657600080fd5b506101dc610285366004612599565b610720565b34801561029657600080fd5b5061029f610836565b60405161020891906125e3565b3480156102b857600080fd5b506101dc6102c736600461262d565b610847565b6101fe6102da366004612699565b61086d565b6101dc6102ed366004612710565b6108bf565b3480156102fe57600080fd5b506101dc610a33565b34801561031357600080fd5b506102457f000000000000000000000000000000000000000000000000000000000000000081565b34801561034757600080fd5b506101fe610356366004612599565b610cf1565b34801561036757600080fd5b506101dc610dac565b34801561037c57600080fd5b506101dc61038b3660046127af565b610dbe565b34801561039c57600080fd5b506101fe6103ab3660046125c8565b60ca6020526000908152604090205481565b3480156103c957600080fd5b506065546102459073ffffffffffffffffffffffffffffffffffffffff1681565b6101fe6103f83660046127f1565b610e19565b34801561040957600080fd5b506104317f000000000000000000000000000000000000000000000000000000000000000081565b60405163ffffffff9091168152602001610208565b34801561045257600080fd5b5060335473ffffffffffffffffffffffffffffffffffffffff16610245565b34801561047d57600080fd5b506102457f000000000000000000000000000000000000000000000000000000000000000081565b3480156104b157600080fd5b506101dc6104c0366004612824565b610e27565b3480156104d157600080fd5b506101dc6104e036600461262d565b610eb1565b3480156104f157600080fd5b506101dc610500366004612899565b610ec3565b34801561051157600080fd5b506102457f000000000000000000000000000000000000000000000000000000000000000081565b34801561054557600080fd5b506066546102459073ffffffffffffffffffffffffffffffffffffffff1681565b34801561057257600080fd5b506101dc6105813660046128dc565b611062565b34801561059257600080fd5b506101dc6105a13660046125c8565b611141565b3480156105b257600080fd5b506101fe6105c13660046125c8565b611155565b3480156105d257600080fd5b506101dc6105e1366004612599565b61118c565b8073ffffffffffffffffffffffffffffffffffffffff81163b151580610620575073ffffffffffffffffffffffffffffffffffffffff8116155b6106b1576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152602760248201527f4d61696c626f78436c69656e743a20696e76616c696420636f6e74726163742060448201527f73657474696e670000000000000000000000000000000000000000000000000060648201526084015b60405180910390fd5b6106b9611240565b50606680547fffffffffffffffffffffffff00000000000000000000000000000000000000001673ffffffffffffffffffffffffffffffffffffffff92909216919091179055565b600080610718609763ffffffff808616906112c116565b949350505050565b8073ffffffffffffffffffffffffffffffffffffffff81163b15158061075a575073ffffffffffffffffffffffffffffffffffffffff8116155b6107e6576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152602760248201527f4d61696c626f78436c69656e743a20696e76616c696420636f6e74726163742060448201527f73657474696e670000000000000000000000000000000000000000000000000060648201526084016106a8565b6107ee611240565b50606580547fffffffffffffffffffffffff00000000000000000000000000000000000000001673ffffffffffffffffffffffffffffffffffffffff92909216919091179055565b606061084260976112da565b905090565b61084f611240565b63ffffffff91909116600090815260ca6020526040902055565b5050565b60006108b48787873488888080601f0160208091040260200160405190810160405280939291908181526020018383808284376000920191909152508a9250611395915050565b979650505050505050565b3373ffffffffffffffffffffffffffffffffffffffff7f00000000000000000000000000000000000000000000000000000000000000001614610984576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152602160248201527f4d61696c626f78436c69656e743a2073656e646572206e6f74206d61696c626f60448201527f780000000000000000000000000000000000000000000000000000000000000060648201526084016106a8565b600061098f8561140e565b9050838114610a20576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152602560248201527f456e726f6c6c656420726f7574657220646f6573206e6f74206d61746368207360448201527f656e64657200000000000000000000000000000000000000000000000000000060648201526084016106a8565b610a2c85858585611474565b5050505050565b6040517f095ea7b300000000000000000000000000000000000000000000000000000000815273ffffffffffffffffffffffffffffffffffffffff7f0000000000000000000000000000000000000000000000000000000000000000811660048301527fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff60248301527f0000000000000000000000000000000000000000000000000000000000000000169063095ea7b3906044016020604051808303816000875af1158015610b07573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190610b2b919061293c565b610b91576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152601c60248201527f6572633230206c6f636b626f7820617070726f7665206661696c65640000000060448201526064016106a8565b6040517f095ea7b300000000000000000000000000000000000000000000000000000000815273ffffffffffffffffffffffffffffffffffffffff7f0000000000000000000000000000000000000000000000000000000000000000811660048301527fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff60248301527f0000000000000000000000000000000000000000000000000000000000000000169063095ea7b3906044016020604051808303816000875af1158015610c65573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190610c89919061293c565b610cef576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152601d60248201527f786572633230206c6f636b626f7820617070726f7665206661696c656400000060448201526064016106a8565b565b6040517f70a0823100000000000000000000000000000000000000000000000000000000815273ffffffffffffffffffffffffffffffffffffffff82811660048301526000917f0000000000000000000000000000000000000000000000000000000000000000909116906370a0823190602401602060405180830381865afa158015610d82573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190610da6919061295e565b92915050565b610db4611240565b610cef60006114fe565b610dc6611240565b8060005b81811015610e1357610e01848483818110610de757610de7612977565b9050602002016020810190610dfc91906125c8565b611575565b610e0c6001826129d5565b9050610dca565b50505050565b6000610718848484346115ca565b610e2f611240565b60005b81811015610eac57610e9a838383818110610e4f57610e4f612977565b610e6592602060409092020190810191506125c8565b848484818110610e7757610e77612977565b9050604002016020013563ffffffff909116600090815260ca6020526040902055565b610ea56001826129d5565b9050610e32565b505050565b610eb9611240565b6108698282611603565b600054610100900460ff1615808015610ee35750600054600160ff909116105b80610efd5750303b158015610efd575060005460ff166001145b610f89576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152602e60248201527f496e697469616c697a61626c653a20636f6e747261637420697320616c72656160448201527f647920696e697469616c697a656400000000000000000000000000000000000060648201526084016106a8565b600080547fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff001660011790558015610fe757600080547fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff00ff166101001790555b610fef610a33565b610ffa848484611619565b8015610e1357600080547fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff00ff169055604051600181527f7f26b83ff96e1f2b6a682f133852f6798a09c465da95921460cefb38474024989060200160405180910390a150505050565b61106a611240565b8281146110d3576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152600760248201527f216c656e6774680000000000000000000000000000000000000000000000000060448201526064016106a8565b8260005b81811015611139576111278686838181106110f4576110f4612977565b905060200201602081019061110991906125c8565b85858481811061111b5761111b612977565b90506020020135611603565b6111326001826129d5565b90506110d7565b505050505050565b611149611240565b61115281611575565b50565b604080516020810190915260008082526065549091610da69184919073ffffffffffffffffffffffffffffffffffffffff166116d3565b611194611240565b73ffffffffffffffffffffffffffffffffffffffff8116611237576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152602660248201527f4f776e61626c653a206e6577206f776e657220697320746865207a65726f206160448201527f646472657373000000000000000000000000000000000000000000000000000060648201526084016106a8565b611152816114fe565b60335473ffffffffffffffffffffffffffffffffffffffff163314610cef576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820181905260248201527f4f776e61626c653a2063616c6c6572206973206e6f7420746865206f776e657260448201526064016106a8565b6000806112ce84846116e9565b915091505b9250929050565b606060006112e783611723565b9050805167ffffffffffffffff811115611303576113036129e8565b60405190808252806020026020018201604052801561132c578160200160208202803683370190505b50915060005b815181101561138e5781818151811061134d5761134d612977565b602002602001015183828151811061136757611367612977565b63ffffffff909216602092830291909101909101528061138681612a17565b915050611332565b5050919050565b6000806113a1866117be565b905060006113b0888884611925565b90506113bf8987838888611954565b9250878963ffffffff167fd229aacb94204188fe8042965fa6b269c62dc5818b21238779ab64bdd17efeec896040516113fa91815260200190565b60405180910390a350509695505050505050565b60008080611426609763ffffffff808716906112c116565b915091508161143485611a07565b9061146c576040517f08c379a00000000000000000000000000000000000000000000000000000000081526004016106a89190612abd565b509392505050565b60006114808383611a3e565b9050600061148e8484611a5d565b905036600061149d8686611a6d565b915091506114b36114ab8590565b848484611a7d565b838863ffffffff167fba20947a325f450d232530e5f5fce293e7963499d5309a07cee84a269f2f15a6856040516114ec91815260200190565b60405180910390a35050505050505050565b6033805473ffffffffffffffffffffffffffffffffffffffff8381167fffffffffffffffffffffffff0000000000000000000000000000000000000000831681179093556040519116919082907f8be0079c531659141344cd1fd0a4f28419497f9722a3daafe3b4186f6b6457e090600090a35050565b611589609763ffffffff80841690611bd116565b61159282611a07565b90610869576040517f08c379a00000000000000000000000000000000000000000000000000000000081526004016106a89190612abd565b60006115fa858585856115dc8a611bdd565b60655473ffffffffffffffffffffffffffffffffffffffff16611395565b95945050505050565b610869609763ffffffff808516908490611bff16565b600054610100900460ff166116b0576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152602b60248201527f496e697469616c697a61626c653a20636f6e7472616374206973206e6f74206960448201527f6e697469616c697a696e6700000000000000000000000000000000000000000060648201526084016106a8565b6116b8611c0a565b6116c183610720565b6116ca826105e6565b610eac816114fe565b600061071884846116e387611bdd565b85611ca9565b60008181526002830160205260408120548190806117185761170b8585611d7d565b9250600091506112d39050565b6001925090506112d3565b6060600061173083611d89565b90508067ffffffffffffffff81111561174b5761174b6129e8565b604051908082528060200260200182016040528015611774578160200160208202803683370190505b50915060005b8181101561138e5761178c8482611d94565b60001c8382815181106117a1576117a1612977565b6020908102919091010152806117b681612a17565b91505061177a565b60606117c982611da0565b506040517fb6b55f25000000000000000000000000000000000000000000000000000000008152600481018390527f000000000000000000000000000000000000000000000000000000000000000073ffffffffffffffffffffffffffffffffffffffff169063b6b55f2590602401600060405180830381600087803b15801561185257600080fd5b505af1158015611866573d6000803e3d6000fd5b50506040517f9dc29fac000000000000000000000000000000000000000000000000000000008152306004820152602481018590527f000000000000000000000000000000000000000000000000000000000000000073ffffffffffffffffffffffffffffffffffffffff169250639dc29fac9150604401600060405180830381600087803b1580156118f857600080fd5b505af115801561190c573d6000803e3d6000fd5b5050604080516020810190915260008152949350505050565b606083838360405160200161193c93929190612ad0565b60405160208183030381529060405290509392505050565b6000806119608761140e565b90507f000000000000000000000000000000000000000000000000000000000000000073ffffffffffffffffffffffffffffffffffffffff166310b83dc08789848989896040518763ffffffff1660e01b81526004016119c4959493929190612afd565b60206040518083038185885af11580156119e2573d6000803e3d6000fd5b50505050506040513d601f19601f820116820180604052508101906108b4919061295e565b6060611a188263ffffffff16611df8565b604051602001611a289190612b5d565b6040516020818303038152906040529050919050565b6000611a4d6020828486612ba2565b611a5691612bcc565b9392505050565b6000611a4d604060208486612ba2565b3660006112ce8360408187612ba2565b6040517f40c10f19000000000000000000000000000000000000000000000000000000008152306004820152602481018490527f000000000000000000000000000000000000000000000000000000000000000073ffffffffffffffffffffffffffffffffffffffff16906340c10f1990604401600060405180830381600087803b158015611b0b57600080fd5b505af1158015611b1f573d6000803e3d6000fd5b50506040517f205c287800000000000000000000000000000000000000000000000000000000815273ffffffffffffffffffffffffffffffffffffffff8781166004830152602482018790527f000000000000000000000000000000000000000000000000000000000000000016925063205c28789150604401600060405180830381600087803b158015611bb357600080fd5b505af1158015611bc7573d6000803e3d6000fd5b5050505050505050565b6000611a568383611eb6565b63ffffffff8116600090815260ca6020526040902054606090610da690611ed3565b610e13838383611ef1565b600054610100900460ff16611ca1576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152602b60248201527f496e697469616c697a61626c653a20636f6e7472616374206973206e6f74206960448201527f6e697469616c697a696e6700000000000000000000000000000000000000000060648201526084016106a8565b610cef611f0e565b600080611cb58661140e565b6040517f81d2ea9500000000000000000000000000000000000000000000000000000000815290915073ffffffffffffffffffffffffffffffffffffffff7f000000000000000000000000000000000000000000000000000000000000000016906381d2ea9590611d3290899085908a908a908a90600401612afd565b602060405180830381865afa158015611d4f573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190611d73919061295e565b9695505050505050565b6000611a568383611fae565b6000610da682611fc6565b6000611a568383611fd0565b6060611de473ffffffffffffffffffffffffffffffffffffffff7f000000000000000000000000000000000000000000000000000000000000000016333085611ffa565b505060408051602081019091526000815290565b60606000611e058361208f565b600101905060008167ffffffffffffffff811115611e2557611e256129e8565b6040519080825280601f01601f191660200182016040528015611e4f576020820181803683370190505b5090508181016020015b7fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff017f3031323334353637383961626364656600000000000000000000000000000000600a86061a8153600a8504945084611e5957509392505050565b60008181526002830160205260408120819055611a568383612171565b6060610da6600083336040518060200160405280600081525061217d565b6000828152600284016020526040812082905561071884846121b2565b600054610100900460ff16611fa5576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152602b60248201527f496e697469616c697a61626c653a20636f6e7472616374206973206e6f74206960448201527f6e697469616c697a696e6700000000000000000000000000000000000000000060648201526084016106a8565b610cef336114fe565b60008181526001830160205260408120541515611a56565b6000610da6825490565b6000826000018281548110611fe757611fe7612977565b9060005260206000200154905092915050565b6040805173ffffffffffffffffffffffffffffffffffffffff85811660248301528416604482015260648082018490528251808303909101815260849091019091526020810180517bffffffffffffffffffffffffffffffffffffffffffffffffffffffff167f23b872dd00000000000000000000000000000000000000000000000000000000179052610e139085906121be565b6000807a184f03e93ff9f4daa797ed6e38ed64bf6a1f01000000000000000083106120d8577a184f03e93ff9f4daa797ed6e38ed64bf6a1f010000000000000000830492506040015b6d04ee2d6d415b85acef81000000008310612104576d04ee2d6d415b85acef8100000000830492506020015b662386f26fc10000831061212257662386f26fc10000830492506010015b6305f5e100831061213a576305f5e100830492506008015b612710831061214e57612710830492506004015b60648310612160576064830492506002015b600a8310610da65760010192915050565b6000611a5683836122cd565b6060600185858585604051602001612199959493929190612c08565b6040516020818303038152906040529050949350505050565b6000611a5683836123c0565b6000612220826040518060400160405280602081526020017f5361666545524332303a206c6f772d6c6576656c2063616c6c206661696c65648152508573ffffffffffffffffffffffffffffffffffffffff1661240f9092919063ffffffff16565b9050805160001480612241575080806020019051810190612241919061293c565b610eac576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152602a60248201527f5361666545524332303a204552433230206f7065726174696f6e20646964206e60448201527f6f7420737563636565640000000000000000000000000000000000000000000060648201526084016106a8565b600081815260018301602052604081205480156123b65760006122f1600183612c8d565b855490915060009061230590600190612c8d565b905081811461236a57600086600001828154811061232557612325612977565b906000526020600020015490508087600001848154811061234857612348612977565b6000918252602080832090910192909255918252600188019052604090208390555b855486908061237b5761237b612ca0565b600190038181906000526020600020016000905590558560010160008681526020019081526020016000206000905560019350505050610da6565b6000915050610da6565b600081815260018301602052604081205461240757508154600181810184556000848152602080822090930184905584548482528286019093526040902091909155610da6565b506000610da6565b60606107188484600085856000808673ffffffffffffffffffffffffffffffffffffffff1685876040516124439190612ccf565b60006040518083038185875af1925050503d8060008114612480576040519150601f19603f3d011682016040523d82523d6000602084013e612485565b606091505b50915091506108b487838387606083156125275782516000036125205773ffffffffffffffffffffffffffffffffffffffff85163b612520576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152601d60248201527f416464726573733a2063616c6c20746f206e6f6e2d636f6e747261637400000060448201526064016106a8565b5081610718565b610718838381511561253c5781518083602001fd5b806040517f08c379a00000000000000000000000000000000000000000000000000000000081526004016106a89190612abd565b803573ffffffffffffffffffffffffffffffffffffffff8116811461259457600080fd5b919050565b6000602082840312156125ab57600080fd5b611a5682612570565b803563ffffffff8116811461259457600080fd5b6000602082840312156125da57600080fd5b611a56826125b4565b6020808252825182820181905260009190848201906040850190845b8181101561262157835163ffffffff16835292840192918401916001016125ff565b50909695505050505050565b6000806040838503121561264057600080fd5b612649836125b4565b946020939093013593505050565b60008083601f84011261266957600080fd5b50813567ffffffffffffffff81111561268157600080fd5b6020830191508360208285010111156112d357600080fd5b60008060008060008060a087890312156126b257600080fd5b6126bb876125b4565b95506020870135945060408701359350606087013567ffffffffffffffff8111156126e557600080fd5b6126f189828a01612657565b9094509250612704905060808801612570565b90509295509295509295565b6000806000806060858703121561272657600080fd5b61272f856125b4565b935060208501359250604085013567ffffffffffffffff81111561275257600080fd5b61275e87828801612657565b95989497509550505050565b60008083601f84011261277c57600080fd5b50813567ffffffffffffffff81111561279457600080fd5b6020830191508360208260051b85010111156112d357600080fd5b600080602083850312156127c257600080fd5b823567ffffffffffffffff8111156127d957600080fd5b6127e58582860161276a565b90969095509350505050565b60008060006060848603121561280657600080fd5b61280f846125b4565b95602085013595506040909401359392505050565b6000806020838503121561283757600080fd5b823567ffffffffffffffff8082111561284f57600080fd5b818501915085601f83011261286357600080fd5b81358181111561287257600080fd5b8660208260061b850101111561288757600080fd5b60209290920196919550909350505050565b6000806000606084860312156128ae57600080fd5b6128b784612570565b92506128c560208501612570565b91506128d360408501612570565b90509250925092565b600080600080604085870312156128f257600080fd5b843567ffffffffffffffff8082111561290a57600080fd5b6129168883890161276a565b9096509450602087013591508082111561292f57600080fd5b5061275e8782880161276a565b60006020828403121561294e57600080fd5b81518015158114611a5657600080fd5b60006020828403121561297057600080fd5b5051919050565b7f4e487b7100000000000000000000000000000000000000000000000000000000600052603260045260246000fd5b7f4e487b7100000000000000000000000000000000000000000000000000000000600052601160045260246000fd5b80820180821115610da657610da66129a6565b7f4e487b7100000000000000000000000000000000000000000000000000000000600052604160045260246000fd5b60007fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff8203612a4857612a486129a6565b5060010190565b60005b83811015612a6a578181015183820152602001612a52565b50506000910152565b60008151808452612a8b816020860160208601612a4f565b601f017fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffe0169290920160200192915050565b602081526000611a566020830184612a73565b83815282602082015260008251612aee816040850160208701612a4f565b91909101604001949350505050565b63ffffffff8616815284602082015260a060408201526000612b2260a0830186612a73565b8281036060840152612b348186612a73565b91505073ffffffffffffffffffffffffffffffffffffffff831660808301529695505050505050565b7f4e6f20726f7574657220656e726f6c6c656420666f7220646f6d61696e3a2000815260008251612b9581601f850160208701612a4f565b91909101601f0192915050565b60008085851115612bb257600080fd5b83861115612bbf57600080fd5b5050820193919092039150565b80356020831015610da6577fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff602084900360031b1b1692915050565b7fffff0000000000000000000000000000000000000000000000000000000000008660f01b1681528460028201528360228201527fffffffffffffffffffffffffffffffffffffffff0000000000000000000000008360601b16604282015260008251612c7c816056850160208701612a4f565b919091016056019695505050505050565b81810381811115610da657610da66129a6565b7f4e487b7100000000000000000000000000000000000000000000000000000000600052603160045260246000fd5b60008251612ce1818460208701612a4f565b919091019291505056fea2646970667358221220863c53da358d51b7f81bd7e8d2ab4b730910c8213130443d0cd4172050ef15c864736f6c63430008130033000000000000000000000000c8140da31e6bca19b287cc35531c2212763c2059000000000000000000000000c005dc82818d67af737725bd4bf75435d065d239
Deployed Bytecode
0x6080604052600436106101b75760003560e01c80637f5a7c7b116100ec578063c0c53b8b1161008a578063e9198bf911610064578063e9198bf914610566578063efae508a14610586578063f2ed8c53146105a6578063f2fde38b146105c657600080fd5b8063c0c53b8b146104e5578063d5438eae14610505578063de523cf31461053957600080fd5b80638da5cb5b116100c65780638da5cb5b14610446578063996c6cc314610471578063b1bd6436146104a5578063b49c53a7146104c557600080fd5b80637f5a7c7b146103bd57806381b4e8b4146103ea5780638d3638f4146103fd57600080fd5b806356d5d4751161015957806370a082311161013357806370a082311461033b578063715018a61461035b57806371a15b3814610370578063775313a11461039057600080fd5b806356d5d475146102df57806358f58c5d146102f257806366cc57021461030757600080fd5b80633dfd3873116101955780633dfd38731461026a578063440df4f41461028a57806349d462ef146102ac57806351debffc146102cc57600080fd5b80630e72cc06146101bc5780632ead72f6146101de578063366a596e14610211575b600080fd5b3480156101c857600080fd5b506101dc6101d7366004612599565b6105e6565b005b3480156101ea57600080fd5b506101fe6101f93660046125c8565b610701565b6040519081526020015b60405180910390f35b34801561021d57600080fd5b506102457f0000000000000000000000002416092f143378750bb29b79ed961ab195cceea581565b60405173ffffffffffffffffffffffffffffffffffffffff9091168152602001610208565b34801561027657600080fd5b506101dc610285366004612599565b610720565b34801561029657600080fd5b5061029f610836565b60405161020891906125e3565b3480156102b857600080fd5b506101dc6102c736600461262d565b610847565b6101fe6102da366004612699565b61086d565b6101dc6102ed366004612710565b6108bf565b3480156102fe57600080fd5b506101dc610a33565b34801561031357600080fd5b506102457f000000000000000000000000c8140da31e6bca19b287cc35531c2212763c205981565b34801561034757600080fd5b506101fe610356366004612599565b610cf1565b34801561036757600080fd5b506101dc610dac565b34801561037c57600080fd5b506101dc61038b3660046127af565b610dbe565b34801561039c57600080fd5b506101fe6103ab3660046125c8565b60ca6020526000908152604090205481565b3480156103c957600080fd5b506065546102459073ffffffffffffffffffffffffffffffffffffffff1681565b6101fe6103f83660046127f1565b610e19565b34801561040957600080fd5b506104317f000000000000000000000000000000000000000000000000000000000000000181565b60405163ffffffff9091168152602001610208565b34801561045257600080fd5b5060335473ffffffffffffffffffffffffffffffffffffffff16610245565b34801561047d57600080fd5b506102457f000000000000000000000000bf5495efe5db9ce00f80364c8b423567e58d211081565b3480156104b157600080fd5b506101dc6104c0366004612824565b610e27565b3480156104d157600080fd5b506101dc6104e036600461262d565b610eb1565b3480156104f157600080fd5b506101dc610500366004612899565b610ec3565b34801561051157600080fd5b506102457f000000000000000000000000c005dc82818d67af737725bd4bf75435d065d23981565b34801561054557600080fd5b506066546102459073ffffffffffffffffffffffffffffffffffffffff1681565b34801561057257600080fd5b506101dc6105813660046128dc565b611062565b34801561059257600080fd5b506101dc6105a13660046125c8565b611141565b3480156105b257600080fd5b506101fe6105c13660046125c8565b611155565b3480156105d257600080fd5b506101dc6105e1366004612599565b61118c565b8073ffffffffffffffffffffffffffffffffffffffff81163b151580610620575073ffffffffffffffffffffffffffffffffffffffff8116155b6106b1576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152602760248201527f4d61696c626f78436c69656e743a20696e76616c696420636f6e74726163742060448201527f73657474696e670000000000000000000000000000000000000000000000000060648201526084015b60405180910390fd5b6106b9611240565b50606680547fffffffffffffffffffffffff00000000000000000000000000000000000000001673ffffffffffffffffffffffffffffffffffffffff92909216919091179055565b600080610718609763ffffffff808616906112c116565b949350505050565b8073ffffffffffffffffffffffffffffffffffffffff81163b15158061075a575073ffffffffffffffffffffffffffffffffffffffff8116155b6107e6576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152602760248201527f4d61696c626f78436c69656e743a20696e76616c696420636f6e74726163742060448201527f73657474696e670000000000000000000000000000000000000000000000000060648201526084016106a8565b6107ee611240565b50606580547fffffffffffffffffffffffff00000000000000000000000000000000000000001673ffffffffffffffffffffffffffffffffffffffff92909216919091179055565b606061084260976112da565b905090565b61084f611240565b63ffffffff91909116600090815260ca6020526040902055565b5050565b60006108b48787873488888080601f0160208091040260200160405190810160405280939291908181526020018383808284376000920191909152508a9250611395915050565b979650505050505050565b3373ffffffffffffffffffffffffffffffffffffffff7f000000000000000000000000c005dc82818d67af737725bd4bf75435d065d2391614610984576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152602160248201527f4d61696c626f78436c69656e743a2073656e646572206e6f74206d61696c626f60448201527f780000000000000000000000000000000000000000000000000000000000000060648201526084016106a8565b600061098f8561140e565b9050838114610a20576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152602560248201527f456e726f6c6c656420726f7574657220646f6573206e6f74206d61746368207360448201527f656e64657200000000000000000000000000000000000000000000000000000060648201526084016106a8565b610a2c85858585611474565b5050505050565b6040517f095ea7b300000000000000000000000000000000000000000000000000000000815273ffffffffffffffffffffffffffffffffffffffff7f000000000000000000000000c8140da31e6bca19b287cc35531c2212763c2059811660048301527fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff60248301527f000000000000000000000000bf5495efe5db9ce00f80364c8b423567e58d2110169063095ea7b3906044016020604051808303816000875af1158015610b07573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190610b2b919061293c565b610b91576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152601c60248201527f6572633230206c6f636b626f7820617070726f7665206661696c65640000000060448201526064016106a8565b6040517f095ea7b300000000000000000000000000000000000000000000000000000000815273ffffffffffffffffffffffffffffffffffffffff7f000000000000000000000000c8140da31e6bca19b287cc35531c2212763c2059811660048301527fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff60248301527f0000000000000000000000002416092f143378750bb29b79ed961ab195cceea5169063095ea7b3906044016020604051808303816000875af1158015610c65573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190610c89919061293c565b610cef576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152601d60248201527f786572633230206c6f636b626f7820617070726f7665206661696c656400000060448201526064016106a8565b565b6040517f70a0823100000000000000000000000000000000000000000000000000000000815273ffffffffffffffffffffffffffffffffffffffff82811660048301526000917f000000000000000000000000bf5495efe5db9ce00f80364c8b423567e58d2110909116906370a0823190602401602060405180830381865afa158015610d82573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190610da6919061295e565b92915050565b610db4611240565b610cef60006114fe565b610dc6611240565b8060005b81811015610e1357610e01848483818110610de757610de7612977565b9050602002016020810190610dfc91906125c8565b611575565b610e0c6001826129d5565b9050610dca565b50505050565b6000610718848484346115ca565b610e2f611240565b60005b81811015610eac57610e9a838383818110610e4f57610e4f612977565b610e6592602060409092020190810191506125c8565b848484818110610e7757610e77612977565b9050604002016020013563ffffffff909116600090815260ca6020526040902055565b610ea56001826129d5565b9050610e32565b505050565b610eb9611240565b6108698282611603565b600054610100900460ff1615808015610ee35750600054600160ff909116105b80610efd5750303b158015610efd575060005460ff166001145b610f89576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152602e60248201527f496e697469616c697a61626c653a20636f6e747261637420697320616c72656160448201527f647920696e697469616c697a656400000000000000000000000000000000000060648201526084016106a8565b600080547fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff001660011790558015610fe757600080547fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff00ff166101001790555b610fef610a33565b610ffa848484611619565b8015610e1357600080547fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff00ff169055604051600181527f7f26b83ff96e1f2b6a682f133852f6798a09c465da95921460cefb38474024989060200160405180910390a150505050565b61106a611240565b8281146110d3576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152600760248201527f216c656e6774680000000000000000000000000000000000000000000000000060448201526064016106a8565b8260005b81811015611139576111278686838181106110f4576110f4612977565b905060200201602081019061110991906125c8565b85858481811061111b5761111b612977565b90506020020135611603565b6111326001826129d5565b90506110d7565b505050505050565b611149611240565b61115281611575565b50565b604080516020810190915260008082526065549091610da69184919073ffffffffffffffffffffffffffffffffffffffff166116d3565b611194611240565b73ffffffffffffffffffffffffffffffffffffffff8116611237576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152602660248201527f4f776e61626c653a206e6577206f776e657220697320746865207a65726f206160448201527f646472657373000000000000000000000000000000000000000000000000000060648201526084016106a8565b611152816114fe565b60335473ffffffffffffffffffffffffffffffffffffffff163314610cef576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820181905260248201527f4f776e61626c653a2063616c6c6572206973206e6f7420746865206f776e657260448201526064016106a8565b6000806112ce84846116e9565b915091505b9250929050565b606060006112e783611723565b9050805167ffffffffffffffff811115611303576113036129e8565b60405190808252806020026020018201604052801561132c578160200160208202803683370190505b50915060005b815181101561138e5781818151811061134d5761134d612977565b602002602001015183828151811061136757611367612977565b63ffffffff909216602092830291909101909101528061138681612a17565b915050611332565b5050919050565b6000806113a1866117be565b905060006113b0888884611925565b90506113bf8987838888611954565b9250878963ffffffff167fd229aacb94204188fe8042965fa6b269c62dc5818b21238779ab64bdd17efeec896040516113fa91815260200190565b60405180910390a350509695505050505050565b60008080611426609763ffffffff808716906112c116565b915091508161143485611a07565b9061146c576040517f08c379a00000000000000000000000000000000000000000000000000000000081526004016106a89190612abd565b509392505050565b60006114808383611a3e565b9050600061148e8484611a5d565b905036600061149d8686611a6d565b915091506114b36114ab8590565b848484611a7d565b838863ffffffff167fba20947a325f450d232530e5f5fce293e7963499d5309a07cee84a269f2f15a6856040516114ec91815260200190565b60405180910390a35050505050505050565b6033805473ffffffffffffffffffffffffffffffffffffffff8381167fffffffffffffffffffffffff0000000000000000000000000000000000000000831681179093556040519116919082907f8be0079c531659141344cd1fd0a4f28419497f9722a3daafe3b4186f6b6457e090600090a35050565b611589609763ffffffff80841690611bd116565b61159282611a07565b90610869576040517f08c379a00000000000000000000000000000000000000000000000000000000081526004016106a89190612abd565b60006115fa858585856115dc8a611bdd565b60655473ffffffffffffffffffffffffffffffffffffffff16611395565b95945050505050565b610869609763ffffffff808516908490611bff16565b600054610100900460ff166116b0576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152602b60248201527f496e697469616c697a61626c653a20636f6e7472616374206973206e6f74206960448201527f6e697469616c697a696e6700000000000000000000000000000000000000000060648201526084016106a8565b6116b8611c0a565b6116c183610720565b6116ca826105e6565b610eac816114fe565b600061071884846116e387611bdd565b85611ca9565b60008181526002830160205260408120548190806117185761170b8585611d7d565b9250600091506112d39050565b6001925090506112d3565b6060600061173083611d89565b90508067ffffffffffffffff81111561174b5761174b6129e8565b604051908082528060200260200182016040528015611774578160200160208202803683370190505b50915060005b8181101561138e5761178c8482611d94565b60001c8382815181106117a1576117a1612977565b6020908102919091010152806117b681612a17565b91505061177a565b60606117c982611da0565b506040517fb6b55f25000000000000000000000000000000000000000000000000000000008152600481018390527f000000000000000000000000c8140da31e6bca19b287cc35531c2212763c205973ffffffffffffffffffffffffffffffffffffffff169063b6b55f2590602401600060405180830381600087803b15801561185257600080fd5b505af1158015611866573d6000803e3d6000fd5b50506040517f9dc29fac000000000000000000000000000000000000000000000000000000008152306004820152602481018590527f0000000000000000000000002416092f143378750bb29b79ed961ab195cceea573ffffffffffffffffffffffffffffffffffffffff169250639dc29fac9150604401600060405180830381600087803b1580156118f857600080fd5b505af115801561190c573d6000803e3d6000fd5b5050604080516020810190915260008152949350505050565b606083838360405160200161193c93929190612ad0565b60405160208183030381529060405290509392505050565b6000806119608761140e565b90507f000000000000000000000000c005dc82818d67af737725bd4bf75435d065d23973ffffffffffffffffffffffffffffffffffffffff166310b83dc08789848989896040518763ffffffff1660e01b81526004016119c4959493929190612afd565b60206040518083038185885af11580156119e2573d6000803e3d6000fd5b50505050506040513d601f19601f820116820180604052508101906108b4919061295e565b6060611a188263ffffffff16611df8565b604051602001611a289190612b5d565b6040516020818303038152906040529050919050565b6000611a4d6020828486612ba2565b611a5691612bcc565b9392505050565b6000611a4d604060208486612ba2565b3660006112ce8360408187612ba2565b6040517f40c10f19000000000000000000000000000000000000000000000000000000008152306004820152602481018490527f0000000000000000000000002416092f143378750bb29b79ed961ab195cceea573ffffffffffffffffffffffffffffffffffffffff16906340c10f1990604401600060405180830381600087803b158015611b0b57600080fd5b505af1158015611b1f573d6000803e3d6000fd5b50506040517f205c287800000000000000000000000000000000000000000000000000000000815273ffffffffffffffffffffffffffffffffffffffff8781166004830152602482018790527f000000000000000000000000c8140da31e6bca19b287cc35531c2212763c205916925063205c28789150604401600060405180830381600087803b158015611bb357600080fd5b505af1158015611bc7573d6000803e3d6000fd5b5050505050505050565b6000611a568383611eb6565b63ffffffff8116600090815260ca6020526040902054606090610da690611ed3565b610e13838383611ef1565b600054610100900460ff16611ca1576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152602b60248201527f496e697469616c697a61626c653a20636f6e7472616374206973206e6f74206960448201527f6e697469616c697a696e6700000000000000000000000000000000000000000060648201526084016106a8565b610cef611f0e565b600080611cb58661140e565b6040517f81d2ea9500000000000000000000000000000000000000000000000000000000815290915073ffffffffffffffffffffffffffffffffffffffff7f000000000000000000000000c005dc82818d67af737725bd4bf75435d065d23916906381d2ea9590611d3290899085908a908a908a90600401612afd565b602060405180830381865afa158015611d4f573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190611d73919061295e565b9695505050505050565b6000611a568383611fae565b6000610da682611fc6565b6000611a568383611fd0565b6060611de473ffffffffffffffffffffffffffffffffffffffff7f000000000000000000000000bf5495efe5db9ce00f80364c8b423567e58d211016333085611ffa565b505060408051602081019091526000815290565b60606000611e058361208f565b600101905060008167ffffffffffffffff811115611e2557611e256129e8565b6040519080825280601f01601f191660200182016040528015611e4f576020820181803683370190505b5090508181016020015b7fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff017f3031323334353637383961626364656600000000000000000000000000000000600a86061a8153600a8504945084611e5957509392505050565b60008181526002830160205260408120819055611a568383612171565b6060610da6600083336040518060200160405280600081525061217d565b6000828152600284016020526040812082905561071884846121b2565b600054610100900460ff16611fa5576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152602b60248201527f496e697469616c697a61626c653a20636f6e7472616374206973206e6f74206960448201527f6e697469616c697a696e6700000000000000000000000000000000000000000060648201526084016106a8565b610cef336114fe565b60008181526001830160205260408120541515611a56565b6000610da6825490565b6000826000018281548110611fe757611fe7612977565b9060005260206000200154905092915050565b6040805173ffffffffffffffffffffffffffffffffffffffff85811660248301528416604482015260648082018490528251808303909101815260849091019091526020810180517bffffffffffffffffffffffffffffffffffffffffffffffffffffffff167f23b872dd00000000000000000000000000000000000000000000000000000000179052610e139085906121be565b6000807a184f03e93ff9f4daa797ed6e38ed64bf6a1f01000000000000000083106120d8577a184f03e93ff9f4daa797ed6e38ed64bf6a1f010000000000000000830492506040015b6d04ee2d6d415b85acef81000000008310612104576d04ee2d6d415b85acef8100000000830492506020015b662386f26fc10000831061212257662386f26fc10000830492506010015b6305f5e100831061213a576305f5e100830492506008015b612710831061214e57612710830492506004015b60648310612160576064830492506002015b600a8310610da65760010192915050565b6000611a5683836122cd565b6060600185858585604051602001612199959493929190612c08565b6040516020818303038152906040529050949350505050565b6000611a5683836123c0565b6000612220826040518060400160405280602081526020017f5361666545524332303a206c6f772d6c6576656c2063616c6c206661696c65648152508573ffffffffffffffffffffffffffffffffffffffff1661240f9092919063ffffffff16565b9050805160001480612241575080806020019051810190612241919061293c565b610eac576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152602a60248201527f5361666545524332303a204552433230206f7065726174696f6e20646964206e60448201527f6f7420737563636565640000000000000000000000000000000000000000000060648201526084016106a8565b600081815260018301602052604081205480156123b65760006122f1600183612c8d565b855490915060009061230590600190612c8d565b905081811461236a57600086600001828154811061232557612325612977565b906000526020600020015490508087600001848154811061234857612348612977565b6000918252602080832090910192909255918252600188019052604090208390555b855486908061237b5761237b612ca0565b600190038181906000526020600020016000905590558560010160008681526020019081526020016000206000905560019350505050610da6565b6000915050610da6565b600081815260018301602052604081205461240757508154600181810184556000848152602080822090930184905584548482528286019093526040902091909155610da6565b506000610da6565b60606107188484600085856000808673ffffffffffffffffffffffffffffffffffffffff1685876040516124439190612ccf565b60006040518083038185875af1925050503d8060008114612480576040519150601f19603f3d011682016040523d82523d6000602084013e612485565b606091505b50915091506108b487838387606083156125275782516000036125205773ffffffffffffffffffffffffffffffffffffffff85163b612520576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152601d60248201527f416464726573733a2063616c6c20746f206e6f6e2d636f6e747261637400000060448201526064016106a8565b5081610718565b610718838381511561253c5781518083602001fd5b806040517f08c379a00000000000000000000000000000000000000000000000000000000081526004016106a89190612abd565b803573ffffffffffffffffffffffffffffffffffffffff8116811461259457600080fd5b919050565b6000602082840312156125ab57600080fd5b611a5682612570565b803563ffffffff8116811461259457600080fd5b6000602082840312156125da57600080fd5b611a56826125b4565b6020808252825182820181905260009190848201906040850190845b8181101561262157835163ffffffff16835292840192918401916001016125ff565b50909695505050505050565b6000806040838503121561264057600080fd5b612649836125b4565b946020939093013593505050565b60008083601f84011261266957600080fd5b50813567ffffffffffffffff81111561268157600080fd5b6020830191508360208285010111156112d357600080fd5b60008060008060008060a087890312156126b257600080fd5b6126bb876125b4565b95506020870135945060408701359350606087013567ffffffffffffffff8111156126e557600080fd5b6126f189828a01612657565b9094509250612704905060808801612570565b90509295509295509295565b6000806000806060858703121561272657600080fd5b61272f856125b4565b935060208501359250604085013567ffffffffffffffff81111561275257600080fd5b61275e87828801612657565b95989497509550505050565b60008083601f84011261277c57600080fd5b50813567ffffffffffffffff81111561279457600080fd5b6020830191508360208260051b85010111156112d357600080fd5b600080602083850312156127c257600080fd5b823567ffffffffffffffff8111156127d957600080fd5b6127e58582860161276a565b90969095509350505050565b60008060006060848603121561280657600080fd5b61280f846125b4565b95602085013595506040909401359392505050565b6000806020838503121561283757600080fd5b823567ffffffffffffffff8082111561284f57600080fd5b818501915085601f83011261286357600080fd5b81358181111561287257600080fd5b8660208260061b850101111561288757600080fd5b60209290920196919550909350505050565b6000806000606084860312156128ae57600080fd5b6128b784612570565b92506128c560208501612570565b91506128d360408501612570565b90509250925092565b600080600080604085870312156128f257600080fd5b843567ffffffffffffffff8082111561290a57600080fd5b6129168883890161276a565b9096509450602087013591508082111561292f57600080fd5b5061275e8782880161276a565b60006020828403121561294e57600080fd5b81518015158114611a5657600080fd5b60006020828403121561297057600080fd5b5051919050565b7f4e487b7100000000000000000000000000000000000000000000000000000000600052603260045260246000fd5b7f4e487b7100000000000000000000000000000000000000000000000000000000600052601160045260246000fd5b80820180821115610da657610da66129a6565b7f4e487b7100000000000000000000000000000000000000000000000000000000600052604160045260246000fd5b60007fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff8203612a4857612a486129a6565b5060010190565b60005b83811015612a6a578181015183820152602001612a52565b50506000910152565b60008151808452612a8b816020860160208601612a4f565b601f017fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffe0169290920160200192915050565b602081526000611a566020830184612a73565b83815282602082015260008251612aee816040850160208701612a4f565b91909101604001949350505050565b63ffffffff8616815284602082015260a060408201526000612b2260a0830186612a73565b8281036060840152612b348186612a73565b91505073ffffffffffffffffffffffffffffffffffffffff831660808301529695505050505050565b7f4e6f20726f7574657220656e726f6c6c656420666f7220646f6d61696e3a2000815260008251612b9581601f850160208701612a4f565b91909101601f0192915050565b60008085851115612bb257600080fd5b83861115612bbf57600080fd5b5050820193919092039150565b80356020831015610da6577fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff602084900360031b1b1692915050565b7fffff0000000000000000000000000000000000000000000000000000000000008660f01b1681528460028201528360228201527fffffffffffffffffffffffffffffffffffffffff0000000000000000000000008360601b16604282015260008251612c7c816056850160208701612a4f565b919091016056019695505050505050565b81810381811115610da657610da66129a6565b7f4e487b7100000000000000000000000000000000000000000000000000000000600052603160045260246000fd5b60008251612ce1818460208701612a4f565b919091019291505056fea2646970667358221220863c53da358d51b7f81bd7e8d2ab4b730910c8213130443d0cd4172050ef15c864736f6c63430008130033
Constructor Arguments (ABI-Encoded and is the last bytes of the Contract Creation Code above)
000000000000000000000000c8140da31e6bca19b287cc35531c2212763c2059000000000000000000000000c005dc82818d67af737725bd4bf75435d065d239
-----Decoded View---------------
Arg [0] : _lockbox (address): 0xC8140dA31E6bCa19b287cC35531c2212763C2059
Arg [1] : _mailbox (address): 0xc005dc82818d67AF737725bD4bf75435d065D239
-----Encoded View---------------
2 Constructor Arguments found :
Arg [0] : 000000000000000000000000c8140da31e6bca19b287cc35531c2212763c2059
Arg [1] : 000000000000000000000000c005dc82818d67af737725bd4bf75435d065d239
Deployed Bytecode Sourcemap
259:2043:25:-:0;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;2265:191:14;;;;;;;;;;-1:-1:-1;2265:191:14;;;;;:::i;:::-;;:::i;:::-;;1454:158:15;;;;;;;;;;-1:-1:-1;1454:158:15;;;;;:::i;:::-;;:::i;:::-;;;909:25:30;;;897:2;882:18;1454:158:15;;;;;;;;404:31:25;;;;;;;;;;;;;;;;;;1137:42:30;1125:55;;;1107:74;;1095:2;1080:18;404:31:25;945:242:30;1954:123:14;;;;;;;;;;-1:-1:-1;1954:123:14;;;;;:::i;:::-;;:::i;1051:104:15:-;;;;;;;;;;;;;:::i;:::-;;;;;;;:::i;1048:122:13:-;;;;;;;;;;-1:-1:-1;1048:122:13;;;;;:::i;:::-;;:::i;3230:463:29:-;;;;;;:::i;:::-;;:::i;3450:338:15:-;;;;;;:::i;:::-;;:::i;897:307:25:-;;;;;;;;;;;;;:::i;359:39::-;;;;;;;;;;;;;;;1108:148:24;;;;;;;;;;-1:-1:-1;1108:148:24;;;;;:::i;:::-;;:::i;2064:101:0:-;;;;;;;;;;;;;:::i;3012:257:15:-;;;;;;;;;;-1:-1:-1;3012:257:15;;;;;:::i;:::-;;:::i;281:48:13:-;;;;;;;;;;-1:-1:-1;281:48:13;;;;;:::i;:::-;;;;;;;;;;;;;;770:29:14;;;;;;;;;;-1:-1:-1;770:29:14;;;;;;;;2152:270:29;;;;;;:::i;:::-;;:::i;728:35:14:-;;;;;;;;;;;;;;;;;;5708:10:30;5696:23;;;5678:42;;5666:2;5651:18;728:35:14;5534:192:30;1441:85:0;;;;;;;;;;-1:-1:-1;1513:6:0;;;;1441:85;;608:36:24;;;;;;;;;;;;;;;623:250:13;;;;;;;;;;-1:-1:-1;623:250:13;;;;;:::i;:::-;;:::i;2112:164:15:-;;;;;;;;;;-1:-1:-1;2112:164:15;;;;;:::i;:::-;;:::i;1429:212:25:-;;;;;;;;;;-1:-1:-1;1429:212:25;;;;;:::i;:::-;;:::i;688:33:14:-;;;;;;;;;;;;;;;806:57;;;;;;;;;;-1:-1:-1;806:57:14;;;;;;;;2493:373:15;;;;;;;;;;-1:-1:-1;2493:373:15;;;;;:::i;:::-;;:::i;1737:120::-;;;;;;;;;;-1:-1:-1;1737:120:15;;;;;:::i;:::-;;:::i;1437:185:13:-;;;;;;;;;;-1:-1:-1;1437:185:13;;;;;:::i;:::-;;:::i;2314:198:0:-;;;;;;;;;;-1:-1:-1;2314:198:0;;;;;:::i;:::-;;:::i;2265:191:14:-;2359:7;1702:19:7;;;;:23;;1230:56:14;;;-1:-1:-1;1263:23:14;;;;1230:56;1209:142;;;;;;;8948:2:30;1209:142:14;;;8930:21:30;8987:2;8967:18;;;8960:30;9026:34;9006:18;;;8999:62;9097:9;9077:18;;;9070:37;9124:19;;1209:142:14;;;;;;;;;1334:13:0::1;:11;:13::i;:::-;-1:-1:-1::0;2388:24:14::2;:61:::0;;;::::2;;::::0;;;::::2;::::0;;;::::2;::::0;;2265:191::o;1454:158:15:-;1516:7;;1557:24;:8;:24;;;;;:15;:24;:::i;:::-;1535:46;1454:158;-1:-1:-1;;;;1454:158:15:o;1954:123:14:-;2012:5;1702:19:7;;;;:23;;1230:56:14;;;-1:-1:-1;1263:23:14;;;;1230:56;1209:142;;;;;;;8948:2:30;1209:142:14;;;8930:21:30;8987:2;8967:18;;;8960:30;9026:34;9006:18;;;8999:62;9097:9;9077:18;;;9070:37;9124:19;;1209:142:14;8746:403:30;1209:142:14;1334:13:0::1;:11;:13::i;:::-;-1:-1:-1::0;2039:4:14::2;:31:::0;;;::::2;;::::0;;;::::2;::::0;;;::::2;::::0;;1954:123::o;1051:104:15:-;1093:15;1127:21;:8;:19;:21::i;:::-;1120:28;;1051:104;:::o;1048:122:13:-;1334:13:0;:11;:13::i;:::-;1917:22:13;;;;;;;;;:14;:22;;;;;:28;1048:122::o;1132:31::-:1;1048:122:::0;;:::o;3230:463:29:-;3441:17;3489:197;3522:12;3552:10;3580:11;3609:9;3636:13;;3489:197;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;-1:-1:-1;3667:5:29;;-1:-1:-1;3489:15:29;;-1:-1:-1;;3489:197:29:i;:::-;3470:216;3230:463;-1:-1:-1;;;;;;;3230:463:29:o;3450:338:15:-;1516:10:14;:30;1538:7;1516:30;;1495:110;;;;;;;9356:2:30;1495:110:14;;;9338:21:30;9395:2;9375:18;;;9368:30;9434:34;9414:18;;;9407:62;9505:3;9485:18;;;9478:31;9526:19;;1495:110:14;9154:397:30;1495:110:14;3610:15:15::1;3628:30;3650:7;3628:21;:30::i;:::-;3610:48;;3687:7;3676;:18;3668:68;;;::::0;::::1;::::0;;9758:2:30;3668:68:15::1;::::0;::::1;9740:21:30::0;9797:2;9777:18;;;9770:30;9836:34;9816:18;;;9809:62;9907:7;9887:18;;;9880:35;9932:19;;3668:68:15::1;9556:401:30::0;3668:68:15::1;3746:35;3754:7;3763;3772:8;;3746:7;:35::i;:::-;3600:188;3450:338:::0;;;;:::o;897:307:25:-;961:55;;;;;:28;998:7;10154:55:30;;961::25;;;10136:74:30;340:12:25;10226:18:30;;;10219:34;968:12:25;961:28;;;;10109:18:30;;961:55:25;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;940:130;;;;;;;10748:2:30;940:130:25;;;10730:21:30;10787:2;10767:18;;;10760:30;10826;10806:18;;;10799:58;10874:18;;940:130:25;10546:352:30;940:130:25;1101:41;;;;;:14;1124:7;10154:55:30;;1101:41:25;;;10136:74:30;340:12:25;10226:18:30;;;10219:34;1101:6:25;:14;;;;10109:18:30;;1101:41:25;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;1080:117;;;;;;;11105:2:30;1080:117:25;;;11087:21:30;11144:2;11124:18;;;11117:30;11183:31;11163:18;;;11156:59;11232:18;;1080:117:25;10903:353:30;1080:117:25;897:307::o;1108:148:24:-;1217:32;;;;;:22;1125:55:30;;;1217:32:24;;;1107:74:30;1191:7:24;;1217:12;:22;;;;;;1080:18:30;;1217:32:24;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;1210:39;1108:148;-1:-1:-1;;1108:148:24:o;2064:101:0:-;1334:13;:11;:13::i;:::-;2128:30:::1;2155:1;2128:18;:30::i;3012:257:15:-:0;1334:13:0;:11;:13::i;:::-;3139:8:15;3122:14:::1;3164:99;3188:6;3184:1;:10;3164:99;;;3218:34;3240:8;;3249:1;3240:11;;;;;;;:::i;:::-;;;;;;;;;;;;;;:::i;:::-;3218:21;:34::i;:::-;3196:6;3201:1;3196:6:::0;::::1;:::i;:::-;;;3164:99;;;;3112:157;3012:257:::0;;:::o;2152:270:29:-;2302:17;2350:65;2366:12;2380:10;2392:11;2405:9;2350:15;:65::i;623:250:13:-;1334:13:0;:11;:13::i;:::-;737:9:13::1;732:135;752:21:::0;;::::1;732:135;;;797:59;816:10;;827:1;816:13;;;;;;;:::i;:::-;:20;::::0;::::1;:13;::::0;;::::1;;:20:::0;;::::1;::::0;-1:-1:-1;816:20:13::1;:::i;:::-;838:10;;849:1;838:13;;;;;;;:::i;:::-;;;;;;:17;;;1917:22:::0;;;;;;;;:14;:22;;;;;:28;1842:110;797:59:::1;775:6;780:1;775:6:::0;::::1;:::i;:::-;;;732:135;;;;623:250:::0;;:::o;2112:164:15:-;1334:13:0;:11;:13::i;:::-;2232:37:15::1;2252:7;2261;2232:19;:37::i;1429:212:25:-:0;3279:19:1;3302:13;;;;;;3301:14;;3347:34;;;;-1:-1:-1;3365:12:1;;3380:1;3365:12;;;;:16;3347:34;3346:108;;;-1:-1:-1;3426:4:1;1702:19:7;:23;;;3387:66:1;;-1:-1:-1;3436:12:1;;;;;:17;3387:66;3325:201;;;;;;;12160:2:30;3325:201:1;;;12142:21:30;12199:2;12179:18;;;12172:30;12238:34;12218:18;;;12211:62;12309:16;12289:18;;;12282:44;12343:19;;3325:201:1;11958:410:30;3325:201:1;3536:12;:16;;;;3551:1;3536:16;;;3562:65;;;;3596:13;:20;;;;;;;;3562:65;1562:16:25::1;:14;:16::i;:::-;1588:46;1614:5;1621:4;1627:6;1588:25;:46::i;:::-;3651:14:1::0;3647:99;;;3697:5;3681:21;;;;;;3721:14;;-1:-1:-1;12525:36:30;;3721:14:1;;12513:2:30;12498:18;3721:14:1;;;;;;;3269:483;1429:212:25;;;:::o;2493:373:15:-;1334:13:0;:11;:13::i;:::-;2648:36:15;;::::1;2640:56;;;::::0;::::1;::::0;;12774:2:30;2640:56:15::1;::::0;::::1;12756:21:30::0;12813:1;12793:18;;;12786:29;12851:9;12831:18;;;12824:37;12878:18;;2640:56:15::1;12572:330:30::0;2640:56:15::1;2723:8:::0;2706:14:::1;2748:112;2772:6;2768:1;:10;2748:112;;;2802:47;2822:8;;2831:1;2822:11;;;;;;;:::i;:::-;;;;;;;;;;;;;;:::i;:::-;2835:10;;2846:1;2835:13;;;;;;;:::i;:::-;;;;;;;2802:19;:47::i;:::-;2780:6;2785:1;2780:6:::0;::::1;:::i;:::-;;;2748:112;;;;2630:236;2493:373:::0;;;;:::o;1737:120::-;1334:13:0;:11;:13::i;:::-;1820:30:15::1;1842:7;1820:21;:30::i;:::-;1737:120:::0;:::o;1437:185:13:-;1552:63;;;;;;;;;1526:7;1552:63;;;1609:4;;1526:7;;1552:63;;1577:18;;1552:63;1609:4;;1552:24;:63::i;2314:198:0:-;1334:13;:11;:13::i;:::-;2402:22:::1;::::0;::::1;2394:73;;;::::0;::::1;::::0;;13109:2:30;2394:73:0::1;::::0;::::1;13091:21:30::0;13148:2;13128:18;;;13121:30;13187:34;13167:18;;;13160:62;13258:8;13238:18;;;13231:36;13284:19;;2394:73:0::1;12907:402:30::0;2394:73:0::1;2477:28;2496:8;2477:18;:28::i;1599:130::-:0;1513:6;;1662:23;1513:6;929:10:3;1662:23:0;1654:68;;;;;;;13516:2:30;1654:68:0;;;13498:21:30;;;13535:18;;;13528:30;13594:34;13574:18;;;13567:62;13646:18;;1654:68:0;13314:356:30;1719:174:21;1823:4;;1855:31;:3;1881;1855:17;:31::i;:::-;1848:38;;;;1719:174;;;;;;:::o;1044:333::-;1131:21;1164:28;1195:9;1200:3;1195:4;:9::i;:::-;1164:40;;1235:11;:18;1222:32;;;;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;-1:-1:-1;1222:32:21;;1214:40;;1269:9;1264:107;1288:11;:18;1284:1;:22;1264:107;;;1345:11;1357:1;1345:14;;;;;;;;:::i;:::-;;;;;;;1327:5;1333:1;1327:8;;;;;;;;:::i;:::-;:33;;;;:8;;;;;;;;;;;:33;1308:3;;;;:::i;:::-;;;;1264:107;;;;1154:223;1044:333;;;:::o;4145:710:29:-;4371:17;4400:27;4430:32;4450:11;4430:19;:32::i;:::-;4400:62;;4472:26;4501:106;4534:10;4558:11;4583:14;4501:19;:106::i;:::-;4472:135;;4630:145;4660:12;4686:6;4706:13;4733;4760:5;4630:16;:145::i;:::-;4618:157;;4824:10;4810:12;4791:57;;;4836:11;4791:57;;;;909:25:30;;897:2;882:18;;763:177;4791:57:29;;;;;;;;4390:465;;4145:710;;;;;;;;:::o;5294:253:15:-;5378:7;;;5433:24;:8;:24;;;;;:15;:24;:::i;:::-;5397:60;;;;5475:9;5486:29;5507:7;5486:20;:29::i;:::-;5467:49;;;;;;;;;;;;;;:::i;:::-;-1:-1:-1;5533:7:15;5294:253;-1:-1:-1;;;5294:253:15:o;5841:413:29:-;5974:17;5994:20;:8;;:18;:20::i;:::-;5974:40;;6024:14;6041:17;:8;;:15;:17::i;:::-;6024:34;;6068:23;;6094:19;:8;;:17;:19::i;:::-;6068:45;;;;6123:59;6135:28;:9;411:4:23;298:127;6135:28:29;6165:6;6173:8;;6123:11;:59::i;:::-;6229:9;6220:7;6197:50;;;6240:6;6197:50;;;;909:25:30;;897:2;882:18;;763:177;6197:50:29;;;;;;;;5964:290;;;;5841:413;;;;:::o;2666:187:0:-;2758:6;;;;2774:17;;;;;;;;;;;2806:40;;2758:6;;;2774:17;2758:6;;2806:40;;2739:16;;2806:40;2729:124;2666:187;:::o;4419:145:15:-;4501:24;:8;:24;;;;;:15;:24;:::i;:::-;4527:29;4548:7;4527:20;:29::i;:::-;4493:64;;;;;;;;;;;;;;:::i;3699:440:29:-;3858:17;3906:226;3939:12;3969:10;3997:11;4026:6;4050:37;4074:12;4050:23;:37::i;:::-;4113:4;;;;3906:15;:226::i;:::-;3887:245;3699:440;-1:-1:-1;;;;;3699:440:29:o;4162:150:15:-;4274:31;:8;:31;;;;;4296:8;;4274:12;:31;:::i;2500:314:14:-;5374:13:1;;;;;;;5366:69;;;;;;;15081:2:30;5366:69:1;;;15063:21:30;15120:2;15100:18;;;15093:30;15159:34;15139:18;;;15132:62;15230:13;15210:18;;;15203:41;15261:19;;5366:69:1;14879:407:30;5366:69:1;2667:16:14::1;:14;:16::i;:::-;2693:14;2701:5;2693:7;:14::i;:::-;2717:54;2745:25;2717:27;:54::i;:::-;2781:26;2800:6;2781:18;:26::i;2365:368:13:-:0;2515:7;2553:173;2592:12;2622;2652:37;2676:12;2652:23;:37::i;:::-;2707:5;2553:21;:173::i;4166:298:11:-;4251:4;4292:16;;;:11;;;:16;;;;;;4251:4;;4292:16;4318:140;;4365:18;4374:3;4379;4365:8;:18::i;:::-;4357:39;-1:-1:-1;4393:1:11;;-1:-1:-1;4357:39:11;;-1:-1:-1;4357:39:11;4318:140;4435:4;;-1:-1:-1;4441:5:11;-1:-1:-1;4427:20:11;;725:313:21;806:22;840:15;858:19;:3;:17;:19::i;:::-;840:37;;909:7;895:22;;;;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;-1:-1:-1;895:22:21;;887:30;;932:9;927:105;951:7;947:1;:11;927:105;;;998:22;:3;1018:1;998:19;:22::i;:::-;990:31;;979:5;985:1;979:8;;;;;;;;:::i;:::-;;;;;;;;;;:42;960:3;;;;:::i;:::-;;;;927:105;;1647:353:25;1734:12;1796:34;1822:7;1796:25;:34::i;:::-;-1:-1:-1;1875:24:25;;;;;;;;909:25:30;;;1875:7:25;:15;;;;;882:18:30;;1875:24:25;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;-1:-1:-1;;1932:35:25;;;;;1952:4;1932:35;;;10136:74:30;10226:18;;;10219:34;;;1932:6:25;:11;;;-1:-1:-1;1932:11:25;;-1:-1:-1;10109:18:30;;1932:35:25;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;-1:-1:-1;;1984:9:25;;;;;;;;;-1:-1:-1;1984:9:25;;;1647:353;-1:-1:-1;;;;1647:353:25:o;99:216:28:-;229:12;277:10;289:7;298:9;260:48;;;;;;;;;;:::i;:::-;;;;;;;;;;;;;253:55;;99:216;;;;;:::o;5807:526:15:-;6010:7;6029:15;6047:41;6069:18;6047:21;:41::i;:::-;6029:59;;6117:7;:16;;;6141:6;6166:18;6202:7;6227:12;6257:13;6306:5;6117:209;;;;;;;;;;;;;;;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;:::i;5553:248::-;5636:13;5762:18;:7;:16;;;:18::i;:::-;5680:114;;;;;;;;:::i;:::-;;;;;;;;;;;;;5661:133;;5553:248;;;:::o;321:121:28:-;387:7;421:13;431:2;387:7;421;;:13;:::i;:::-;413:22;;;:::i;:::-;406:29;321:121;-1:-1:-1;;;321:121:28:o;448:128::-;511:7;553:14;564:2;561;553:7;;:14;:::i;724:131::-;803:14;;836:12;:7;844:2;836:7;;:12;:::i;2006:294:25:-;2174:35;;;;;2194:4;2174:35;;;10136:74:30;10226:18;;;10219:34;;;2174:6:25;:11;;;;;10109:18:30;;2174:35:25;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;-1:-1:-1;;2254:39:25;;;;;:18;10154:55:30;;;2254:39:25;;;10136:74:30;10226:18;;;10219:34;;;2254:7:25;:18;;-1:-1:-1;2254:18:25;;-1:-1:-1;10109:18:30;;2254:39:25;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;2006:294;;;;:::o;1899:160:21:-;1998:4;2021:31;:3;2047;2021:17;:31::i;1628:208:13:-;1800:28;;;;;;;:14;:28;;;;;;1719:12;;1762:67;;:37;:67::i;1383:162:21:-;1503:35;:3;1526;1532:5;1503:14;:35::i;1003:95:0:-;5374:13:1;;;;;;;5366:69;;;;;;;15081:2:30;5366:69:1;;;15063:21:30;15120:2;15100:18;;;15093:30;15159:34;15139:18;;;15132:62;15230:13;15210:18;;;15203:41;15261:19;;5366:69:1;14879:407:30;5366:69:1;1065:26:0::1;:24;:26::i;6807:502:15:-:0;6996:7;7015:15;7033:41;7055:18;7033:21;:41::i;:::-;7103:199;;;;;7015:59;;-1:-1:-1;7103:21:15;:7;:21;;;;:199;;7142:18;;7015:59;;7203:12;;7233:13;;7282:5;;7103:199;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;7084:218;6807:502;-1:-1:-1;;;;;;6807:502:15:o;3128:140:11:-;3215:4;3238:23;:3;3257;3238:18;:23::i;3358:123::-;3430:7;3456:18;:3;:16;:18::i;7096:129:12:-;7170:7;7196:22;7200:3;7212:5;7196:3;:22::i;1395:232:24:-;1490:12;1514:65;:29;:12;:29;1544:10;1564:4;1571:7;1514:29;:65::i;:::-;-1:-1:-1;;1596:9:24;;;;;;;;;-1:-1:-1;1596:9:24;;;1395:232::o;447:696:8:-;503:13;552:14;569:17;580:5;569:10;:17::i;:::-;589:1;569:21;552:38;;604:20;638:6;627:18;;;;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;-1:-1:-1;627:18:8;-1:-1:-1;604:41:8;-1:-1:-1;765:28:8;;;781:2;765:28;820:280;851:5;;990:8;985:2;974:14;;969:30;851:5;956:44;1044:2;1035:11;;;-1:-1:-1;1064:21:8;820:280;1064:21;-1:-1:-1;1120:6:8;447:696;-1:-1:-1;;;447:696:8:o;2885:164:11:-;2965:4;2988:16;;;:11;;;:16;;;;;2981:23;;;3021:21;2988:3;3000;3021:16;:21::i;4969:173:16:-;5051:12;5082:53;5105:1;5109:9;5120:10;5082:53;;;;;;;;;;;;:14;:53::i;2543:174:11:-;2635:4;2651:16;;;:11;;;:16;;;;;:24;;;2692:18;2651:3;2663;2692:13;:18::i;1104:111:0:-;5374:13:1;;;;;;;5366:69;;;;;;;15081:2:30;5366:69:1;;;15063:21:30;15120:2;15100:18;;;15093:30;15159:34;15139:18;;;15132:62;15230:13;15210:18;;;15203:41;15261:19;;5366:69:1;14879:407:30;5366:69:1;1176:32:0::1;929:10:3::0;1176:18:0::1;:32::i;6420:138:12:-:0;6500:4;4351:19;;;:12;;;:19;;;;;;:24;;6523:28;4255:127;6639:115;6702:7;6728:19;6736:3;4545:18;;4463:107;4912:118;4979:7;5005:3;:11;;5017:5;5005:18;;;;;;;;:::i;:::-;;;;;;;;;4998:25;;4912:118;;;;:::o;1355:203:6:-;1482:68;;;18114:42:30;18183:15;;;1482:68:6;;;18165:34:30;18235:15;;18215:18;;;18208:43;18267:18;;;;18260:34;;;1482:68:6;;;;;;;;;;18077:18:30;;;;1482:68:6;;;;;;;;;;1505:27;1482:68;;;1455:96;;1475:5;;1455:19;:96::i;10139:916:9:-;10192:7;;10276:8;10267:17;;10263:103;;10313:8;10304:17;;;-1:-1:-1;10349:2:9;10339:12;10263:103;10392:8;10383:5;:17;10379:103;;10429:8;10420:17;;;-1:-1:-1;10465:2:9;10455:12;10379:103;10508:8;10499:5;:17;10495:103;;10545:8;10536:17;;;-1:-1:-1;10581:2:9;10571:12;10495:103;10624:7;10615:5;:16;10611:100;;10660:7;10651:16;;;-1:-1:-1;10695:1:9;10685:11;10611:100;10737:7;10728:5;:16;10724:100;;10773:7;10764:16;;;-1:-1:-1;10808:1:9;10798:11;10724:100;10850:7;10841:5;:16;10837:100;;10886:7;10877:16;;;-1:-1:-1;10921:1:9;10911:11;10837:100;10963:7;10954:5;:16;10950:66;;11000:1;10990:11;11042:6;10139:916;-1:-1:-1;;10139:916:9:o;6210:129:12:-;6283:4;6306:26;6314:3;6326:5;6306:7;:26::i;3957:401:16:-;4134:12;1025:1;4236:9;4263;4290:14;4322:15;4177:174;;;;;;;;;;;;:::i;:::-;;;;;;;;;;;;;4158:193;;3957:401;;;;;;:::o;5919:123:12:-;5989:4;6012:23;6017:3;6029:5;6012:4;:23::i;5196:642:6:-;5615:23;5641:69;5669:4;5641:69;;;;;;;;;;;;;;;;;5649:5;5641:27;;;;:69;;;;;:::i;:::-;5615:95;;5728:10;:17;5749:1;5728:22;:56;;;;5765:10;5754:30;;;;;;;;;;;;:::i;:::-;5720:111;;;;;;;19230:2:30;5720:111:6;;;19212:21:30;19269:2;19249:18;;;19242:30;19308:34;19288:18;;;19281:62;19379:12;19359:18;;;19352:40;19409:19;;5720:111:6;19028:406:30;2786:1388:12;2852:4;2989:19;;;:12;;;:19;;;;;;3023:15;;3019:1149;;3392:21;3416:14;3429:1;3416:10;:14;:::i;:::-;3464:18;;3392:38;;-1:-1:-1;3444:17:12;;3464:22;;3485:1;;3464:22;:::i;:::-;3444:42;;3518:13;3505:9;:26;3501:398;;3551:17;3571:3;:11;;3583:9;3571:22;;;;;;;;:::i;:::-;;;;;;;;;3551:42;;3722:9;3693:3;:11;;3705:13;3693:26;;;;;;;;:::i;:::-;;;;;;;;;;;;:38;;;;3805:23;;;:12;;;:23;;;;;:36;;;3501:398;3977:17;;:3;;:17;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;4069:3;:12;;:19;4082:5;4069:19;;;;;;;;;;;4062:26;;;4110:4;4103:11;;;;;;;3019:1149;4152:5;4145:12;;;;;2214:404;2277:4;4351:19;;;:12;;;:19;;;;;;2293:319;;-1:-1:-1;2335:23:12;;;;;;;;:11;:23;;;;;;;;;;;;;2515:18;;2493:19;;;:12;;;:19;;;;;;:40;;;;2547:11;;2293:319;-1:-1:-1;2596:5:12;2589:12;;4108:223:7;4241:12;4272:52;4294:6;4302:4;4308:1;4311:12;4241;5446;5460:23;5487:6;:11;;5506:5;5513:4;5487:31;;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;5445:73;;;;5535:69;5562:6;5570:7;5579:10;5591:12;7851;7879:7;7875:418;;;7906:10;:17;7927:1;7906:22;7902:286;;1702:19;;;;8113:60;;;;;;;20662:2:30;8113:60:7;;;20644:21:30;20701:2;20681:18;;;20674:30;20740:31;20720:18;;;20713:59;20789:18;;8113:60:7;20460:353:30;8113:60:7;-1:-1:-1;8208:10:7;8201:17;;7875:418;8249:33;8257:10;8269:12;8980:17;;:21;8976:379;;9208:10;9202:17;9264:15;9251:10;9247:2;9243:19;9236:44;8976:379;9331:12;9324:20;;;;;;;;;;;:::i;14:196:30:-;82:20;;142:42;131:54;;121:65;;111:93;;200:1;197;190:12;111:93;14:196;;;:::o;215:186::-;274:6;327:2;315:9;306:7;302:23;298:32;295:52;;;343:1;340;333:12;295:52;366:29;385:9;366:29;:::i;406:163::-;473:20;;533:10;522:22;;512:33;;502:61;;559:1;556;549:12;574:184;632:6;685:2;673:9;664:7;660:23;656:32;653:52;;;701:1;698;691:12;653:52;724:28;742:9;724:28;:::i;1192:647::-;1361:2;1413:21;;;1483:13;;1386:18;;;1505:22;;;1332:4;;1361:2;1584:15;;;;1558:2;1543:18;;;1332:4;1627:186;1641:6;1638:1;1635:13;1627:186;;;1706:13;;1721:10;1702:30;1690:43;;1788:15;;;;1753:12;;;;1663:1;1656:9;1627:186;;;-1:-1:-1;1830:3:30;;1192:647;-1:-1:-1;;;;;;1192:647:30:o;1844:252::-;1911:6;1919;1972:2;1960:9;1951:7;1947:23;1943:32;1940:52;;;1988:1;1985;1978:12;1940:52;2011:28;2029:9;2011:28;:::i;:::-;2001:38;2086:2;2071:18;;;;2058:32;;-1:-1:-1;;;1844:252:30:o;2101:347::-;2152:8;2162:6;2216:3;2209:4;2201:6;2197:17;2193:27;2183:55;;2234:1;2231;2224:12;2183:55;-1:-1:-1;2257:20:30;;2300:18;2289:30;;2286:50;;;2332:1;2329;2322:12;2286:50;2369:4;2361:6;2357:17;2345:29;;2421:3;2414:4;2405:6;2397;2393:19;2389:30;2386:39;2383:59;;;2438:1;2435;2428:12;2453:693;2558:6;2566;2574;2582;2590;2598;2651:3;2639:9;2630:7;2626:23;2622:33;2619:53;;;2668:1;2665;2658:12;2619:53;2691:28;2709:9;2691:28;:::i;:::-;2681:38;;2766:2;2755:9;2751:18;2738:32;2728:42;;2817:2;2806:9;2802:18;2789:32;2779:42;;2872:2;2861:9;2857:18;2844:32;2899:18;2891:6;2888:30;2885:50;;;2931:1;2928;2921:12;2885:50;2970:58;3020:7;3011:6;3000:9;2996:22;2970:58;:::i;:::-;3047:8;;-1:-1:-1;2944:84:30;-1:-1:-1;3101:39:30;;-1:-1:-1;3135:3:30;3120:19;;3101:39;:::i;:::-;3091:49;;2453:693;;;;;;;;:::o;3151:549::-;3238:6;3246;3254;3262;3315:2;3303:9;3294:7;3290:23;3286:32;3283:52;;;3331:1;3328;3321:12;3283:52;3354:28;3372:9;3354:28;:::i;:::-;3344:38;;3429:2;3418:9;3414:18;3401:32;3391:42;;3484:2;3473:9;3469:18;3456:32;3511:18;3503:6;3500:30;3497:50;;;3543:1;3540;3533:12;3497:50;3582:58;3632:7;3623:6;3612:9;3608:22;3582:58;:::i;:::-;3151:549;;;;-1:-1:-1;3659:8:30;-1:-1:-1;;;;3151:549:30:o;4141:366::-;4203:8;4213:6;4267:3;4260:4;4252:6;4248:17;4244:27;4234:55;;4285:1;4282;4275:12;4234:55;-1:-1:-1;4308:20:30;;4351:18;4340:30;;4337:50;;;4383:1;4380;4373:12;4337:50;4420:4;4412:6;4408:17;4396:29;;4480:3;4473:4;4463:6;4460:1;4456:14;4448:6;4444:27;4440:38;4437:47;4434:67;;;4497:1;4494;4487:12;4512:435;4597:6;4605;4658:2;4646:9;4637:7;4633:23;4629:32;4626:52;;;4674:1;4671;4664:12;4626:52;4714:9;4701:23;4747:18;4739:6;4736:30;4733:50;;;4779:1;4776;4769:12;4733:50;4818:69;4879:7;4870:6;4859:9;4855:22;4818:69;:::i;:::-;4906:8;;4792:95;;-1:-1:-1;4512:435:30;-1:-1:-1;;;;4512:435:30:o;5209:320::-;5285:6;5293;5301;5354:2;5342:9;5333:7;5329:23;5325:32;5322:52;;;5370:1;5367;5360:12;5322:52;5393:28;5411:9;5393:28;:::i;:::-;5383:38;5468:2;5453:18;;5440:32;;-1:-1:-1;5519:2:30;5504:18;;;5491:32;;5209:320;-1:-1:-1;;;5209:320:30:o;6207:650::-;6328:6;6336;6389:2;6377:9;6368:7;6364:23;6360:32;6357:52;;;6405:1;6402;6395:12;6357:52;6445:9;6432:23;6474:18;6515:2;6507:6;6504:14;6501:34;;;6531:1;6528;6521:12;6501:34;6569:6;6558:9;6554:22;6544:32;;6614:7;6607:4;6603:2;6599:13;6595:27;6585:55;;6636:1;6633;6626:12;6585:55;6676:2;6663:16;6702:2;6694:6;6691:14;6688:34;;;6718:1;6715;6708:12;6688:34;6771:7;6766:2;6756:6;6753:1;6749:14;6745:2;6741:23;6737:32;6734:45;6731:65;;;6792:1;6789;6782:12;6731:65;6823:2;6815:11;;;;;6845:6;;-1:-1:-1;6207:650:30;;-1:-1:-1;;;;6207:650:30:o;7119:334::-;7196:6;7204;7212;7265:2;7253:9;7244:7;7240:23;7236:32;7233:52;;;7281:1;7278;7271:12;7233:52;7304:29;7323:9;7304:29;:::i;:::-;7294:39;;7352:38;7386:2;7375:9;7371:18;7352:38;:::i;:::-;7342:48;;7409:38;7443:2;7432:9;7428:18;7409:38;:::i;:::-;7399:48;;7119:334;;;;;:::o;7971:770::-;8092:6;8100;8108;8116;8169:2;8157:9;8148:7;8144:23;8140:32;8137:52;;;8185:1;8182;8175:12;8137:52;8225:9;8212:23;8254:18;8295:2;8287:6;8284:14;8281:34;;;8311:1;8308;8301:12;8281:34;8350:69;8411:7;8402:6;8391:9;8387:22;8350:69;:::i;:::-;8438:8;;-1:-1:-1;8324:95:30;-1:-1:-1;8526:2:30;8511:18;;8498:32;;-1:-1:-1;8542:16:30;;;8539:36;;;8571:1;8568;8561:12;8539:36;;8610:71;8673:7;8662:8;8651:9;8647:24;8610:71;:::i;10264:277::-;10331:6;10384:2;10372:9;10363:7;10359:23;10355:32;10352:52;;;10400:1;10397;10390:12;10352:52;10432:9;10426:16;10485:5;10478:13;10471:21;10464:5;10461:32;10451:60;;10507:1;10504;10497:12;11261:184;11331:6;11384:2;11372:9;11363:7;11359:23;11355:32;11352:52;;;11400:1;11397;11390:12;11352:52;-1:-1:-1;11423:16:30;;11261:184;-1:-1:-1;11261:184:30:o;11450:::-;11502:77;11499:1;11492:88;11599:4;11596:1;11589:15;11623:4;11620:1;11613:15;11639:184;11691:77;11688:1;11681:88;11788:4;11785:1;11778:15;11812:4;11809:1;11802:15;11828:125;11893:9;;;11914:10;;;11911:36;;;11927:18;;:::i;13675:184::-;13727:77;13724:1;13717:88;13824:4;13821:1;13814:15;13848:4;13845:1;13838:15;13864:195;13903:3;13934:66;13927:5;13924:77;13921:103;;14004:18;;:::i;:::-;-1:-1:-1;14051:1:30;14040:13;;13864:195::o;14064:250::-;14149:1;14159:113;14173:6;14170:1;14167:13;14159:113;;;14249:11;;;14243:18;14230:11;;;14223:39;14195:2;14188:10;14159:113;;;-1:-1:-1;;14306:1:30;14288:16;;14281:27;14064:250::o;14319:330::-;14361:3;14399:5;14393:12;14426:6;14421:3;14414:19;14442:76;14511:6;14504:4;14499:3;14495:14;14488:4;14481:5;14477:16;14442:76;:::i;:::-;14563:2;14551:15;14568:66;14547:88;14538:98;;;;14638:4;14534:109;;14319:330;-1:-1:-1;;14319:330:30:o;14654:220::-;14803:2;14792:9;14785:21;14766:4;14823:45;14864:2;14853:9;14849:18;14841:6;14823:45;:::i;15291:424::-;15506:6;15501:3;15494:19;15543:6;15538:2;15533:3;15529:12;15522:28;15476:3;15579:6;15573:13;15595:73;15661:6;15656:2;15651:3;15647:12;15642:2;15634:6;15630:15;15595:73;:::i;:::-;15688:16;;;;15706:2;15684:25;;15291:424;-1:-1:-1;;;;15291:424:30:o;15720:685::-;16033:10;16025:6;16021:23;16010:9;16003:42;16081:6;16076:2;16065:9;16061:18;16054:34;16124:3;16119:2;16108:9;16104:18;16097:31;15984:4;16151:46;16192:3;16181:9;16177:19;16169:6;16151:46;:::i;:::-;16245:9;16237:6;16233:22;16228:2;16217:9;16213:18;16206:50;16273:33;16299:6;16291;16273:33;:::i;:::-;16265:41;;;16355:42;16347:6;16343:55;16337:3;16326:9;16322:19;16315:84;15720:685;;;;;;;;:::o;16599:453::-;16851:33;16846:3;16839:46;16821:3;16914:6;16908:13;16930:75;16998:6;16993:2;16988:3;16984:12;16977:4;16969:6;16965:17;16930:75;:::i;:::-;17025:16;;;;17043:2;17021:25;;16599:453;-1:-1:-1;;16599:453:30:o;17057:331::-;17162:9;17173;17215:8;17203:10;17200:24;17197:44;;;17237:1;17234;17227:12;17197:44;17266:6;17256:8;17253:20;17250:40;;;17286:1;17283;17276:12;17250:40;-1:-1:-1;;17312:23:30;;;17357:25;;;;;-1:-1:-1;17057:331:30:o;17393:315::-;17513:19;;17552:2;17544:11;;17541:161;;;17624:66;17613:2;17609:12;;;17606:1;17602:20;17598:93;17587:105;17393:315;;;;:::o;18305:718::-;18596:66;18587:6;18582:3;18578:16;18574:89;18569:3;18562:102;18693:6;18689:1;18684:3;18680:11;18673:27;18730:6;18725:2;18720:3;18716:12;18709:28;18788:66;18779:6;18775:2;18771:15;18767:88;18762:2;18757:3;18753:12;18746:110;18544:3;18885:6;18879:13;18901:75;18969:6;18964:2;18959:3;18955:12;18948:4;18940:6;18936:17;18901:75;:::i;:::-;18996:16;;;;19014:2;18992:25;;18305:718;-1:-1:-1;;;;;;18305:718:30:o;19439:128::-;19506:9;;;19527:11;;;19524:37;;;19541:18;;:::i;19572:184::-;19624:77;19621:1;19614:88;19721:4;19718:1;19711:15;19745:4;19742:1;19735:15;20168:287;20297:3;20335:6;20329:13;20351:66;20410:6;20405:3;20398:4;20390:6;20386:17;20351:66;:::i;:::-;20433:16;;;;;20168:287;-1:-1:-1;;20168:287:30:o
Swarm Source
ipfs://863c53da358d51b7f81bd7e8d2ab4b730910c8213130443d0cd4172050ef15c8
Loading...
Loading
Loading...
Loading
Multichain Portfolio | 30 Chains
Chain | Token | Portfolio % | Price | Amount | Value |
---|
Loading...
Loading
[ 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.