From e5349d4c6feeb779ce0e8d8b904e9377ac9d8b28 Mon Sep 17 00:00:00 2001 From: Geoff Stuart Date: Mon, 2 Dec 2024 18:40:48 -0500 Subject: [PATCH 1/3] Reorganize READMEs --- CONTRIBUTING.md | 8 +-- README.md | 107 ++++----------------------------- contracts/README.md | 2 +- contracts/teleporter/README.md | 87 ++++++++++++++++++++++++++- 4 files changed, 102 insertions(+), 102 deletions(-) diff --git a/CONTRIBUTING.md b/CONTRIBUTING.md index e3ac5926b..5dd02652e 100644 --- a/CONTRIBUTING.md +++ b/CONTRIBUTING.md @@ -31,13 +31,9 @@ To start developing on Teleporter, you'll need Solidity >= v0.8.25. [Foundry](ht ### Testing -#### Local +See [E2E Tests](./README.md#e2e-tests) -- Run the end-to-end tests - -```sh -./scripts/e2e_test.sh -``` +### Linting - Run the Solidity and Golang linters diff --git a/README.md b/README.md index 3a0af74cd..0af2901c1 100644 --- a/README.md +++ b/README.md @@ -1,46 +1,17 @@ -

- teleporter -

+# ICM Contracts -To get started with building ICM contracts, refer to [the avalanche-starter-kit repository](https://github.com/ava-labs/avalanche-starter-kit). This README is focused on the development of the `TeleporterMessenger` contract itself. +For help getting started with building ICM contracts, refer to [the avalanche-starter-kit repository](https://github.com/ava-labs/avalanche-starter-kit). -`TeleporterMessenger` is a smart contract that serves as the interface for ICM contracts to [Avalanche Interchain Messaging (ICM)](https://academy.avax.network/course/interchain-messaging/04-icm-basics/01-icm-basics). It provides a mechanism to asynchronously invoke smart contract functions on other EVM blockchains within Avalanche. `TeleporterMessenger` provides a handful of useful features of ICM, such as specifying relayer incentives for message delivery, replay protection, message delivery and execution retries, and a standard interface for sending and receiving messages within a dApp deployed across multiple Avalanche L1s. - -The `TeleporterMessenger` contract is a user-friendly interface to ICM, aimed at dApp developers. All of the message signing and verification is abstracted away from developers. Instead, developers simply call `sendCrossChainMessage` on the `TeleporterMessenger` contract to send a message invoking a smart contract on another Avalanche L1, and implement the `ITeleporterReceiver` interface to receive messages on the destination Avalanche L1. `TeleporterMessenger` handles all of the ICM message construction and sending, as well as the message delivery and execution. - -To get started with using `TeleporterMessenger`, see [How to Deploy ICM Enabled Avalanche L1s on a Local Network](https://docs.avax.network/tooling/cli-cross-chain/teleporter-on-local-networks) - -- [Deployed Addresses](#deployed-addresses) -- [A Note on Versioning](#a-note-on-versioning) - [Setup](#setup) - [Initialize the repository](#initialize-the-repository) - [Dependencies](#dependencies) - [Structure](#structure) - [E2E tests](#e2e-tests) - [Run specific E2E tests](#run-specific-e2e-tests) -- [Upgradability](#upgradability) -- [Deploy TeleporterMessenger to an L1](#deploy-teleportermessenger-to-an-avalanche-l1) -- [Deploy TeleporterRegistry to an L1](#deploy-teleporterregistry-to-an-avalanche-l1) - [ABI Bindings](#abi-bindings) - [Docs](#docs) - [Resources](#resources) -## Deployed Addresses - -| Contract | Address | Chain | -| --------------------- | ---------------------------------------------- | ------------------------ | -| `TeleporterMessenger` | **0x253b2784c75e510dD0fF1da844684a1aC0aa5fcf** | All chains, all networks | -| `TeleporterRegistry` | **0x7C43605E14F391720e1b37E49C78C4b03A488d98** | Mainnet C-Chain | -| `TeleporterRegistry` | **0xF86Cb19Ad8405AEFa7d09C778215D2Cb6eBfB228** | Fuji C-Chain | - -- Using [Nick's method](https://yamenmerhi.medium.com/nicks-method-ethereum-keyless-execution-168a6659479c#), `TeleporterMessenger` deploys at a universal address across all chains, varying with each `teleporter` Major release. **Compatibility exists only between same-version `TeleporterMessenger` instances.** See [TeleporterMessenger Contract Deployment](./utils/contract-deployment/README.md) and [Deploy TeleporterMessenger to an Avalanche L1](#deploy-teleportermessenger-to-an-avalanche-l1) for more details. - -- `TeleporterRegistry` can be deployed to any address. See [Deploy TeleporterRegistry to an Avalanche L1](#deploy-teleporterregistry-to-an-avalanche-l1) for details. The table above enumerates the canonical registry addresses on the Mainnet and Fuji C-Chains. - -## A Note on Versioning - -Release versions follow the [semver](https://semver.org/) convention of incompatible Major releases. A new Major version is released whenever the `TeleporterMessenger` bytecode is changed, and a new version of `TeleporterMessenger` is meant to be deployed. Due to the use of Nick's method to deploy the contract to the same address on all chains (see [TeleporterMessenger Contract Deployment](./utils/contract-deployment/README.md) for details), this also means that new release versions would result in different `TeleporterMessenger` contract addresses. Minor and Patch versions may pertain to contract changes that do not change the `TeleporterMessenger` bytecode, or to changes in the test frameworks, and will only be included in tags. - ## Setup ### Initialize the repository @@ -50,15 +21,18 @@ Release versions follow the [semver](https://semver.org/) convention of incompat ### Dependencies - [Ginkgo](https://onsi.github.io/ginkgo/#installing-ginkgo) for running the end-to-end tests. -- Docker and Docker Compose v2 for running the local test network. - - The docker image installs the following: - - [Foundry](https://book.getfoundry.sh/) Use `/scripts/install_foundry.sh` to install the Ava Labs [fork](https://github.com/ava-labs/foundry). - - [Python3](https://www.python.org/downloads/) +- [Foundry](https://book.getfoundry.sh/) Use `./scripts/install_foundry.sh` to install the Ava Labs [fork](https://github.com/ava-labs/foundry) for building contracts. ## Structure -- `contracts/` is a [Foundry](https://github.com/foundry-rs/foundry) project that includes the implementation of the `TeleporterMessenger` contract and example dApps that demonstrate how to write contracts that interact with Teleporter. +- `contracts/` + - [`governance/`](./contracts/governance/README.md) includes contracts related to L1 governance. + - [`ictt/`](./contracts/ictt/README.md) Interchain Token Transfer contracts. Facilities the transfer of tokens among L1s. + - [`teleporter/`](./contracts/teleporter/README.md) includes `TeleporterMessenger`, which serves as the interface for most contracts to use ICM. + - [`registry/`](./contracts/teleporter/registry/README.md) includes a registry contract for managing different versions of `TeleporterMessenger`. + - [`validator-manager/`](./contracts/validator-manager/README.md) includes contracts for managing the validator set of an L1. - `abi-bindings/` includes Go ABI bindings for the contracts in `contracts/`. +- [`audits/`](./audits/README.md) includes all audits conducted on contracts in this repository. - `tests/` includes integration tests for the contracts in `contracts/`, written using the [Ginkgo](https://onsi.github.io/ginkgo/) testing framework. - `utils/` includes Go utility functions for interacting with the contracts in `contracts/`. Included are Golang scripts to derive the expected EVM contract address deployed from a given EOA at a specific nonce, and also construct a transaction to deploy provided byte code to the same address on any EVM chain using [Nick's method](https://yamenmerhi.medium.com/nicks-method-ethereum-keyless-execution-168a6659479c#). - `scripts/` includes bash scripts for interacting with TeleporterMessenger in various environments, as well as utility scripts. @@ -91,70 +65,15 @@ A substring of the full test description can be used as well: GINKGO_FOCUS="Calculate Teleporter" ./scripts/e2e_test.sh ``` -The E2E tests also supports `GINKGO_LABEL_FILTER`, making it easy to group test cases and run them together. For example, to run all E2E tests for the example cross chain applications: - -```bash - ginkgo.It("Send native tokens from L1 A to B and back", - ginkgo.Label("cross chain apps"), - func() { - flows.NativeTokenBridge(LocalNetworkInstance) - }) -``` - -```bash -GINKGO_LABEL_FILTER="cross chain apps" ./scripts/e2e_test.sh -``` - -## Upgradability - -`TeleporterMessenger` is a non-upgradeable contract and can not be changed once it is deployed. This provides immutability to the contracts, and ensures that the contract's behavior at each address is unchanging. However, to allow for new features and potential bug fixes, new versions of `TeleporterMessenger` can be deployed to different addresses. The [TeleporterRegistry](./contracts/teleporter/TeleporterRegistry.sol) is used to keep track of the deployed versions of Teleporter, and to provide a standard interface for dApps to interact with the different `TeleporterMessenger` versions. - -`TeleporterRegistry` **is not mandatory** for dApps built on top of ICM, but dApp's are recommended to leverage the registry to ensure they use the latest `TeleporterMessenger` version available. Another recommendation standard is to have a single canonical `TeleporterRegistry` for each Avalanche L1, and unlike the `TeleporterMessenger` contract, the registry does not need to be deployed to the same address on every chain. This means the registry does not need a Nick's method deployment, and can be at different contract addresses on different chains. - -For more information on the registry and how to integrate with ICM contracts, see the [Upgradability doc](./contracts/teleporter/registry/README.md). - -## Deploy TeleporterMessenger to an Avalanche L1 - -From the root of the repo, the TeleporterMessenger contract can be deployed by calling - -```bash -./scripts/deploy_teleporter.sh --version --rpc-url [OPTIONS] -``` - -Required arguments: - -- `--version ` Specify the release version to deploy. These will all be of the form `v1.X.0`. Each `TeleporterMessenger` version can only send and receive messages from the **same** `TeleporterMessenger` version on another chain. You can see a list of released versions at https://github.com/ava-labs/teleporter/releases. -- `--rpc-url ` Specify the rpc url of the node to use. - -Options: - -- `--private-key ` Funds the deployer address with the account held by `` - -To ensure that `TeleporterMessenger` can be deployed to the same address on every EVM based chain, it uses [Nick's Method](https://yamenmerhi.medium.com/nicks-method-ethereum-keyless-execution-168a6659479c) to deploy from a static deployer address. Teleporter costs exactly `10eth` in the Avalanche L1's native gas token to deploy, which must be sent to the deployer address. - -`deploy_teleporter.sh` will send the necessary native tokens to the deployer address if it is provided with a private key for an account with sufficient funds. Alternatively, the deployer address can be funded externally. The deployer address for each version can be found by looking up the appropriate version at https://github.com/ava-labs/teleporter/releases and downloading `TeleporterMessenger_Deployer_Address_.txt`. - -Alternatively for new Avalanche L1s, the `TeleporterMessenger` contract can be directly included in the genesis file as documented [here](./contracts/teleporter/README.md#teleporter-messenger-contract-deployment). - -## Deploy TeleporterRegistry to an Avalanche L1 - -There should only be one canonical `TeleporterRegistry` deployed for each chain, but if one does not exist, it is recommended to deploy the registry so ICM contracts can always use the most recent `TeleporterMessenger` version available. The registry does not need to be deployed to the same address on every chain, and therefore does not need a Nick's method transaction. To deploy, run the following command from the root of the repository: +The E2E test script also supports a `--components` flag, making it easy to run all the test cases for a particular project. For example, to run all E2E tests for the `tests/flows/ictt/` folder: ```bash -./scripts/deploy_registry.sh --version --rpc-url --private-key [OPTIONS] +./scripts/e2e_test.sh --components "ictt" ``` -Required arguments: - -- `--version ` Specify the release version to deploy. These will all be of the form `v1.X.0`. -- `--rpc-url ` Specify the rpc url of the node to use. -- `--private-key ` Funds the deployer address with the account held by `` - -`deploy_registry.sh` will deploy a new `TeleporterRegistry` contract for the intended release version, and will also have the corresponding `TeleporterMessenger` contract registered as the initial protocol version. - ## ABI Bindings -To generate Golang ABI bindings for the Solidity smart contracts, run: +The E2E tests written in Golang interface with the solidity contracts by use of generated ABI bindings. To regenerate Golang ABI bindings for the Solidity smart contracts, run: ```bash ./scripts/abi_go_bindings.sh diff --git a/contracts/README.md b/contracts/README.md index 12d85f932..64694aae4 100644 --- a/contracts/README.md +++ b/contracts/README.md @@ -1,7 +1,7 @@ ## Contracts This directory contains Solidity contracts that leverage Avalanche Interchain Messaging (ICM) to implement unique cross-chain functions. -This directory is set up as a [Foundry](https://github.com/foundry-rs/foundry) project. Use the `scripts/install_foundry.sh` to install the correct version of the ava-labs fork of foundry. +This repository is set up as a [Foundry](https://github.com/foundry-rs/foundry) project. Use the `scripts/install_foundry.sh` to install the correct version of the ava-labs fork of foundry. ## Building and Running - To compile the contracts run `forge build` from this directory. diff --git a/contracts/teleporter/README.md b/contracts/teleporter/README.md index 5689b8cf0..d1da2f750 100644 --- a/contracts/teleporter/README.md +++ b/contracts/teleporter/README.md @@ -1,8 +1,27 @@ # ICM Protocol +- [Overview](#overview) +- [Data Flow](#data-flow) +- [Properties](#properties) +- [Fees](#fees) +- [Message Receipts and Fee Redemption](#message-receipts-and-fee-redemption) +- [Required Interface](#required-interface) +- [Message Delivery and Execution](#message-delivery-and-execution) +- [Resending a Message](#resending-a-message) +- [TeleporterMessenger Contract Deployment](#teleportermessenger-contract-deployment) +- [Deployed Addresses](#deployed-addresses) +- [A Note on Versioning](#a-note-on-versioning) +- [Upgradability](#upgradability) +- [Deploy TeleporterMessenger to an L1](#deploy-teleportermessenger-to-an-avalanche-l1) +- [Deploy TeleporterRegistry to an L1](#deploy-teleporterregistry-to-an-avalanche-l1) + ## Overview -`TeleporterMessenger` is a smart contract that is part of [Avalanche Interchain Messaging (ICM)](https://docs.avax.network/learn/avalanche/awm) that provides a developer-friendly interface for sending and receiving cross-chain messages from within the EVM. +`TeleporterMessenger` is a smart contract that serves as the interface for ICM contracts to [Avalanche Interchain Messaging (ICM)](https://academy.avax.network/course/interchain-messaging/04-icm-basics/01-icm-basics). It provides a mechanism to asynchronously invoke smart contract functions on other EVM blockchains within Avalanche. `TeleporterMessenger` provides a handful of useful features of ICM, such as specifying relayer incentives for message delivery, replay protection, message delivery and execution retries, and a standard interface for sending and receiving messages within a dApp deployed across multiple Avalanche L1s. + +The `TeleporterMessenger` contract is a user-friendly interface to ICM, aimed at dApp developers. All of the message signing and verification is abstracted away from developers. Instead, developers simply call `sendCrossChainMessage` on the `TeleporterMessenger` contract to send a message invoking a smart contract on another Avalanche L1, and implement the `ITeleporterReceiver` interface to receive messages on the destination Avalanche L1. `TeleporterMessenger` handles all of the ICM message construction and sending, as well as the message delivery and execution. + +To get started with using `TeleporterMessenger`, see [How to Deploy ICM Enabled Avalanche L1s on a Local Network](https://docs.avax.network/tooling/cli-cross-chain/teleporter-on-local-networks) The `ITeleporterMessenger` interface provides two primary methods: @@ -67,6 +86,9 @@ Once sent on chain, ICM messages cannot be re-signed by a new validator set in s ## TeleporterMessenger Contract Deployment +> [!CAUTION] +> DO NOT USE UN-AUDITED CODE IN PRODUCTION! + **Do not deploy the `TeleporterMessenger` contract using `forge create`**. The `TeleporterMessenger` contract must be deployed to the same contract address on every chain. To achieve this, the contract can be deployed using a static transaction that uses Nick's method as documented in [this guide](../..//utils/contract-deployment/README.md). Alternatively, if creating a new L1, the contract can be pre-allocated with the proper address and state in the new chain's [genesis file](https://docs.avax.network/build/subnet/upgrade/customize-a-subnet#setting-the-genesis-allocation). As an example, to include `TeleporterMessenger` `v1.0.0` in the genesis file, include the following values in the `alloc` settings, as documented at the link above. The `storage` values included below correspond to the two contract values that are initialized as part of the default constructor of `TeleporterMessenger`. These are the `ReentrancyGuard` values set in this [abstract contract](../utilities/ReentrancyGuards.sol). Future versions of `TeleporterMessenger` may require different storage value initializations. @@ -89,3 +111,66 @@ As an example, to include `TeleporterMessenger` `v1.0.0` in the genesis file, in ``` The values above are taken from the `v1.0.0` [release artifacts](https://github.com/ava-labs/teleporter/releases/tag/v1.0.0). The contract address, deployed bytecode, and deployer address are unique per major release. All of the other values should remain the same. + +## Deployed Addresses + +| Contract | Address | Chain | +| --------------------- | ---------------------------------------------- | ------------------------ | +| `TeleporterMessenger` | **0x253b2784c75e510dD0fF1da844684a1aC0aa5fcf** | All chains, all networks | +| `TeleporterRegistry` | **0x7C43605E14F391720e1b37E49C78C4b03A488d98** | Mainnet C-Chain | +| `TeleporterRegistry` | **0xF86Cb19Ad8405AEFa7d09C778215D2Cb6eBfB228** | Fuji C-Chain | + +- Using [Nick's method](https://yamenmerhi.medium.com/nicks-method-ethereum-keyless-execution-168a6659479c#), `TeleporterMessenger` deploys at a universal address across all chains, varying with each `teleporter` Major release. **Compatibility exists only between same-version `TeleporterMessenger` instances.** See [TeleporterMessenger Contract Deployment](./utils/contract-deployment/README.md) and [Deploy TeleporterMessenger to an Avalanche L1](#deploy-teleportermessenger-to-an-avalanche-l1) for more details. + +- `TeleporterRegistry` can be deployed to any address. See [Deploy TeleporterRegistry to an Avalanche L1](#deploy-teleporterregistry-to-an-avalanche-l1) for details. The table above enumerates the canonical registry addresses on the Mainnet and Fuji C-Chains. + +## A Note on Versioning + +Release versions follow the [semver](https://semver.org/) convention of incompatible Major releases. A new Major version is released whenever the `TeleporterMessenger` bytecode is changed, and a new version of `TeleporterMessenger` is meant to be deployed. Due to the use of Nick's method to deploy the contract to the same address on all chains (see [TeleporterMessenger Contract Deployment](./utils/contract-deployment/README.md) for details), this also means that new release versions would result in different `TeleporterMessenger` contract addresses. Minor and Patch versions may pertain to contract changes that do not change the `TeleporterMessenger` bytecode, or to changes in the test frameworks, and will only be included in tags. + +## Upgradability + +`TeleporterMessenger` is a non-upgradeable contract and can not be changed once it is deployed. This provides immutability to the contracts, and ensures that the contract's behavior at each address is unchanging. However, to allow for new features and potential bug fixes, new versions of `TeleporterMessenger` can be deployed to different addresses. The [TeleporterRegistry](./contracts/teleporter/TeleporterRegistry.sol) is used to keep track of the deployed versions of Teleporter, and to provide a standard interface for dApps to interact with the different `TeleporterMessenger` versions. + +`TeleporterRegistry` **is not mandatory** for dApps built on top of ICM, but dApp's are recommended to leverage the registry to ensure they use the latest `TeleporterMessenger` version available. Another recommendation standard is to have a single canonical `TeleporterRegistry` for each Avalanche L1, and unlike the `TeleporterMessenger` contract, the registry does not need to be deployed to the same address on every chain. This means the registry does not need a Nick's method deployment, and can be at different contract addresses on different chains. + +For more information on the registry and how to integrate with ICM contracts, see the [Upgradability doc](./contracts/teleporter/registry/README.md). + +## Deploy TeleporterMessenger to an Avalanche L1 + +From the root of the repo, the TeleporterMessenger contract can be deployed by calling + +```bash +./scripts/deploy_teleporter.sh --version --rpc-url [OPTIONS] +``` + +Required arguments: + +- `--version ` Specify the release version to deploy. These will all be of the form `v1.X.0`. Each `TeleporterMessenger` version can only send and receive messages from the **same** `TeleporterMessenger` version on another chain. You can see a list of released versions at https://github.com/ava-labs/teleporter/releases. +- `--rpc-url ` Specify the rpc url of the node to use. + +Options: + +- `--private-key ` Funds the deployer address with the account held by `` + +To ensure that `TeleporterMessenger` can be deployed to the same address on every EVM based chain, it uses [Nick's Method](https://yamenmerhi.medium.com/nicks-method-ethereum-keyless-execution-168a6659479c) to deploy from a static deployer address. Teleporter costs exactly `10eth` in the Avalanche L1's native gas token to deploy, which must be sent to the deployer address. + +`deploy_teleporter.sh` will send the necessary native tokens to the deployer address if it is provided with a private key for an account with sufficient funds. Alternatively, the deployer address can be funded externally. The deployer address for each version can be found by looking up the appropriate version at https://github.com/ava-labs/teleporter/releases and downloading `TeleporterMessenger_Deployer_Address_.txt`. + +Alternatively for new Avalanche L1s, the `TeleporterMessenger` contract can be directly included in the genesis file as documented [here](./contracts/teleporter/README.md#teleporter-messenger-contract-deployment). + +## Deploy TeleporterRegistry to an Avalanche L1 + +There should only be one canonical `TeleporterRegistry` deployed for each chain, but if one does not exist, it is recommended to deploy the registry so ICM contracts can always use the most recent `TeleporterMessenger` version available. The registry does not need to be deployed to the same address on every chain, and therefore does not need a Nick's method transaction. To deploy, run the following command from the root of the repository: + +```bash +./scripts/deploy_registry.sh --version --rpc-url --private-key [OPTIONS] +``` + +Required arguments: + +- `--version ` Specify the release version to deploy. These will all be of the form `v1.X.0`. +- `--rpc-url ` Specify the rpc url of the node to use. +- `--private-key ` Funds the deployer address with the account held by `` + +`deploy_registry.sh` will deploy a new `TeleporterRegistry` contract for the intended release version, and will also have the corresponding `TeleporterMessenger` contract registered as the initial protocol version. \ No newline at end of file From c7714a2a5a112550bd158355f50bdd71dd6044b4 Mon Sep 17 00:00:00 2001 From: Geoff Stuart Date: Tue, 3 Dec 2024 11:10:01 -0500 Subject: [PATCH 2/3] Update README.md Co-authored-by: bernard-avalabs <53795885+bernard-avalabs@users.noreply.github.com> Signed-off-by: Geoff Stuart --- README.md | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/README.md b/README.md index 0af2901c1..5ec757c13 100644 --- a/README.md +++ b/README.md @@ -27,7 +27,7 @@ For help getting started with building ICM contracts, refer to [the avalanche-st - `contracts/` - [`governance/`](./contracts/governance/README.md) includes contracts related to L1 governance. - - [`ictt/`](./contracts/ictt/README.md) Interchain Token Transfer contracts. Facilities the transfer of tokens among L1s. + - [`ictt/`](./contracts/ictt/README.md) Interchain Token Transfer contracts. Facilitates the transfer of tokens among L1s. - [`teleporter/`](./contracts/teleporter/README.md) includes `TeleporterMessenger`, which serves as the interface for most contracts to use ICM. - [`registry/`](./contracts/teleporter/registry/README.md) includes a registry contract for managing different versions of `TeleporterMessenger`. - [`validator-manager/`](./contracts/validator-manager/README.md) includes contracts for managing the validator set of an L1. From 89565b90c25173b1ce9401ebf3c57027b976f483 Mon Sep 17 00:00:00 2001 From: Geoff Stuart Date: Tue, 3 Dec 2024 11:10:26 -0500 Subject: [PATCH 3/3] Update contracts/teleporter/README.md Co-authored-by: bernard-avalabs <53795885+bernard-avalabs@users.noreply.github.com> Signed-off-by: Geoff Stuart --- contracts/teleporter/README.md | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/contracts/teleporter/README.md b/contracts/teleporter/README.md index d1da2f750..ec73df0a6 100644 --- a/contracts/teleporter/README.md +++ b/contracts/teleporter/README.md @@ -17,7 +17,7 @@ ## Overview -`TeleporterMessenger` is a smart contract that serves as the interface for ICM contracts to [Avalanche Interchain Messaging (ICM)](https://academy.avax.network/course/interchain-messaging/04-icm-basics/01-icm-basics). It provides a mechanism to asynchronously invoke smart contract functions on other EVM blockchains within Avalanche. `TeleporterMessenger` provides a handful of useful features of ICM, such as specifying relayer incentives for message delivery, replay protection, message delivery and execution retries, and a standard interface for sending and receiving messages within a dApp deployed across multiple Avalanche L1s. +`TeleporterMessenger` is a smart contract that serves as the interface for ICM contracts to [Avalanche Interchain Messaging (ICM)](https://academy.avax.network/course/interchain-messaging/04-icm-basics/01-icm-basics). It provides a mechanism to asynchronously invoke smart contract functions on other EVM L1s within Avalanche. `TeleporterMessenger` provides a handful of useful features to ICM, such as specifying relayer incentives for message delivery, replay protection, message delivery and execution retries, and a standard interface for sending and receiving messages within a dApp deployed across multiple Avalanche L1s. The `TeleporterMessenger` contract is a user-friendly interface to ICM, aimed at dApp developers. All of the message signing and verification is abstracted away from developers. Instead, developers simply call `sendCrossChainMessage` on the `TeleporterMessenger` contract to send a message invoking a smart contract on another Avalanche L1, and implement the `ITeleporterReceiver` interface to receive messages on the destination Avalanche L1. `TeleporterMessenger` handles all of the ICM message construction and sending, as well as the message delivery and execution.