Address Contract Verified
Address
0xa3b31028893c20bEAA882d1508Fe423acA4A70e5
Balance
0.000164410 ETH ($0.34)
Nonce
1
Code Size
5268 bytes
Creator
Create2 Deployer at tx 0x501579a2...150272
Indexed Transactions
0
Contract Bytecode
5268 bytes
0x6080604052600436106100a75760003560e01c80638da5cb5b116100645780638da5cb5b1461018e578063cc473be3146101c0578063d294f0931461021f578063dcc6329014610234578063ddca3f4314610254578063f2fde38b1461026a57600080fd5b806349905b6d146100ac5780635fd4b08a146100c157806369fe0e2d146100f7578063715018a6146101175780637315937d1461012c5780637cf8a2eb1461015a575b600080fd5b6100bf6100ba366004610f78565b61028a565b005b3480156100cd57600080fd5b506100e16100dc366004611026565b61035d565b6040516100ee9190611093565b60405180910390f35b34801561010357600080fd5b506100bf6101123660046110a6565b610618565b34801561012357600080fd5b506100bf610625565b34801561013857600080fd5b5061014c610147366004611026565b610639565b6040519081526020016100ee565b34801561016657600080fd5b5061014c7f91d1777781884d03a6757a803996e38de2a42967fb37eeaca72729271025a9e281565b34801561019a57600080fd5b506000546001600160a01b03165b6040516001600160a01b0390911681526020016100ee565b3480156101cc57600080fd5b5061014c60408051600060208201527f4f5b812789fc606be1b3b16908db13fc7a9adf7ca72641f84d75b47069d3d7f0918101919091526060016040516020818303038152906040528051906020012081565b34801561022b57600080fd5b506100bf610694565b34801561024057600080fd5b506002546101a8906001600160a01b031681565b34801561026057600080fd5b5061014c60015481565b34801561027657600080fd5b506100bf610285366004611026565b6106d9565b6001543410156102bf5760015460405163a458261b60e01b815234600482015260248101919091526044015b60405180910390fd5b60006102de82336001600160a01b0316316102d93361035d565b61074f565b60025460405163341a354960e11b81529192506001600160a01b0316906368346a9290610313908790879086906004016110bf565b6020604051808303816000875af1158015610332573d6000803e3d6000fd5b505050506040513d601f19601f8201168201806040525081019061035691906110f8565b5050505050565b60606e0c2e074ec69a0dfb2997ba6c7d2e1e3b6103835761037d8261086e565b92915050565b600061038e83610639565b604051630178b8bf60e01b8152600481018290529091506000906e0c2e074ec69a0dfb2997ba6c7d2e1e90630178b8bf90602401602060405180830381865afa1580156103df573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906104039190611111565b90506001600160a01b038116158061042357506001600160a01b0381163b155b15610439576104318461086e565b949350505050565b60405163691f343160e01b8152600481018390526001600160a01b0382169063691f343190602401600060405180830381865afa15801561047e573d6000803e3d6000fd5b505050506040513d6000823e601f3d908101601f191682016040526104a6919081019061112e565b925082516000036104ba576104318461086e565b60006104c68482610884565b604051630178b8bf60e01b8152600481018290529091506000906e0c2e074ec69a0dfb2997ba6c7d2e1e90630178b8bf90602401602060405180830381865afa158015610517573d6000803e3d6000fd5b505050506040513d601f19601f8201168201806040525081019061053b9190611111565b90506001600160a01b038116158061055b57506001600160a01b0381163b155b15610573576105698661086e565b9695505050505050565b604051631d9dabef60e11b8152600481018390526000906001600160a01b03831690633b3b57de90602401602060405180830381865afa1580156105bb573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906105df9190611111565b9050866001600160a01b0316816001600160a01b031603610604575050505050919050565b61060d8761086e565b979650505050505050565b6106206108fe565b600155565b61062d6108fe565b6106376000610958565b565b60007f91d1777781884d03a6757a803996e38de2a42967fb37eeaca72729271025a9e2610665836109a8565b604080516020810193909352820152606001604051602081830303815290604052805190602001209050919050565b61069c6108fe565b600080546040516001600160a01b03909116914780156108fc02929091818181858888f193505050501580156106d6573d6000803e3d6000fd5b50565b6106e16108fe565b6001600160a01b0381166107465760405162461bcd60e51b815260206004820152602660248201527f4f776e61626c653a206e6577206f776e657220697320746865207a65726f206160448201526564647265737360d01b60648201526084016102b6565b6106d681610958565b606083600061075d856109ff565b90506060845160000361077a576107733361086e565b905061077d565b50835b600083604051602001610790919061119c565b60408051601f19818403018152908290526107ad916020016111c5565b60405160208183030381529060405290506000826040516020016107d191906111eb565b60408051601f19818403018152908290526107ee91602001611215565b60408051601f198184030181529082905261080d91869060200161123b565b60408051601f198184030181529082905261082a9160200161126a565b60405160208183030381529060405290506000828260405160200161085092919061123b565b60408051808303601f190181529190529a9950505050505050505050565b606061037d6001600160a01b0383166014610cca565b6000818351116108965750600061037d565b60006108a28484610e6d565b90506108c2846108b283866112a5565b6108bd9060016112a5565b610884565b6108cd858584610ed0565b6040805160208101939093528201526060016040516020818303038152906040528051906020012091505092915050565b6000546001600160a01b031633146106375760405162461bcd60e51b815260206004820181905260248201527f4f776e61626c653a2063616c6c6572206973206e6f7420746865206f776e657260448201526064016102b6565b600080546001600160a01b038381166001600160a01b0319831681178455604051919092169283917f8be0079c531659141344cd1fd0a4f28419497f9722a3daafe3b4186f6b6457e09190a35050565b60006f181899199a1a9b1b9c1cb0b131b232b360811b60285b80156109f25760001901600f841682901a815360109093049260001901600f841682901a81536010840493506109c1565b5050602860002092915050565b60606000610a15670de0b6b3a7640000846112ce565b905060008115610a48575b610a2b81600a6113c6565b8210610a435780610a3b816113d2565b915050610a20565b610a4c565b5060015b6000610a598260036112a5565b67ffffffffffffffff811115610a7157610a71610f09565b6040519080825280601f01601f191660200182016040528015610a9b576020820181803683370190505b509050815b8015610b1157610ab1600a856113eb565b610abc9060306112a5565b60f81b82610acb6001846113ff565b81518110610adb57610adb611412565b60200101906001600160f81b031916908160001a905350610afd600a856112ce565b935080610b0981611428565b915050610aa0565b601760f91b828481518110610b2857610b28611412565b60200101906001600160f81b031916908160001a905350600a610b5367016345785d8a0000886112ce565b610b5d91906113eb565b610b689060306112a5565b60f81b82610b778560016112a5565b81518110610b8757610b87611412565b60200101906001600160f81b031916908160001a905350600a610bb1662386f26fc10000886112ce565b610bbb91906113eb565b610bc69060306112a5565b60f81b82610bd58560026112a5565b81518110610be557610be5611412565b60200101906001600160f81b031916908160001a90535081606046600503610c295750604080518082019091526005815264040ce8aa8960db1b6020820152610c9b565b46606403610c535750604080518082019091526005815264207844414960d81b6020820152610c9b565b46608903610c7e5750604080518082019091526006815265204d4154494360d01b6020820152610c9b565b506040805180820190915260048152630408aa8960e31b60208201525b8181604051602001610cae92919061123b565b6040516020818303038152906040529650505050505050919050565b60606000610cd983600261143f565b610ce49060026112a5565b67ffffffffffffffff811115610cfc57610cfc610f09565b6040519080825280601f01601f191660200182016040528015610d26576020820181803683370190505b509050600360fc1b81600081518110610d4157610d41611412565b60200101906001600160f81b031916908160001a905350600f60fb1b81600181518110610d7057610d70611412565b60200101906001600160f81b031916908160001a9053506000610d9484600261143f565b610d9f9060016112a5565b90505b6001811115610e17576f181899199a1a9b1b9c1cb0b131b232b360811b85600f1660108110610dd357610dd3611412565b1a60f81b828281518110610de957610de9611412565b60200101906001600160f81b031916908160001a90535060049490941c93610e1081611428565b9050610da2565b508315610e665760405162461bcd60e51b815260206004820181905260248201527f537472696e67733a20686578206c656e67746820696e73756666696369656e7460448201526064016102b6565b9392505050565b6000805b8351610e7d82856112a5565b14158015610eb9575083610e9182856112a5565b81518110610ea157610ea1611412565b6020910101516001600160f81b031916601760f91b14155b15610e665780610ec8816113d2565b915050610e71565b8251600090610edf83856112a5565b1115610eea57600080fd5b5091016020012090565b6001600160a01b03811681146106d657600080fd5b634e487b7160e01b600052604160045260246000fd5b604051601f8201601f1916810167ffffffffffffffff81118282101715610f4857610f48610f09565b604052919050565b600067ffffffffffffffff821115610f6a57610f6a610f09565b50601f01601f191660200190565b600080600060608486031215610f8d57600080fd5b833563ffffffff81168114610fa157600080fd5b92506020840135610fb181610ef4565b9150604084013567ffffffffffffffff811115610fcd57600080fd5b8401601f81018613610fde57600080fd5b8035610ff1610fec82610f50565b610f1f565b81815287602083850101111561100657600080fd5b816020840160208301376000602083830101528093505050509250925092565b60006020828403121561103857600080fd5b8135610e6681610ef4565b60005b8381101561105e578181015183820152602001611046565b50506000910152565b6000815180845261107f816020860160208601611043565b601f01601f19169290920160200192915050565b602081526000610e666020830184611067565b6000602082840312156110b857600080fd5b5035919050565b63ffffffff841681526001600160a01b03831660208201526060604082018190526000906110ef90830184611067565b95945050505050565b60006020828403121561110a57600080fd5b5051919050565b60006020828403121561112357600080fd5b8151610e6681610ef4565b60006020828403121561114057600080fd5b815167ffffffffffffffff81111561115757600080fd5b8201601f8101841361116857600080fd5b8051611176610fec82610f50565b81815285602083850101111561118b57600080fd5b6110ef826020830160208601611043565b602760f81b8152600082516111b8816001850160208701611043565b9190910160010192915050565b600082516111d7818460208701611043565b61138560f11b920191825250600201919050565b61016960f51b815260008251611208816002850160208701611043565b9190910160020192915050565b60008251611227818460208701611043565b61040560f31b920191825250600201919050565b6000835161124d818460208801611043565b835190830190611261818360208801611043565b01949350505050565b6000825161127c818460208701611043565b602960f81b920191825250600101919050565b634e487b7160e01b600052601160045260246000fd5b8082018082111561037d5761037d61128f565b634e487b7160e01b600052601260045260246000fd5b6000826112dd576112dd6112b8565b500490565b600181815b8085111561131d5781600019048211156113035761130361128f565b8085161561131057918102915b93841c93908002906112e7565b509250929050565b6000826113345750600161037d565b816113415750600061037d565b816001811461135757600281146113615761137d565b600191505061037d565b60ff8411156113725761137261128f565b50506001821b61037d565b5060208310610133831016604e8410600b84101617156113a0575081810a61037d565b6113aa83836112e2565b80600019048211156113be576113be61128f565b029392505050565b6000610e668383611325565b6000600182016113e4576113e461128f565b5060010190565b6000826113fa576113fa6112b8565b500690565b8181038181111561037d5761037d61128f565b634e487b7160e01b600052603260045260246000fd5b6000816114375761143761128f565b506000190190565b60008160001904831182151516156114595761145961128f565b50029056fea2646970667358221220a68f3c617c435c62f7e6432f8fa73f0251a40c45c58ec8e95fa11f51a2f5955164736f6c63430008100033
Verified Source Code Full Match
Compiler: v0.8.16+commit.07a7930e
EVM: london
Optimization: Yes (200 runs)
ENSHelper.sol 119 lines
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)
}
}
}
CrossChainMailbox.sol 59 lines
// 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;
}
}
FeeCollector.sol 22 lines
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);
}
}
StringHelper.sol 82 lines
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);
}
}
ENS.sol 64 lines
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);
}
ITelepathy.sol 70 lines
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);
}
TelepathyHandler.sol 29 lines
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;
}
IAddrResolver.sol 16 lines
// 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);
}
INameResolver.sol 14 lines
// 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);
}
Address.sol 244 lines
// 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);
}
}
}
Context.sol 24 lines
// 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;
}
}
Strings.sol 70 lines
// 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);
}
}
Ownable.sol 83 lines
// 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);
}
}
Math.sol 345 lines
// 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);
}
}
}
Read Contract
ADDR_REVERSE_NODE 0x7cf8a2eb → bytes32
ETH_NODE 0xcc473be3 → bytes32
fee 0xddca3f43 → uint256
getName 0x5fd4b08a → string
owner 0x8da5cb5b → address
reverseNode 0x7315937d → bytes32
telepathyRouter 0xdcc63290 → address
Write Contract 5 functions
These functions modify contract state and require a wallet transaction to execute.
claimFees 0xd294f093
No parameters
renounceOwnership 0x715018a6
No parameters
sendMail 0x49905b6d
uint32 _destinationChainId
address _destinationMailbox
bytes _message
setFee 0x69fe0e2d
uint256 _fee
transferOwnership 0xf2fde38b
address newOwner
Recent Transactions
No transactions found for this address