Overview
ETH Balance
0.00016441001 ETH
Eth Value
$0.55 (@ $3,343.89/ETH)More Info
Private Name Tags
ContractCreator
Transaction Hash |
Method
|
Block
|
From
|
To
|
|||||
---|---|---|---|---|---|---|---|---|---|
0xf49193a17c369127ccf8462a85c12bc58be1075d79e30b8ca009beed86895b3a | Send Mail | (pending) | 11 mins ago | IN | 0 ETH | (Pending) | |||
0x127fd55b672a8ab6c89f9f0af014be654e1af75790b2fdca0f1f2ea3542d49d2 | Send Mail | (pending) | 15 days ago | IN | 0 ETH | (Pending) | |||
Send Mail | 21211314 | 35 days ago | IN | 0 ETH | 0.00177546 | ||||
Send Mail | 21100507 | 50 days ago | IN | 0 ETH | 0.00055933 | ||||
Send Mail | 20760931 | 98 days ago | IN | 0 ETH | 0.00008411 | ||||
Send Mail | 20759124 | 98 days ago | IN | 0 ETH | 0.0000726 | ||||
Send Mail | 20758853 | 98 days ago | IN | 0 ETH | 0.00009808 | ||||
Send Mail | 20758847 | 98 days ago | IN | 0 ETH | 0.00009635 | ||||
Send Mail | 20758838 | 98 days ago | IN | 0 ETH | 0.00009807 | ||||
Send Mail | 20757664 | 98 days ago | IN | 0 ETH | 0.00015061 | ||||
Send Mail | 20757621 | 98 days ago | IN | 0 ETH | 0.00015028 | ||||
Send Mail | 20757595 | 98 days ago | IN | 0 ETH | 0.00015717 | ||||
Send Mail | 20757264 | 98 days ago | IN | 0 ETH | 0.00014988 | ||||
Send Mail | 20756694 | 98 days ago | IN | 0 ETH | 0.00013645 | ||||
Send Mail | 20755779 | 98 days ago | IN | 0 ETH | 0.00010454 | ||||
Send Mail | 20755484 | 99 days ago | IN | 0 ETH | 0.00011288 | ||||
Send Mail | 20755461 | 99 days ago | IN | 0 ETH | 0.00011737 | ||||
Send Mail | 20754787 | 99 days ago | IN | 0 ETH | 0.00011623 | ||||
Send Mail | 20752461 | 99 days ago | IN | 0 ETH | 0.00009287 | ||||
Send Mail | 20752261 | 99 days ago | IN | 0 ETH | 0.00006736 | ||||
Send Mail | 20752242 | 99 days ago | IN | 0 ETH | 0.00007419 | ||||
Send Mail | 20752242 | 99 days ago | IN | 0 ETH | 0.00007296 | ||||
Send Mail | 20751868 | 99 days ago | IN | 0 ETH | 0.000095 | ||||
Send Mail | 20749733 | 99 days ago | IN | 0 ETH | 0.00014188 | ||||
Send Mail | 20748616 | 99 days ago | IN | 0 ETH | 0.00012541 |
Latest 1 internal transaction
Advanced mode:
Parent Transaction Hash | Block |
From
|
To
|
|||
---|---|---|---|---|---|---|
16839028 | 648 days ago | Contract Creation | 0 ETH |
Loading...
Loading
Contract Name:
CrossChainMailer
Compiler Version
v0.8.16+commit.07a7930e
Optimization Enabled:
Yes with 200 runs
Other Settings:
default evmVersion
Contract Source Code (Solidity Standard Json-Input format)
// SPDX-License-Identifier: UNLICENSED pragma solidity ^0.8.16; import {FeeCollector} from "contracts/src/utils/FeeCollector.sol"; import {ENSHelper} from "contracts/src/utils/ENSHelper.sol"; import {StringHelper} from "contracts/src/utils/StringHelper.sol"; import {ITelepathyRouter} from "telepathy-contracts/amb/interfaces/ITelepathy.sol"; import {TelepathyHandler} from "telepathy-contracts/amb/interfaces/TelepathyHandler.sol"; /// @title CrossChainMailer /// @author Succinct Labs /// @notice An example contract for sending messages to other chains, using the TelepathyRouter. /// @dev The FeeCollector is for discouraging spam on non-mainnet chains. contract CrossChainMailer is FeeCollector, ENSHelper { /// @notice The TelepathyRouter contract, which sends messages to other chains. ITelepathyRouter public telepathyRouter; constructor(address _telepathyRouter) { telepathyRouter = ITelepathyRouter(_telepathyRouter); } /// @notice Sends a message to a destination mailbox. /// @param _destinationChainId The chain ID where the destination CrossChainMailbox. /// @param _destinationMailbox The address of the destination CrossChainMailbox. /// @param _message The message to send. function sendMail(uint32 _destinationChainId, address _destinationMailbox, bytes memory _message) external payable { if (msg.value < fee) { revert InsufficientFee(msg.value, fee); } string memory data = StringHelper.formatMessage(_message, msg.sender.balance, ENSHelper.getName(msg.sender)); telepathyRouter.send(_destinationChainId, _destinationMailbox, bytes(data)); } } /// @title CrossChainMailbox /// @author Succinct Labs /// @notice An example contract for receiving messages from other chains, using the TelepathyHandler. contract CrossChainMailbox is TelepathyHandler { string[] public messages; event MessageReceived(uint32 indexed sourceChainId, address indexed sourceAddress, string message); constructor(address _telepathyRouter) TelepathyHandler(_telepathyRouter) {} function handleTelepathyImpl(uint32 _sourceChainId, address _sourceAddress, bytes memory _message) internal override { messages.push(string(_message)); emit MessageReceived(_sourceChainId, _sourceAddress, string(_message)); } function messagesLength() external view returns (uint256) { return messages.length; } }
pragma solidity >=0.8.4; interface ENS { // Logged when the owner of a node assigns a new owner to a subnode. event NewOwner(bytes32 indexed node, bytes32 indexed label, address owner); // Logged when the owner of a node transfers ownership to a new account. event Transfer(bytes32 indexed node, address owner); // Logged when the resolver for a node changes. event NewResolver(bytes32 indexed node, address resolver); // Logged when the TTL of a node changes event NewTTL(bytes32 indexed node, uint64 ttl); // Logged when an operator is added or removed. event ApprovalForAll( address indexed owner, address indexed operator, bool approved ); function setRecord( bytes32 node, address owner, address resolver, uint64 ttl ) external; function setSubnodeRecord( bytes32 node, bytes32 label, address owner, address resolver, uint64 ttl ) external; function setSubnodeOwner( bytes32 node, bytes32 label, address owner ) external returns (bytes32); function setResolver(bytes32 node, address resolver) external; function setOwner(bytes32 node, address owner) external; function setTTL(bytes32 node, uint64 ttl) external; function setApprovalForAll(address operator, bool approved) external; function owner(bytes32 node) external view returns (address); function resolver(bytes32 node) external view returns (address); function ttl(bytes32 node) external view returns (uint64); function recordExists(bytes32 node) external view returns (bool); function isApprovedForAll( address owner, address operator ) external view returns (bool); }
// SPDX-License-Identifier: MIT pragma solidity >=0.8.4; /** * Interface for the legacy (ETH-only) addr function. */ interface IAddrResolver { event AddrChanged(bytes32 indexed node, address a); /** * Returns the address associated with an ENS node. * @param node The ENS node to query. * @return The associated address. */ function addr(bytes32 node) external view returns (address payable); }
// SPDX-License-Identifier: MIT pragma solidity >=0.8.4; interface INameResolver { event NameChanged(bytes32 indexed node, string name); /** * Returns the name associated with an ENS node, for reverse records. * Defined in EIP181. * @param node The ENS node to query. * @return The associated name. */ function name(bytes32 node) external view returns (string memory); }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v4.7.0) (access/Ownable.sol) pragma solidity ^0.8.0; import "../utils/Context.sol"; /** * @dev Contract module which provides a basic access control mechanism, where * there is an account (an owner) that can be granted exclusive access to * specific functions. * * By default, the owner account will be the one that deploys the contract. This * can later be changed with {transferOwnership}. * * This module is used through inheritance. It will make available the modifier * `onlyOwner`, which can be applied to your functions to restrict their use to * the owner. */ abstract contract Ownable is Context { address private _owner; event OwnershipTransferred(address indexed previousOwner, address indexed newOwner); /** * @dev Initializes the contract setting the deployer as the initial owner. */ constructor() { _transferOwnership(_msgSender()); } /** * @dev Throws if called by any account other than the owner. */ modifier onlyOwner() { _checkOwner(); _; } /** * @dev Returns the address of the current owner. */ function owner() public view virtual returns (address) { return _owner; } /** * @dev Throws if the sender is not the owner. */ function _checkOwner() internal view virtual { require(owner() == _msgSender(), "Ownable: caller is not the owner"); } /** * @dev Leaves the contract without owner. It will not be possible to call * `onlyOwner` functions anymore. Can only be called by the current owner. * * NOTE: Renouncing ownership will leave the contract without an owner, * thereby removing any functionality that is only available to the owner. */ function renounceOwnership() public virtual onlyOwner { _transferOwnership(address(0)); } /** * @dev Transfers ownership of the contract to a new account (`newOwner`). * Can only be called by the current owner. */ function transferOwnership(address newOwner) public virtual onlyOwner { require(newOwner != address(0), "Ownable: new owner is the zero address"); _transferOwnership(newOwner); } /** * @dev Transfers ownership of the contract to a new account (`newOwner`). * Internal function without access restriction. */ function _transferOwnership(address newOwner) internal virtual { address oldOwner = _owner; _owner = newOwner; emit OwnershipTransferred(oldOwner, newOwner); } }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v4.8.0) (utils/Address.sol) pragma solidity ^0.8.1; /** * @dev Collection of functions related to the address type */ library Address { /** * @dev Returns true if `account` is a contract. * * [IMPORTANT] * ==== * It is unsafe to assume that an address for which this function returns * false is an externally-owned account (EOA) and not a contract. * * Among others, `isContract` will return false for the following * types of addresses: * * - an externally-owned account * - a contract in construction * - an address where a contract will be created * - an address where a contract lived, but was destroyed * ==== * * [IMPORTANT] * ==== * You shouldn't rely on `isContract` to protect against flash loan attacks! * * Preventing calls from contracts is highly discouraged. It breaks composability, breaks support for smart wallets * like Gnosis Safe, and does not provide security since it can be circumvented by calling from a contract * constructor. * ==== */ function isContract(address account) internal view returns (bool) { // This method relies on extcodesize/address.code.length, which returns 0 // for contracts in construction, since the code is only stored at the end // of the constructor execution. return account.code.length > 0; } /** * @dev Replacement for Solidity's `transfer`: sends `amount` wei to * `recipient`, forwarding all available gas and reverting on errors. * * https://eips.ethereum.org/EIPS/eip-1884[EIP1884] increases the gas cost * of certain opcodes, possibly making contracts go over the 2300 gas limit * imposed by `transfer`, making them unable to receive funds via * `transfer`. {sendValue} removes this limitation. * * https://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.5.11/security-considerations.html#use-the-checks-effects-interactions-pattern[checks-effects-interactions pattern]. */ function sendValue(address payable recipient, uint256 amount) internal { require(address(this).balance >= amount, "Address: insufficient balance"); (bool success, ) = recipient.call{value: amount}(""); require(success, "Address: unable to send value, recipient may have reverted"); } /** * @dev Performs a Solidity function call using a low level `call`. A * plain `call` is an unsafe replacement for a function call: use this * function instead. * * If `target` reverts with a revert reason, it is bubbled up by this * function (like regular Solidity function calls). * * Returns the raw returned data. To convert to the expected return value, * use https://solidity.readthedocs.io/en/latest/units-and-global-variables.html?highlight=abi.decode#abi-encoding-and-decoding-functions[`abi.decode`]. * * Requirements: * * - `target` must be a contract. * - calling `target` with `data` must not revert. * * _Available since v3.1._ */ function functionCall(address target, bytes memory data) internal returns (bytes memory) { return functionCallWithValue(target, data, 0, "Address: low-level call failed"); } /** * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`], but with * `errorMessage` as a fallback revert reason when `target` reverts. * * _Available since v3.1._ */ function functionCall( address target, bytes memory data, string memory errorMessage ) internal returns (bytes memory) { return functionCallWithValue(target, data, 0, errorMessage); } /** * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`], * but also transferring `value` wei to `target`. * * Requirements: * * - the calling contract must have an ETH balance of at least `value`. * - the called Solidity function must be `payable`. * * _Available since v3.1._ */ function functionCallWithValue( address target, bytes memory data, uint256 value ) internal returns (bytes memory) { return functionCallWithValue(target, data, value, "Address: low-level call with value failed"); } /** * @dev Same as {xref-Address-functionCallWithValue-address-bytes-uint256-}[`functionCallWithValue`], but * with `errorMessage` as a fallback revert reason when `target` reverts. * * _Available since v3.1._ */ function functionCallWithValue( address target, bytes memory data, uint256 value, string memory errorMessage ) internal returns (bytes memory) { require(address(this).balance >= value, "Address: insufficient balance for call"); (bool success, bytes memory returndata) = target.call{value: value}(data); return verifyCallResultFromTarget(target, success, returndata, errorMessage); } /** * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`], * but performing a static call. * * _Available since v3.3._ */ function functionStaticCall(address target, bytes memory data) internal view returns (bytes memory) { return functionStaticCall(target, data, "Address: low-level static call failed"); } /** * @dev Same as {xref-Address-functionCall-address-bytes-string-}[`functionCall`], * but performing a static call. * * _Available since v3.3._ */ function functionStaticCall( address target, bytes memory data, string memory errorMessage ) internal view returns (bytes memory) { (bool success, bytes memory returndata) = target.staticcall(data); return verifyCallResultFromTarget(target, success, returndata, errorMessage); } /** * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`], * but performing a delegate call. * * _Available since v3.4._ */ function functionDelegateCall(address target, bytes memory data) internal returns (bytes memory) { return functionDelegateCall(target, data, "Address: low-level delegate call failed"); } /** * @dev Same as {xref-Address-functionCall-address-bytes-string-}[`functionCall`], * but performing a delegate call. * * _Available since v3.4._ */ function functionDelegateCall( address target, bytes memory data, string memory errorMessage ) internal returns (bytes memory) { (bool success, bytes memory returndata) = target.delegatecall(data); return verifyCallResultFromTarget(target, success, returndata, errorMessage); } /** * @dev Tool to verify that a low level call to smart-contract was successful, and revert (either by bubbling * the revert reason or using the provided one) in case of unsuccessful call or if target was not a contract. * * _Available since v4.8._ */ function verifyCallResultFromTarget( address target, bool success, bytes memory returndata, string memory errorMessage ) internal view returns (bytes memory) { if (success) { if (returndata.length == 0) { // only check isContract if the call was successful and the return data is empty // otherwise we already know that it was a contract require(isContract(target), "Address: call to non-contract"); } return returndata; } else { _revert(returndata, errorMessage); } } /** * @dev Tool to verify that a low level call was successful, and revert if it wasn't, either by bubbling the * revert reason or using the provided one. * * _Available since v4.3._ */ function verifyCallResult( bool success, bytes memory returndata, string memory errorMessage ) internal pure returns (bytes memory) { if (success) { return returndata; } else { _revert(returndata, errorMessage); } } function _revert(bytes memory returndata, string memory errorMessage) private pure { // Look for revert reason and bubble it up if present if (returndata.length > 0) { // The easiest way to bubble the revert reason is using memory via assembly /// @solidity memory-safe-assembly assembly { let returndata_size := mload(returndata) revert(add(32, returndata), returndata_size) } } else { revert(errorMessage); } } }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts v4.4.1 (utils/Context.sol) pragma solidity ^0.8.0; /** * @dev Provides information about the current execution context, including the * sender of the transaction and its data. While these are generally available * via msg.sender and msg.data, they should not be accessed in such a direct * manner, since when dealing with meta-transactions the account sending and * paying for execution may not be the actual sender (as far as an application * is concerned). * * This contract is only required for intermediate, library-like contracts. */ abstract contract Context { function _msgSender() internal view virtual returns (address) { return msg.sender; } function _msgData() internal view virtual returns (bytes calldata) { return msg.data; } }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v4.8.0) (utils/Strings.sol) pragma solidity ^0.8.0; import "./math/Math.sol"; /** * @dev String operations. */ library Strings { bytes16 private constant _SYMBOLS = "0123456789abcdef"; uint8 private constant _ADDRESS_LENGTH = 20; /** * @dev Converts a `uint256` to its ASCII `string` decimal representation. */ function toString(uint256 value) internal pure returns (string memory) { unchecked { uint256 length = Math.log10(value) + 1; string memory buffer = new string(length); uint256 ptr; /// @solidity memory-safe-assembly assembly { ptr := add(buffer, add(32, length)) } while (true) { ptr--; /// @solidity memory-safe-assembly assembly { mstore8(ptr, byte(mod(value, 10), _SYMBOLS)) } value /= 10; if (value == 0) break; } return buffer; } } /** * @dev Converts a `uint256` to its ASCII `string` hexadecimal representation. */ function toHexString(uint256 value) internal pure returns (string memory) { unchecked { return toHexString(value, Math.log256(value) + 1); } } /** * @dev Converts a `uint256` to its ASCII `string` hexadecimal representation with fixed length. */ function toHexString(uint256 value, uint256 length) internal pure returns (string memory) { bytes memory buffer = new bytes(2 * length + 2); buffer[0] = "0"; buffer[1] = "x"; for (uint256 i = 2 * length + 1; i > 1; --i) { buffer[i] = _SYMBOLS[value & 0xf]; value >>= 4; } require(value == 0, "Strings: hex length insufficient"); return string(buffer); } /** * @dev Converts an `address` with fixed length of 20 bytes to its not checksummed ASCII `string` hexadecimal representation. */ function toHexString(address addr) internal pure returns (string memory) { return toHexString(uint256(uint160(addr)), _ADDRESS_LENGTH); } }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v4.8.0) (utils/math/Math.sol) pragma solidity ^0.8.0; /** * @dev Standard math utilities missing in the Solidity language. */ library Math { enum Rounding { Down, // Toward negative infinity Up, // Toward infinity Zero // Toward zero } /** * @dev Returns the largest of two numbers. */ function max(uint256 a, uint256 b) internal pure returns (uint256) { return a > b ? a : b; } /** * @dev Returns the smallest of two numbers. */ function min(uint256 a, uint256 b) internal pure returns (uint256) { return a < b ? a : b; } /** * @dev Returns the average of two numbers. The result is rounded towards * zero. */ function average(uint256 a, uint256 b) internal pure returns (uint256) { // (a + b) / 2 can overflow. return (a & b) + (a ^ b) / 2; } /** * @dev Returns the ceiling of the division of two numbers. * * This differs from standard division with `/` in that it rounds up instead * of rounding down. */ function ceilDiv(uint256 a, uint256 b) internal pure returns (uint256) { // (a + b - 1) / b can overflow on addition, so we distribute. return a == 0 ? 0 : (a - 1) / b + 1; } /** * @notice Calculates floor(x * y / denominator) with full precision. Throws if result overflows a uint256 or denominator == 0 * @dev Original credit to Remco Bloemen under MIT license (https://xn--2-umb.com/21/muldiv) * with further edits by Uniswap Labs also under MIT license. */ function mulDiv( uint256 x, uint256 y, uint256 denominator ) internal pure returns (uint256 result) { unchecked { // 512-bit multiply [prod1 prod0] = x * y. Compute the product mod 2^256 and mod 2^256 - 1, then use // use the Chinese Remainder Theorem to reconstruct the 512 bit result. The result is stored in two 256 // variables such that product = prod1 * 2^256 + prod0. uint256 prod0; // Least significant 256 bits of the product uint256 prod1; // Most significant 256 bits of the product assembly { let mm := mulmod(x, y, not(0)) prod0 := mul(x, y) prod1 := sub(sub(mm, prod0), lt(mm, prod0)) } // Handle non-overflow cases, 256 by 256 division. if (prod1 == 0) { return prod0 / denominator; } // Make sure the result is less than 2^256. Also prevents denominator == 0. require(denominator > prod1); /////////////////////////////////////////////// // 512 by 256 division. /////////////////////////////////////////////// // Make division exact by subtracting the remainder from [prod1 prod0]. uint256 remainder; assembly { // Compute remainder using mulmod. remainder := mulmod(x, y, denominator) // Subtract 256 bit number from 512 bit number. prod1 := sub(prod1, gt(remainder, prod0)) prod0 := sub(prod0, remainder) } // Factor powers of two out of denominator and compute largest power of two divisor of denominator. Always >= 1. // See https://cs.stackexchange.com/q/138556/92363. // Does not overflow because the denominator cannot be zero at this stage in the function. uint256 twos = denominator & (~denominator + 1); assembly { // Divide denominator by twos. denominator := div(denominator, twos) // Divide [prod1 prod0] by twos. prod0 := div(prod0, twos) // Flip twos such that it is 2^256 / twos. If twos is zero, then it becomes one. twos := add(div(sub(0, twos), twos), 1) } // Shift in bits from prod1 into prod0. prod0 |= prod1 * twos; // Invert denominator mod 2^256. Now that denominator is an odd number, it has an inverse modulo 2^256 such // that denominator * inv = 1 mod 2^256. Compute the inverse by starting with a seed that is correct for // four bits. That is, denominator * inv = 1 mod 2^4. uint256 inverse = (3 * denominator) ^ 2; // Use the Newton-Raphson iteration to improve the precision. Thanks to Hensel's lifting lemma, this also works // in modular arithmetic, doubling the correct bits in each step. inverse *= 2 - denominator * inverse; // inverse mod 2^8 inverse *= 2 - denominator * inverse; // inverse mod 2^16 inverse *= 2 - denominator * inverse; // inverse mod 2^32 inverse *= 2 - denominator * inverse; // inverse mod 2^64 inverse *= 2 - denominator * inverse; // inverse mod 2^128 inverse *= 2 - denominator * inverse; // inverse mod 2^256 // Because the division is now exact we can divide by multiplying with the modular inverse of denominator. // This will give us the correct result modulo 2^256. Since the preconditions guarantee that the outcome is // less than 2^256, this is the final result. We don't need to compute the high bits of the result and prod1 // is no longer required. result = prod0 * inverse; return result; } } /** * @notice Calculates x * y / denominator with full precision, following the selected rounding direction. */ function mulDiv( uint256 x, uint256 y, uint256 denominator, Rounding rounding ) internal pure returns (uint256) { uint256 result = mulDiv(x, y, denominator); if (rounding == Rounding.Up && mulmod(x, y, denominator) > 0) { result += 1; } return result; } /** * @dev Returns the square root of a number. If the number is not a perfect square, the value is rounded down. * * Inspired by Henry S. Warren, Jr.'s "Hacker's Delight" (Chapter 11). */ function sqrt(uint256 a) internal pure returns (uint256) { if (a == 0) { return 0; } // For our first guess, we get the biggest power of 2 which is smaller than the square root of the target. // // We know that the "msb" (most significant bit) of our target number `a` is a power of 2 such that we have // `msb(a) <= a < 2*msb(a)`. This value can be written `msb(a)=2**k` with `k=log2(a)`. // // This can be rewritten `2**log2(a) <= a < 2**(log2(a) + 1)` // → `sqrt(2**k) <= sqrt(a) < sqrt(2**(k+1))` // → `2**(k/2) <= sqrt(a) < 2**((k+1)/2) <= 2**(k/2 + 1)` // // Consequently, `2**(log2(a) / 2)` is a good first approximation of `sqrt(a)` with at least 1 correct bit. uint256 result = 1 << (log2(a) >> 1); // At this point `result` is an estimation with one bit of precision. We know the true value is a uint128, // since it is the square root of a uint256. Newton's method converges quadratically (precision doubles at // every iteration). We thus need at most 7 iteration to turn our partial result with one bit of precision // into the expected uint128 result. unchecked { result = (result + a / result) >> 1; result = (result + a / result) >> 1; result = (result + a / result) >> 1; result = (result + a / result) >> 1; result = (result + a / result) >> 1; result = (result + a / result) >> 1; result = (result + a / result) >> 1; return min(result, a / result); } } /** * @notice Calculates sqrt(a), following the selected rounding direction. */ function sqrt(uint256 a, Rounding rounding) internal pure returns (uint256) { unchecked { uint256 result = sqrt(a); return result + (rounding == Rounding.Up && result * result < a ? 1 : 0); } } /** * @dev Return the log in base 2, rounded down, of a positive value. * Returns 0 if given 0. */ function log2(uint256 value) internal pure returns (uint256) { uint256 result = 0; unchecked { if (value >> 128 > 0) { value >>= 128; result += 128; } if (value >> 64 > 0) { value >>= 64; result += 64; } if (value >> 32 > 0) { value >>= 32; result += 32; } if (value >> 16 > 0) { value >>= 16; result += 16; } if (value >> 8 > 0) { value >>= 8; result += 8; } if (value >> 4 > 0) { value >>= 4; result += 4; } if (value >> 2 > 0) { value >>= 2; result += 2; } if (value >> 1 > 0) { result += 1; } } return result; } /** * @dev Return the log in base 2, following the selected rounding direction, of a positive value. * Returns 0 if given 0. */ function log2(uint256 value, Rounding rounding) internal pure returns (uint256) { unchecked { uint256 result = log2(value); return result + (rounding == Rounding.Up && 1 << result < value ? 1 : 0); } } /** * @dev Return the log in base 10, rounded down, of a positive value. * Returns 0 if given 0. */ function log10(uint256 value) internal pure returns (uint256) { uint256 result = 0; unchecked { if (value >= 10**64) { value /= 10**64; result += 64; } if (value >= 10**32) { value /= 10**32; result += 32; } if (value >= 10**16) { value /= 10**16; result += 16; } if (value >= 10**8) { value /= 10**8; result += 8; } if (value >= 10**4) { value /= 10**4; result += 4; } if (value >= 10**2) { value /= 10**2; result += 2; } if (value >= 10**1) { result += 1; } } return result; } /** * @dev Return the log in base 10, following the selected rounding direction, of a positive value. * Returns 0 if given 0. */ function log10(uint256 value, Rounding rounding) internal pure returns (uint256) { unchecked { uint256 result = log10(value); return result + (rounding == Rounding.Up && 10**result < value ? 1 : 0); } } /** * @dev Return the log in base 256, rounded down, of a positive value. * Returns 0 if given 0. * * Adding one to the result gives the number of pairs of hex symbols needed to represent `value` as a hex string. */ function log256(uint256 value) internal pure returns (uint256) { uint256 result = 0; unchecked { if (value >> 128 > 0) { value >>= 128; result += 16; } if (value >> 64 > 0) { value >>= 64; result += 8; } if (value >> 32 > 0) { value >>= 32; result += 4; } if (value >> 16 > 0) { value >>= 16; result += 2; } if (value >> 8 > 0) { result += 1; } } return result; } /** * @dev Return the log in base 10, following the selected rounding direction, of a positive value. * Returns 0 if given 0. */ function log256(uint256 value, Rounding rounding) internal pure returns (uint256) { unchecked { uint256 result = log256(value); return result + (rounding == Rounding.Up && 1 << (result << 3) < value ? 1 : 0); } } }
pragma solidity ^0.8.0; enum MessageStatus { NOT_EXECUTED, EXECUTION_FAILED, EXECUTION_SUCCEEDED } struct Message { uint8 version; uint64 nonce; uint32 sourceChainId; address senderAddress; uint32 recipientChainId; bytes32 recipientAddress; bytes data; } interface ITelepathyRouter { event SentMessage(uint64 indexed nonce, bytes32 indexed msgHash, bytes message); function send(uint32 recipientChainId, bytes32 recipientAddress, bytes calldata data) external returns (bytes32); function send(uint32 recipientChainId, address recipientAddress, bytes calldata data) external returns (bytes32); function sendViaStorage(uint32 recipientChainId, bytes32 recipientAddress, bytes calldata data) external returns (bytes32); function sendViaStorage(uint32 recipientChainId, address recipientAddress, bytes calldata data) external returns (bytes32); } interface ITelepathyReceiver { event ExecutedMessage( uint32 indexed sourceChainId, uint64 indexed nonce, bytes32 indexed msgHash, bytes message, bool status ); function executeMessage( uint64 slot, bytes calldata message, bytes[] calldata accountProof, bytes[] calldata storageProof ) external; function executeMessageFromLog( bytes calldata srcSlotTxSlotPack, bytes calldata messageBytes, bytes32[] calldata receiptsRootProof, bytes32 receiptsRoot, bytes[] calldata receiptProof, // receipt proof against receipt root bytes memory txIndexRLPEncoded, uint256 logIndex ) external; } interface ITelepathyHandler { function handleTelepathy(uint32 _sourceChainId, address _senderAddress, bytes memory _data) external returns (bytes4); }
pragma solidity ^0.8.0; import {ITelepathyHandler} from "./ITelepathy.sol"; abstract contract TelepathyHandler is ITelepathyHandler { error NotFromTelepathyReceiever(address sender); address private _telepathyReceiever; constructor(address telepathyReceiever) { _telepathyReceiever = telepathyReceiever; } function handleTelepathy(uint32 _sourceChainId, address _senderAddress, bytes memory _data) external override returns (bytes4) { if (msg.sender != _telepathyReceiever) { revert NotFromTelepathyReceiever(msg.sender); } handleTelepathyImpl(_sourceChainId, _senderAddress, _data); return ITelepathyHandler.handleTelepathy.selector; } function handleTelepathyImpl(uint32 _sourceChainId, address _senderAddress, bytes memory _data) internal virtual; }
pragma solidity ^0.8.16; import {Address} from "openzeppelin-contracts/contracts/utils/Address.sol"; import {Strings} from "openzeppelin-contracts/contracts/utils/Strings.sol"; import {ENS} from "ens-contracts/registry/ENS.sol"; import {IAddrResolver} from "ens-contracts/resolvers/profiles/IAddrResolver.sol"; import {INameResolver} from "ens-contracts/resolvers/profiles/INameResolver.sol"; contract ENSHelper { using Address for address; using ENSNamehash for bytes; // Same address for Mainet, Ropsten, Rinkerby, Gorli and other networks; address constant ensRegistryAddr = 0x00000000000C2E074eC69A0dFb2997BA6C7d2e1e; /// The namehash of the `eth` TLD in the ENS registry, eg. namehash("eth"). bytes32 public constant ETH_NODE = keccak256(abi.encodePacked(bytes32(0), keccak256("eth"))); /// @notice Returns the ENS name for a given address, or an string address if no name is set. /// @param _addr The address to lookup. /// @return name The ENS name for the given address. /// @dev For this to successfully retrieve a name, the address must have the reverse record /// set, and the forward record must match the address. function getName(address _addr) public view returns (string memory name) { if (!ensRegistryAddr.isContract()) { return Strings.toHexString(_addr); } // Use reverse resolver to get the ENS name that address this has. bytes32 nodeReverse = reverseNode(_addr); address reverseResolverAddr = ENS(ensRegistryAddr).resolver(nodeReverse); if (reverseResolverAddr == address(0) || !reverseResolverAddr.isContract()) { return Strings.toHexString(_addr); } name = INameResolver(reverseResolverAddr).name(nodeReverse); if (bytes(name).length == 0) { return Strings.toHexString(_addr); } // ENS does not enforce the accuracy of reverse records, so you you must always perform a // forward resolution for the returned name and check it matches the original address. bytes32 nodeForward = bytes(name).namehash(0); address forwardResolverAddr = ENS(ensRegistryAddr).resolver(nodeForward); if (forwardResolverAddr == address(0) || !forwardResolverAddr.isContract()) { return Strings.toHexString(_addr); } address forwardAddr = IAddrResolver(forwardResolverAddr).addr(nodeForward); if (forwardAddr == _addr) { return name; } else { return Strings.toHexString(_addr); } } // Below are helper functions from ReverseRecords.sol, used so it's not necassary to maintain // a reference to the contract on each chain. // Source: https://github.com/ensdomains/reverse-records/blob/6ef80ba0a445b3f7cdff7819aaad1efbd8ad22fb/contracts/ReverseRecords.sol /// @notice This is the equivalant of namehash('addr.reverse') bytes32 public constant ADDR_REVERSE_NODE = 0x91d1777781884d03a6757a803996e38de2a42967fb37eeaca72729271025a9e2; /// @notice Returns the node hash for a given account's reverse records. function reverseNode(address _addr) public pure returns (bytes32) { return keccak256(abi.encodePacked(ADDR_REVERSE_NODE, sha3HexAddress(_addr))); } function sha3HexAddress(address addr) private pure returns (bytes32 ret) { addr; ret; // Stop warning us about unused variables assembly { let lookup := 0x3031323334353637383961626364656600000000000000000000000000000000 for { let i := 40 } gt(i, 0) {} { i := sub(i, 1) mstore8(i, byte(and(addr, 0xf), lookup)) addr := div(addr, 0x10) i := sub(i, 1) mstore8(i, byte(and(addr, 0xf), lookup)) addr := div(addr, 0x10) } ret := keccak256(0, 40) } } } /// @dev Source: https://github.com/JonahGroendal/ens-namehash/blob/d956b0be0ae5d14191067ed398c4454e35f4558d/contracts/ENSNamehash.sol library ENSNamehash { function namehash(bytes memory domain) internal pure returns (bytes32) { return namehash(domain, 0); } function namehash(bytes memory domain, uint256 i) internal pure returns (bytes32) { if (domain.length <= i) { return 0x0000000000000000000000000000000000000000000000000000000000000000; } uint256 len = LabelLength(domain, i); return keccak256(abi.encodePacked(namehash(domain, i + len + 1), keccak(domain, i, len))); } function LabelLength(bytes memory domain, uint256 i) private pure returns (uint256) { uint256 len; while (i + len != domain.length && domain[i + len] != 0x2e) { len++; } return len; } function keccak(bytes memory data, uint256 offset, uint256 len) private pure returns (bytes32 ret) { require(offset + len <= data.length); assembly { ret := keccak256(add(add(data, 32), offset), len) } } }
pragma solidity ^0.8.16; import {Ownable} from "openzeppelin-contracts/contracts/access/Ownable.sol"; contract FeeCollector is Ownable { error InsufficientFee(uint256 actual, uint256 expected); /// @notice The fee to pay for sending a message. /// @dev The intention is only set to non-zero when deployed non-mainnet chains, used to discourage spam. uint256 public fee; /// @notice Allows owner to set a new fee. /// @param _fee The new fee to use. function setFee(uint256 _fee) external onlyOwner { fee = _fee; } /// @notice Allows owner to claim all fees sent to this contract. function claimFees() external onlyOwner { payable(owner()).transfer(address(this).balance); } }
pragma solidity ^0.8.16; import {Strings} from "openzeppelin-contracts/contracts/utils/Strings.sol"; library StringHelper { /// @notice Concatenates together a formatted message. /// @param _rawMessage The raw message bytes. /// @param _balance The balance of the sender. /// @param _ensName The ENS name of the sender ("" if none). /// @dev The formatting is like: /// /// 'hello, world!' /// - alice.eth (1.00 ETH) function formatMessage(bytes memory _rawMessage, uint256 _balance, string memory _ensName) internal view returns (string memory) { string memory messageStr = string(_rawMessage); string memory ethBalanceStr = formatBalance(_balance); // Use the ENS name if it exists, otherwise use the address. string memory senderStr; if (bytes(_ensName).length == 0) { senderStr = Strings.toHexString(msg.sender); } else { senderStr = _ensName; } string memory lineOne = string.concat(string.concat("'", messageStr), "'\n"); string memory lineTwo = string.concat(string.concat(string.concat(string.concat("- ", senderStr), " ("), ethBalanceStr), ")"); string memory data = string.concat(lineOne, lineTwo); return data; } /// @notice Formats a native balance to a string with 2 decimal places and native currency /// symbol. For example, 123456789000000000000 wei would be formatted as "123.46 ETH". /// @param _balance The balance to format. function formatBalance(uint256 _balance) public view returns (string memory) { uint256 integerAmount = _balance / 1 ether; uint256 integerDigits; if (integerAmount > 0) { while (true) { if (integerAmount >= 10 ** integerDigits) { integerDigits++; } else { break; } } } else { integerDigits = 1; } bytes memory balanceByteArr = new bytes(integerDigits + 3); // extra 3 for "." plus 2 digits uint256 i = integerDigits; while (i > 0) { balanceByteArr[i - 1] = bytes1(uint8(48 + integerAmount % 10)); integerAmount /= 10; i--; } balanceByteArr[integerDigits] = "."; balanceByteArr[integerDigits + 1] = bytes1(uint8(48 + (_balance / 1e17) % 10)); balanceByteArr[integerDigits + 2] = bytes1(uint8(48 + (_balance / 1e16) % 10)); string memory balanceStr = string(balanceByteArr); // ETH for mainnet, xDAI for Gnosis, etc string memory currencyStr; if (block.chainid == 5) { currencyStr = " gETH"; } else if (block.chainid == 100) { currencyStr = " xDAI"; } else if (block.chainid == 137) { currencyStr = " MATIC"; } else { currencyStr = " ETH"; } return string.concat(balanceStr, currencyStr); } }
{ "remappings": [ "@optimism-bedrock/=contracts/lib/telepathy-contracts/lib/optimism-bedrock-contracts/", "@uniswap/=contracts/lib/telepathy-contracts/lib/", "Solidity-RLP/=contracts/lib/telepathy-contracts/lib/Solidity-RLP/contracts/", "curve-merkle-oracle/=contracts/lib/telepathy-contracts/lib/curve-merkle-oracle/contracts/", "ds-test/=contracts/lib/forge-std/lib/ds-test/src/", "ens-contracts/=contracts/lib/ens-contracts/contracts/", "erc4626-tests/=contracts/lib/telepathy-contracts/lib/openzeppelin-contracts/lib/erc4626-tests/", "forge-std/=contracts/lib/forge-std/src/", "openzeppelin-contracts-upgradeable/=contracts/lib/telepathy-contracts/lib/openzeppelin-contracts-upgradeable/contracts/", "openzeppelin-contracts/=contracts/lib/telepathy-contracts/lib/openzeppelin-contracts/", "optimism-bedrock-contracts/=contracts/lib/telepathy-contracts/lib/optimism-bedrock-contracts/", "telepathy-contracts/=contracts/lib/telepathy-contracts/src/", "v3-core/=contracts/lib/telepathy-contracts/lib/v3-core/" ], "optimizer": { "enabled": true, "runs": 200 }, "metadata": { "bytecodeHash": "ipfs" }, "outputSelection": { "*": { "*": [ "evm.bytecode", "evm.deployedBytecode", "devdoc", "userdoc", "metadata", "abi" ] } }, "evmVersion": "london", "libraries": {} }
Contract Security Audit
- No Contract Security Audit Submitted- Submit Audit Here
[{"inputs":[{"internalType":"address","name":"_telepathyRouter","type":"address"}],"stateMutability":"nonpayable","type":"constructor"},{"inputs":[{"internalType":"uint256","name":"actual","type":"uint256"},{"internalType":"uint256","name":"expected","type":"uint256"}],"name":"InsufficientFee","type":"error"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"previousOwner","type":"address"},{"indexed":true,"internalType":"address","name":"newOwner","type":"address"}],"name":"OwnershipTransferred","type":"event"},{"inputs":[],"name":"ADDR_REVERSE_NODE","outputs":[{"internalType":"bytes32","name":"","type":"bytes32"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"ETH_NODE","outputs":[{"internalType":"bytes32","name":"","type":"bytes32"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"claimFees","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"fee","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"_addr","type":"address"}],"name":"getName","outputs":[{"internalType":"string","name":"name","type":"string"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"owner","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"renounceOwnership","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"_addr","type":"address"}],"name":"reverseNode","outputs":[{"internalType":"bytes32","name":"","type":"bytes32"}],"stateMutability":"pure","type":"function"},{"inputs":[{"internalType":"uint32","name":"_destinationChainId","type":"uint32"},{"internalType":"address","name":"_destinationMailbox","type":"address"},{"internalType":"bytes","name":"_message","type":"bytes"}],"name":"sendMail","outputs":[],"stateMutability":"payable","type":"function"},{"inputs":[{"internalType":"uint256","name":"_fee","type":"uint256"}],"name":"setFee","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"telepathyRouter","outputs":[{"internalType":"contract ITelepathyRouter","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"newOwner","type":"address"}],"name":"transferOwnership","outputs":[],"stateMutability":"nonpayable","type":"function"}]
Contract Creation Code
608060405234801561001057600080fd5b5060405161158038038061158083398101604081905261002f916100ad565b6100383361005d565b600280546001600160a01b0319166001600160a01b03929092169190911790556100dd565b600080546001600160a01b038381166001600160a01b0319831681178455604051919092169283917f8be0079c531659141344cd1fd0a4f28419497f9722a3daafe3b4186f6b6457e09190a35050565b6000602082840312156100bf57600080fd5b81516001600160a01b03811681146100d657600080fd5b9392505050565b611494806100ec6000396000f3fe6080604052600436106100a75760003560e01c80638da5cb5b116100645780638da5cb5b1461018e578063cc473be3146101c0578063d294f0931461021f578063dcc6329014610234578063ddca3f4314610254578063f2fde38b1461026a57600080fd5b806349905b6d146100ac5780635fd4b08a146100c157806369fe0e2d146100f7578063715018a6146101175780637315937d1461012c5780637cf8a2eb1461015a575b600080fd5b6100bf6100ba366004610f78565b61028a565b005b3480156100cd57600080fd5b506100e16100dc366004611026565b61035d565b6040516100ee9190611093565b60405180910390f35b34801561010357600080fd5b506100bf6101123660046110a6565b610618565b34801561012357600080fd5b506100bf610625565b34801561013857600080fd5b5061014c610147366004611026565b610639565b6040519081526020016100ee565b34801561016657600080fd5b5061014c7f91d1777781884d03a6757a803996e38de2a42967fb37eeaca72729271025a9e281565b34801561019a57600080fd5b506000546001600160a01b03165b6040516001600160a01b0390911681526020016100ee565b3480156101cc57600080fd5b5061014c60408051600060208201527f4f5b812789fc606be1b3b16908db13fc7a9adf7ca72641f84d75b47069d3d7f0918101919091526060016040516020818303038152906040528051906020012081565b34801561022b57600080fd5b506100bf610694565b34801561024057600080fd5b506002546101a8906001600160a01b031681565b34801561026057600080fd5b5061014c60015481565b34801561027657600080fd5b506100bf610285366004611026565b6106d9565b6001543410156102bf5760015460405163a458261b60e01b815234600482015260248101919091526044015b60405180910390fd5b60006102de82336001600160a01b0316316102d93361035d565b61074f565b60025460405163341a354960e11b81529192506001600160a01b0316906368346a9290610313908790879086906004016110bf565b6020604051808303816000875af1158015610332573d6000803e3d6000fd5b505050506040513d601f19601f8201168201806040525081019061035691906110f8565b5050505050565b60606e0c2e074ec69a0dfb2997ba6c7d2e1e3b6103835761037d8261086e565b92915050565b600061038e83610639565b604051630178b8bf60e01b8152600481018290529091506000906e0c2e074ec69a0dfb2997ba6c7d2e1e90630178b8bf90602401602060405180830381865afa1580156103df573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906104039190611111565b90506001600160a01b038116158061042357506001600160a01b0381163b155b15610439576104318461086e565b949350505050565b60405163691f343160e01b8152600481018390526001600160a01b0382169063691f343190602401600060405180830381865afa15801561047e573d6000803e3d6000fd5b505050506040513d6000823e601f3d908101601f191682016040526104a6919081019061112e565b925082516000036104ba576104318461086e565b60006104c68482610884565b604051630178b8bf60e01b8152600481018290529091506000906e0c2e074ec69a0dfb2997ba6c7d2e1e90630178b8bf90602401602060405180830381865afa158015610517573d6000803e3d6000fd5b505050506040513d601f19601f8201168201806040525081019061053b9190611111565b90506001600160a01b038116158061055b57506001600160a01b0381163b155b15610573576105698661086e565b9695505050505050565b604051631d9dabef60e11b8152600481018390526000906001600160a01b03831690633b3b57de90602401602060405180830381865afa1580156105bb573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906105df9190611111565b9050866001600160a01b0316816001600160a01b031603610604575050505050919050565b61060d8761086e565b979650505050505050565b6106206108fe565b600155565b61062d6108fe565b6106376000610958565b565b60007f91d1777781884d03a6757a803996e38de2a42967fb37eeaca72729271025a9e2610665836109a8565b604080516020810193909352820152606001604051602081830303815290604052805190602001209050919050565b61069c6108fe565b600080546040516001600160a01b03909116914780156108fc02929091818181858888f193505050501580156106d6573d6000803e3d6000fd5b50565b6106e16108fe565b6001600160a01b0381166107465760405162461bcd60e51b815260206004820152602660248201527f4f776e61626c653a206e6577206f776e657220697320746865207a65726f206160448201526564647265737360d01b60648201526084016102b6565b6106d681610958565b606083600061075d856109ff565b90506060845160000361077a576107733361086e565b905061077d565b50835b600083604051602001610790919061119c565b60408051601f19818403018152908290526107ad916020016111c5565b60405160208183030381529060405290506000826040516020016107d191906111eb565b60408051601f19818403018152908290526107ee91602001611215565b60408051601f198184030181529082905261080d91869060200161123b565b60408051601f198184030181529082905261082a9160200161126a565b60405160208183030381529060405290506000828260405160200161085092919061123b565b60408051808303601f190181529190529a9950505050505050505050565b606061037d6001600160a01b0383166014610cca565b6000818351116108965750600061037d565b60006108a28484610e6d565b90506108c2846108b283866112a5565b6108bd9060016112a5565b610884565b6108cd858584610ed0565b6040805160208101939093528201526060016040516020818303038152906040528051906020012091505092915050565b6000546001600160a01b031633146106375760405162461bcd60e51b815260206004820181905260248201527f4f776e61626c653a2063616c6c6572206973206e6f7420746865206f776e657260448201526064016102b6565b600080546001600160a01b038381166001600160a01b0319831681178455604051919092169283917f8be0079c531659141344cd1fd0a4f28419497f9722a3daafe3b4186f6b6457e09190a35050565b60006f181899199a1a9b1b9c1cb0b131b232b360811b60285b80156109f25760001901600f841682901a815360109093049260001901600f841682901a81536010840493506109c1565b5050602860002092915050565b60606000610a15670de0b6b3a7640000846112ce565b905060008115610a48575b610a2b81600a6113c6565b8210610a435780610a3b816113d2565b915050610a20565b610a4c565b5060015b6000610a598260036112a5565b67ffffffffffffffff811115610a7157610a71610f09565b6040519080825280601f01601f191660200182016040528015610a9b576020820181803683370190505b509050815b8015610b1157610ab1600a856113eb565b610abc9060306112a5565b60f81b82610acb6001846113ff565b81518110610adb57610adb611412565b60200101906001600160f81b031916908160001a905350610afd600a856112ce565b935080610b0981611428565b915050610aa0565b601760f91b828481518110610b2857610b28611412565b60200101906001600160f81b031916908160001a905350600a610b5367016345785d8a0000886112ce565b610b5d91906113eb565b610b689060306112a5565b60f81b82610b778560016112a5565b81518110610b8757610b87611412565b60200101906001600160f81b031916908160001a905350600a610bb1662386f26fc10000886112ce565b610bbb91906113eb565b610bc69060306112a5565b60f81b82610bd58560026112a5565b81518110610be557610be5611412565b60200101906001600160f81b031916908160001a90535081606046600503610c295750604080518082019091526005815264040ce8aa8960db1b6020820152610c9b565b46606403610c535750604080518082019091526005815264207844414960d81b6020820152610c9b565b46608903610c7e5750604080518082019091526006815265204d4154494360d01b6020820152610c9b565b506040805180820190915260048152630408aa8960e31b60208201525b8181604051602001610cae92919061123b565b6040516020818303038152906040529650505050505050919050565b60606000610cd983600261143f565b610ce49060026112a5565b67ffffffffffffffff811115610cfc57610cfc610f09565b6040519080825280601f01601f191660200182016040528015610d26576020820181803683370190505b509050600360fc1b81600081518110610d4157610d41611412565b60200101906001600160f81b031916908160001a905350600f60fb1b81600181518110610d7057610d70611412565b60200101906001600160f81b031916908160001a9053506000610d9484600261143f565b610d9f9060016112a5565b90505b6001811115610e17576f181899199a1a9b1b9c1cb0b131b232b360811b85600f1660108110610dd357610dd3611412565b1a60f81b828281518110610de957610de9611412565b60200101906001600160f81b031916908160001a90535060049490941c93610e1081611428565b9050610da2565b508315610e665760405162461bcd60e51b815260206004820181905260248201527f537472696e67733a20686578206c656e67746820696e73756666696369656e7460448201526064016102b6565b9392505050565b6000805b8351610e7d82856112a5565b14158015610eb9575083610e9182856112a5565b81518110610ea157610ea1611412565b6020910101516001600160f81b031916601760f91b14155b15610e665780610ec8816113d2565b915050610e71565b8251600090610edf83856112a5565b1115610eea57600080fd5b5091016020012090565b6001600160a01b03811681146106d657600080fd5b634e487b7160e01b600052604160045260246000fd5b604051601f8201601f1916810167ffffffffffffffff81118282101715610f4857610f48610f09565b604052919050565b600067ffffffffffffffff821115610f6a57610f6a610f09565b50601f01601f191660200190565b600080600060608486031215610f8d57600080fd5b833563ffffffff81168114610fa157600080fd5b92506020840135610fb181610ef4565b9150604084013567ffffffffffffffff811115610fcd57600080fd5b8401601f81018613610fde57600080fd5b8035610ff1610fec82610f50565b610f1f565b81815287602083850101111561100657600080fd5b816020840160208301376000602083830101528093505050509250925092565b60006020828403121561103857600080fd5b8135610e6681610ef4565b60005b8381101561105e578181015183820152602001611046565b50506000910152565b6000815180845261107f816020860160208601611043565b601f01601f19169290920160200192915050565b602081526000610e666020830184611067565b6000602082840312156110b857600080fd5b5035919050565b63ffffffff841681526001600160a01b03831660208201526060604082018190526000906110ef90830184611067565b95945050505050565b60006020828403121561110a57600080fd5b5051919050565b60006020828403121561112357600080fd5b8151610e6681610ef4565b60006020828403121561114057600080fd5b815167ffffffffffffffff81111561115757600080fd5b8201601f8101841361116857600080fd5b8051611176610fec82610f50565b81815285602083850101111561118b57600080fd5b6110ef826020830160208601611043565b602760f81b8152600082516111b8816001850160208701611043565b9190910160010192915050565b600082516111d7818460208701611043565b61138560f11b920191825250600201919050565b61016960f51b815260008251611208816002850160208701611043565b9190910160020192915050565b60008251611227818460208701611043565b61040560f31b920191825250600201919050565b6000835161124d818460208801611043565b835190830190611261818360208801611043565b01949350505050565b6000825161127c818460208701611043565b602960f81b920191825250600101919050565b634e487b7160e01b600052601160045260246000fd5b8082018082111561037d5761037d61128f565b634e487b7160e01b600052601260045260246000fd5b6000826112dd576112dd6112b8565b500490565b600181815b8085111561131d5781600019048211156113035761130361128f565b8085161561131057918102915b93841c93908002906112e7565b509250929050565b6000826113345750600161037d565b816113415750600061037d565b816001811461135757600281146113615761137d565b600191505061037d565b60ff8411156113725761137261128f565b50506001821b61037d565b5060208310610133831016604e8410600b84101617156113a0575081810a61037d565b6113aa83836112e2565b80600019048211156113be576113be61128f565b029392505050565b6000610e668383611325565b6000600182016113e4576113e461128f565b5060010190565b6000826113fa576113fa6112b8565b500690565b8181038181111561037d5761037d61128f565b634e487b7160e01b600052603260045260246000fd5b6000816114375761143761128f565b506000190190565b60008160001904831182151516156114595761145961128f565b50029056fea2646970667358221220a68f3c617c435c62f7e6432f8fa73f0251a40c45c58ec8e95fa11f51a2f5955164736f6c6343000810003300000000000000000000000041ea857c32c8cb42eefa00af67862ecff4eb795a
Deployed Bytecode
0x6080604052600436106100a75760003560e01c80638da5cb5b116100645780638da5cb5b1461018e578063cc473be3146101c0578063d294f0931461021f578063dcc6329014610234578063ddca3f4314610254578063f2fde38b1461026a57600080fd5b806349905b6d146100ac5780635fd4b08a146100c157806369fe0e2d146100f7578063715018a6146101175780637315937d1461012c5780637cf8a2eb1461015a575b600080fd5b6100bf6100ba366004610f78565b61028a565b005b3480156100cd57600080fd5b506100e16100dc366004611026565b61035d565b6040516100ee9190611093565b60405180910390f35b34801561010357600080fd5b506100bf6101123660046110a6565b610618565b34801561012357600080fd5b506100bf610625565b34801561013857600080fd5b5061014c610147366004611026565b610639565b6040519081526020016100ee565b34801561016657600080fd5b5061014c7f91d1777781884d03a6757a803996e38de2a42967fb37eeaca72729271025a9e281565b34801561019a57600080fd5b506000546001600160a01b03165b6040516001600160a01b0390911681526020016100ee565b3480156101cc57600080fd5b5061014c60408051600060208201527f4f5b812789fc606be1b3b16908db13fc7a9adf7ca72641f84d75b47069d3d7f0918101919091526060016040516020818303038152906040528051906020012081565b34801561022b57600080fd5b506100bf610694565b34801561024057600080fd5b506002546101a8906001600160a01b031681565b34801561026057600080fd5b5061014c60015481565b34801561027657600080fd5b506100bf610285366004611026565b6106d9565b6001543410156102bf5760015460405163a458261b60e01b815234600482015260248101919091526044015b60405180910390fd5b60006102de82336001600160a01b0316316102d93361035d565b61074f565b60025460405163341a354960e11b81529192506001600160a01b0316906368346a9290610313908790879086906004016110bf565b6020604051808303816000875af1158015610332573d6000803e3d6000fd5b505050506040513d601f19601f8201168201806040525081019061035691906110f8565b5050505050565b60606e0c2e074ec69a0dfb2997ba6c7d2e1e3b6103835761037d8261086e565b92915050565b600061038e83610639565b604051630178b8bf60e01b8152600481018290529091506000906e0c2e074ec69a0dfb2997ba6c7d2e1e90630178b8bf90602401602060405180830381865afa1580156103df573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906104039190611111565b90506001600160a01b038116158061042357506001600160a01b0381163b155b15610439576104318461086e565b949350505050565b60405163691f343160e01b8152600481018390526001600160a01b0382169063691f343190602401600060405180830381865afa15801561047e573d6000803e3d6000fd5b505050506040513d6000823e601f3d908101601f191682016040526104a6919081019061112e565b925082516000036104ba576104318461086e565b60006104c68482610884565b604051630178b8bf60e01b8152600481018290529091506000906e0c2e074ec69a0dfb2997ba6c7d2e1e90630178b8bf90602401602060405180830381865afa158015610517573d6000803e3d6000fd5b505050506040513d601f19601f8201168201806040525081019061053b9190611111565b90506001600160a01b038116158061055b57506001600160a01b0381163b155b15610573576105698661086e565b9695505050505050565b604051631d9dabef60e11b8152600481018390526000906001600160a01b03831690633b3b57de90602401602060405180830381865afa1580156105bb573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906105df9190611111565b9050866001600160a01b0316816001600160a01b031603610604575050505050919050565b61060d8761086e565b979650505050505050565b6106206108fe565b600155565b61062d6108fe565b6106376000610958565b565b60007f91d1777781884d03a6757a803996e38de2a42967fb37eeaca72729271025a9e2610665836109a8565b604080516020810193909352820152606001604051602081830303815290604052805190602001209050919050565b61069c6108fe565b600080546040516001600160a01b03909116914780156108fc02929091818181858888f193505050501580156106d6573d6000803e3d6000fd5b50565b6106e16108fe565b6001600160a01b0381166107465760405162461bcd60e51b815260206004820152602660248201527f4f776e61626c653a206e6577206f776e657220697320746865207a65726f206160448201526564647265737360d01b60648201526084016102b6565b6106d681610958565b606083600061075d856109ff565b90506060845160000361077a576107733361086e565b905061077d565b50835b600083604051602001610790919061119c565b60408051601f19818403018152908290526107ad916020016111c5565b60405160208183030381529060405290506000826040516020016107d191906111eb565b60408051601f19818403018152908290526107ee91602001611215565b60408051601f198184030181529082905261080d91869060200161123b565b60408051601f198184030181529082905261082a9160200161126a565b60405160208183030381529060405290506000828260405160200161085092919061123b565b60408051808303601f190181529190529a9950505050505050505050565b606061037d6001600160a01b0383166014610cca565b6000818351116108965750600061037d565b60006108a28484610e6d565b90506108c2846108b283866112a5565b6108bd9060016112a5565b610884565b6108cd858584610ed0565b6040805160208101939093528201526060016040516020818303038152906040528051906020012091505092915050565b6000546001600160a01b031633146106375760405162461bcd60e51b815260206004820181905260248201527f4f776e61626c653a2063616c6c6572206973206e6f7420746865206f776e657260448201526064016102b6565b600080546001600160a01b038381166001600160a01b0319831681178455604051919092169283917f8be0079c531659141344cd1fd0a4f28419497f9722a3daafe3b4186f6b6457e09190a35050565b60006f181899199a1a9b1b9c1cb0b131b232b360811b60285b80156109f25760001901600f841682901a815360109093049260001901600f841682901a81536010840493506109c1565b5050602860002092915050565b60606000610a15670de0b6b3a7640000846112ce565b905060008115610a48575b610a2b81600a6113c6565b8210610a435780610a3b816113d2565b915050610a20565b610a4c565b5060015b6000610a598260036112a5565b67ffffffffffffffff811115610a7157610a71610f09565b6040519080825280601f01601f191660200182016040528015610a9b576020820181803683370190505b509050815b8015610b1157610ab1600a856113eb565b610abc9060306112a5565b60f81b82610acb6001846113ff565b81518110610adb57610adb611412565b60200101906001600160f81b031916908160001a905350610afd600a856112ce565b935080610b0981611428565b915050610aa0565b601760f91b828481518110610b2857610b28611412565b60200101906001600160f81b031916908160001a905350600a610b5367016345785d8a0000886112ce565b610b5d91906113eb565b610b689060306112a5565b60f81b82610b778560016112a5565b81518110610b8757610b87611412565b60200101906001600160f81b031916908160001a905350600a610bb1662386f26fc10000886112ce565b610bbb91906113eb565b610bc69060306112a5565b60f81b82610bd58560026112a5565b81518110610be557610be5611412565b60200101906001600160f81b031916908160001a90535081606046600503610c295750604080518082019091526005815264040ce8aa8960db1b6020820152610c9b565b46606403610c535750604080518082019091526005815264207844414960d81b6020820152610c9b565b46608903610c7e5750604080518082019091526006815265204d4154494360d01b6020820152610c9b565b506040805180820190915260048152630408aa8960e31b60208201525b8181604051602001610cae92919061123b565b6040516020818303038152906040529650505050505050919050565b60606000610cd983600261143f565b610ce49060026112a5565b67ffffffffffffffff811115610cfc57610cfc610f09565b6040519080825280601f01601f191660200182016040528015610d26576020820181803683370190505b509050600360fc1b81600081518110610d4157610d41611412565b60200101906001600160f81b031916908160001a905350600f60fb1b81600181518110610d7057610d70611412565b60200101906001600160f81b031916908160001a9053506000610d9484600261143f565b610d9f9060016112a5565b90505b6001811115610e17576f181899199a1a9b1b9c1cb0b131b232b360811b85600f1660108110610dd357610dd3611412565b1a60f81b828281518110610de957610de9611412565b60200101906001600160f81b031916908160001a90535060049490941c93610e1081611428565b9050610da2565b508315610e665760405162461bcd60e51b815260206004820181905260248201527f537472696e67733a20686578206c656e67746820696e73756666696369656e7460448201526064016102b6565b9392505050565b6000805b8351610e7d82856112a5565b14158015610eb9575083610e9182856112a5565b81518110610ea157610ea1611412565b6020910101516001600160f81b031916601760f91b14155b15610e665780610ec8816113d2565b915050610e71565b8251600090610edf83856112a5565b1115610eea57600080fd5b5091016020012090565b6001600160a01b03811681146106d657600080fd5b634e487b7160e01b600052604160045260246000fd5b604051601f8201601f1916810167ffffffffffffffff81118282101715610f4857610f48610f09565b604052919050565b600067ffffffffffffffff821115610f6a57610f6a610f09565b50601f01601f191660200190565b600080600060608486031215610f8d57600080fd5b833563ffffffff81168114610fa157600080fd5b92506020840135610fb181610ef4565b9150604084013567ffffffffffffffff811115610fcd57600080fd5b8401601f81018613610fde57600080fd5b8035610ff1610fec82610f50565b610f1f565b81815287602083850101111561100657600080fd5b816020840160208301376000602083830101528093505050509250925092565b60006020828403121561103857600080fd5b8135610e6681610ef4565b60005b8381101561105e578181015183820152602001611046565b50506000910152565b6000815180845261107f816020860160208601611043565b601f01601f19169290920160200192915050565b602081526000610e666020830184611067565b6000602082840312156110b857600080fd5b5035919050565b63ffffffff841681526001600160a01b03831660208201526060604082018190526000906110ef90830184611067565b95945050505050565b60006020828403121561110a57600080fd5b5051919050565b60006020828403121561112357600080fd5b8151610e6681610ef4565b60006020828403121561114057600080fd5b815167ffffffffffffffff81111561115757600080fd5b8201601f8101841361116857600080fd5b8051611176610fec82610f50565b81815285602083850101111561118b57600080fd5b6110ef826020830160208601611043565b602760f81b8152600082516111b8816001850160208701611043565b9190910160010192915050565b600082516111d7818460208701611043565b61138560f11b920191825250600201919050565b61016960f51b815260008251611208816002850160208701611043565b9190910160020192915050565b60008251611227818460208701611043565b61040560f31b920191825250600201919050565b6000835161124d818460208801611043565b835190830190611261818360208801611043565b01949350505050565b6000825161127c818460208701611043565b602960f81b920191825250600101919050565b634e487b7160e01b600052601160045260246000fd5b8082018082111561037d5761037d61128f565b634e487b7160e01b600052601260045260246000fd5b6000826112dd576112dd6112b8565b500490565b600181815b8085111561131d5781600019048211156113035761130361128f565b8085161561131057918102915b93841c93908002906112e7565b509250929050565b6000826113345750600161037d565b816113415750600061037d565b816001811461135757600281146113615761137d565b600191505061037d565b60ff8411156113725761137261128f565b50506001821b61037d565b5060208310610133831016604e8410600b84101617156113a0575081810a61037d565b6113aa83836112e2565b80600019048211156113be576113be61128f565b029392505050565b6000610e668383611325565b6000600182016113e4576113e461128f565b5060010190565b6000826113fa576113fa6112b8565b500690565b8181038181111561037d5761037d61128f565b634e487b7160e01b600052603260045260246000fd5b6000816114375761143761128f565b506000190190565b60008160001904831182151516156114595761145961128f565b50029056fea2646970667358221220a68f3c617c435c62f7e6432f8fa73f0251a40c45c58ec8e95fa11f51a2f5955164736f6c63430008100033
Constructor Arguments (ABI-Encoded and is the last bytes of the Contract Creation Code above)
00000000000000000000000041ea857c32c8cb42eefa00af67862ecff4eb795a
-----Decoded View---------------
Arg [0] : _telepathyRouter (address): 0x41EA857C32c8Cb42EEFa00AF67862eCFf4eB795a
-----Encoded View---------------
1 Constructor Arguments found :
Arg [0] : 00000000000000000000000041ea857c32c8cb42eefa00af67862ecff4eb795a
Loading...
Loading
Loading...
Loading
Loading...
Loading
[ Download: CSV Export ]
[ Download: CSV Export ]
A contract address hosts a smart contract, which is a set of code stored on the blockchain that runs when predetermined conditions are met. Learn more about addresses in our Knowledge Base.