编译器
0.8.16+commit.07a7930e
文件 1 的 14:Address.sol
pragma solidity ^0.8.1;
library Address {
function isContract(address account) internal view returns (bool) {
return account.code.length > 0;
}
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");
}
function functionCall(address target, bytes memory data) internal returns (bytes memory) {
return functionCallWithValue(target, data, 0, "Address: low-level call failed");
}
function functionCall(
address target,
bytes memory data,
string memory errorMessage
) internal returns (bytes memory) {
return functionCallWithValue(target, data, 0, errorMessage);
}
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");
}
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);
}
function functionStaticCall(address target, bytes memory data) internal view returns (bytes memory) {
return functionStaticCall(target, data, "Address: low-level static call failed");
}
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);
}
function functionDelegateCall(address target, bytes memory data) internal returns (bytes memory) {
return functionDelegateCall(target, data, "Address: low-level delegate call failed");
}
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);
}
function verifyCallResultFromTarget(
address target,
bool success,
bytes memory returndata,
string memory errorMessage
) internal view returns (bytes memory) {
if (success) {
if (returndata.length == 0) {
require(isContract(target), "Address: call to non-contract");
}
return returndata;
} else {
_revert(returndata, errorMessage);
}
}
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 {
if (returndata.length > 0) {
assembly {
let returndata_size := mload(returndata)
revert(add(32, returndata), returndata_size)
}
} else {
revert(errorMessage);
}
}
}
文件 2 的 14:Context.sol
pragma solidity ^0.8.0;
abstract contract Context {
function _msgSender() internal view virtual returns (address) {
return msg.sender;
}
function _msgData() internal view virtual returns (bytes calldata) {
return msg.data;
}
}
文件 3 的 14:CrossChainMailbox.sol
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";
contract CrossChainMailer is FeeCollector, ENSHelper {
ITelepathyRouter public telepathyRouter;
constructor(address _telepathyRouter) {
telepathyRouter = ITelepathyRouter(_telepathyRouter);
}
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));
}
}
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;
}
}
文件 4 的 14:ENS.sol
pragma solidity >=0.8.4;
interface ENS {
event NewOwner(bytes32 indexed node, bytes32 indexed label, address owner);
event Transfer(bytes32 indexed node, address owner);
event NewResolver(bytes32 indexed node, address resolver);
event NewTTL(bytes32 indexed node, uint64 ttl);
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);
}
文件 5 的 14:ENSHelper.sol
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;
address constant ensRegistryAddr = 0x00000000000C2E074eC69A0dFb2997BA6C7d2e1e;
bytes32 public constant ETH_NODE = keccak256(abi.encodePacked(bytes32(0), keccak256("eth")));
function getName(address _addr) public view returns (string memory name) {
if (!ensRegistryAddr.isContract()) {
return Strings.toHexString(_addr);
}
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);
}
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);
}
}
bytes32 public constant ADDR_REVERSE_NODE = 0x91d1777781884d03a6757a803996e38de2a42967fb37eeaca72729271025a9e2;
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;
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)
}
}
}
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)
}
}
}
文件 6 的 14:FeeCollector.sol
pragma solidity ^0.8.16;
import {Ownable} from "openzeppelin-contracts/contracts/access/Ownable.sol";
contract FeeCollector is Ownable {
error InsufficientFee(uint256 actual, uint256 expected);
uint256 public fee;
function setFee(uint256 _fee) external onlyOwner {
fee = _fee;
}
function claimFees() external onlyOwner {
payable(owner()).transfer(address(this).balance);
}
}
文件 7 的 14:IAddrResolver.sol
pragma solidity >=0.8.4;
interface IAddrResolver {
event AddrChanged(bytes32 indexed node, address a);
function addr(bytes32 node) external view returns (address payable);
}
文件 8 的 14:INameResolver.sol
pragma solidity >=0.8.4;
interface INameResolver {
event NameChanged(bytes32 indexed node, string name);
function name(bytes32 node) external view returns (string memory);
}
文件 9 的 14:ITelepathy.sol
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,
bytes memory txIndexRLPEncoded,
uint256 logIndex
) external;
}
interface ITelepathyHandler {
function handleTelepathy(uint32 _sourceChainId, address _senderAddress, bytes memory _data)
external
returns (bytes4);
}
文件 10 的 14:Math.sol
pragma solidity ^0.8.0;
library Math {
enum Rounding {
Down,
Up,
Zero
}
function max(uint256 a, uint256 b) internal pure returns (uint256) {
return a > b ? a : b;
}
function min(uint256 a, uint256 b) internal pure returns (uint256) {
return a < b ? a : b;
}
function average(uint256 a, uint256 b) internal pure returns (uint256) {
return (a & b) + (a ^ b) / 2;
}
function ceilDiv(uint256 a, uint256 b) internal pure returns (uint256) {
return a == 0 ? 0 : (a - 1) / b + 1;
}
function mulDiv(
uint256 x,
uint256 y,
uint256 denominator
) internal pure returns (uint256 result) {
unchecked {
uint256 prod0;
uint256 prod1;
assembly {
let mm := mulmod(x, y, not(0))
prod0 := mul(x, y)
prod1 := sub(sub(mm, prod0), lt(mm, prod0))
}
if (prod1 == 0) {
return prod0 / denominator;
}
require(denominator > prod1);
uint256 remainder;
assembly {
remainder := mulmod(x, y, denominator)
prod1 := sub(prod1, gt(remainder, prod0))
prod0 := sub(prod0, remainder)
}
uint256 twos = denominator & (~denominator + 1);
assembly {
denominator := div(denominator, twos)
prod0 := div(prod0, twos)
twos := add(div(sub(0, twos), twos), 1)
}
prod0 |= prod1 * twos;
uint256 inverse = (3 * denominator) ^ 2;
inverse *= 2 - denominator * inverse;
inverse *= 2 - denominator * inverse;
inverse *= 2 - denominator * inverse;
inverse *= 2 - denominator * inverse;
inverse *= 2 - denominator * inverse;
inverse *= 2 - denominator * inverse;
result = prod0 * inverse;
return result;
}
}
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;
}
function sqrt(uint256 a) internal pure returns (uint256) {
if (a == 0) {
return 0;
}
uint256 result = 1 << (log2(a) >> 1);
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);
}
}
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);
}
}
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;
}
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);
}
}
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;
}
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);
}
}
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;
}
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);
}
}
}
文件 11 的 14:Ownable.sol
pragma solidity ^0.8.0;
import "../utils/Context.sol";
abstract contract Ownable is Context {
address private _owner;
event OwnershipTransferred(address indexed previousOwner, address indexed newOwner);
constructor() {
_transferOwnership(_msgSender());
}
modifier onlyOwner() {
_checkOwner();
_;
}
function owner() public view virtual returns (address) {
return _owner;
}
function _checkOwner() internal view virtual {
require(owner() == _msgSender(), "Ownable: caller is not the owner");
}
function renounceOwnership() public virtual onlyOwner {
_transferOwnership(address(0));
}
function transferOwnership(address newOwner) public virtual onlyOwner {
require(newOwner != address(0), "Ownable: new owner is the zero address");
_transferOwnership(newOwner);
}
function _transferOwnership(address newOwner) internal virtual {
address oldOwner = _owner;
_owner = newOwner;
emit OwnershipTransferred(oldOwner, newOwner);
}
}
文件 12 的 14:StringHelper.sol
pragma solidity ^0.8.16;
import {Strings} from "openzeppelin-contracts/contracts/utils/Strings.sol";
library StringHelper {
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);
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;
}
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);
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);
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);
}
}
文件 13 的 14:Strings.sol
pragma solidity ^0.8.0;
import "./math/Math.sol";
library Strings {
bytes16 private constant _SYMBOLS = "0123456789abcdef";
uint8 private constant _ADDRESS_LENGTH = 20;
function toString(uint256 value) internal pure returns (string memory) {
unchecked {
uint256 length = Math.log10(value) + 1;
string memory buffer = new string(length);
uint256 ptr;
assembly {
ptr := add(buffer, add(32, length))
}
while (true) {
ptr--;
assembly {
mstore8(ptr, byte(mod(value, 10), _SYMBOLS))
}
value /= 10;
if (value == 0) break;
}
return buffer;
}
}
function toHexString(uint256 value) internal pure returns (string memory) {
unchecked {
return toHexString(value, Math.log256(value) + 1);
}
}
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);
}
function toHexString(address addr) internal pure returns (string memory) {
return toHexString(uint256(uint160(addr)), _ADDRESS_LENGTH);
}
}
文件 14 的 14:TelepathyHandler.sol
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;
}
{
"compilationTarget": {
"contracts/src/CrossChainMailbox.sol": "CrossChainMailer"
},
"evmVersion": "london",
"libraries": {},
"metadata": {
"bytecodeHash": "ipfs"
},
"optimizer": {
"enabled": true,
"runs": 200
},
"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/"
]
}
[{"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"}]