Skip to content

Commit

Permalink
Merge remote-tracking branch 'origin/main' into feat/fix-its-intercha…
Browse files Browse the repository at this point in the history
…in-transfer-event
  • Loading branch information
Foivos committed Nov 20, 2024
2 parents faa0ae7 + d0bfec4 commit 63075d6
Show file tree
Hide file tree
Showing 25 changed files with 1,270 additions and 404 deletions.
5 changes: 5 additions & 0 deletions .changeset/dull-humans-fold.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,5 @@
---
'@axelar-network/axelar-cgp-sui': minor
---

added randomness to all axelar gateway tests
5 changes: 5 additions & 0 deletions .changeset/late-goats-smell.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,5 @@
---
'@axelar-network/axelar-cgp-sui': minor
---

Add events to register and remove transaction on relayer discovery
5 changes: 5 additions & 0 deletions .changeset/wild-eagles-unite.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,5 @@
---
'@axelar-network/axelar-cgp-sui': minor
---

move all events of gas service to gas_service::events and all logic to the versioned contract.
118 changes: 118 additions & 0 deletions INTEGRATION.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,118 @@
# Sui Axelar Gateway and ITS Integration

Sui is a smart contract chain that offers some unique design parameters. For this reason there have been a few differences between the implementation of smart contracts on Sui compared to EVM chains, that are used as reference

## General Limitations

| Limit | Minimum | Recommended | Sui Gateway |
|-|-|-|-|
| Cross-chain Message Size | 16 KB | 64 KB | 16 KB |
| Chain Name Length | < 20 ASCII chars | - | - |
| Signer Set Size | 40 signers | 100 signers | 313 |
| Signature Verification | 27 signatures | 67 signatures | 140 |
| Message Approval Batching | 1 | Configurable | > 105 |
| Storage Limit for Messages | Practically unlimited (2^64) | Practically unlimited (2^64) | Practically unlimited |
| Event Retention | 2 weeks | 2 months | Unlimited |

### Design Limitation

There are quite a few differences between EVM and Sui.

In Sui, there are packages and modules that can run code, similar to EVM smart contracts. A big difference however is that Sui packages/modules do not have any storage available. Storage as a concept only exists on objects, which can be defined and controlled by modules. To access and modify any storage the corresponding object needs to be available. This means that conventional EVM smart contracts effectively combine the functionality of modules and objects in Sui.

Additionally there is a lack of interfaces whatsoever. It is impossible for a module to ever call another module that is published at a later time. This means applications that want to interface with future applications must be called by those future applications, but need to only call pre-existing ones. To expand on this, we expect contract calls that are received to potentially modify the storage of multiple objects in a single call, which makes it impossible to require modules to implement a 'standardized' function that a relayer will call, because the number of arguments required varies depending on the application (or the type of call).

Package "upgrades" result in a new, upgraded package being created, while preserving the old package. This new package can modify objects created by the old package, but any external modules that are using the old package will continued to do so, until they are upgraded too. This means that maintaining packages that depend on multiple existing packages is quite difficult if the dependancies are expected to upgrade.

Finally, we do not want to require the payload of incoming calls to have a certain format, because that would mean that working applications that want to exapnd to Sui need to redesign their working protocoll to accomodate Sui, discouraging them from doing so.

### Basic Limitations

See [this doc](https://move-book.com/guides/building-against-limits.html) for reference. The gas limit for sui is 50 SUI, which is over 100 times larger than any transaction needed for this context. There is also a transaction size limit of 128KB. Objects cannot be more than 256KB in size and there cannot be more than 1024 dynamic fields created on an object on a single transaction. See below for how these limitations affect potential designs.

## Axelar Gateway

The Axelar gateway is the remote interface for the Axelar Network to interract with any given chain. The reference implementation for such a smart contract can be found [here](https://github.com/axelarnetwork/axelar-cgp-solidity), with some design requirements found specifically [here](https://github.com/axelarnetwork/axelar-gmp-sdk-solidity/blob/main/contracts/gateway/INTEGRATION.md)

The Sui message sending looks like this

```mermaid
flowchart LR
User(User)
SG[Source Gateway]
MT[(MessageTicket)]
User -->|prepare_message| SG --> MT --> |send_message| SG
SG -.->|ContractCall event| SG
SG -->|Confirm event| AXL
```

Receiving calls looks like this (see below for relayer discovery)

```mermaid
flowchart LR
DG[Destination Gateway]
RD[Relayer Discovery]
AXL{Axelar Amplifier}
Contract[Destination contract]
Transaction[(Transaction)]
AXL -->|approve_messages| DG
Relayer -->|consume_approved_message| DG --> |ApprovedMessage| Transaction
Relayer -->|get_transaction| RD --> Transaction --> Contract
```

The verifier/signer rotation flow looks like the following:

```mermaid
flowchart LR
AXL{Axelar Amplifier}
DG[Destination Gateway]
AXL -.->|Verifier set change| AXL
AXL -->|rotate_signers| DG
DG -.->|SignersRotated event| DG
DG --->|Confirm event| AXL
```
## Relayer Discovery

The only information about an incoming contract call through axelar are

- Command Id: A unique identifier for the command approving the call, 32 bytes long.
- Source Chain: The name of the source chain, as a String
- Source Address: The caller, as a String.
- Destination Chain: “Sui” for the purposes of this document
- Destination Address: The destination, a Sui address.

As we already explained, the destination address will be the `ID` of a `GatewayChannel` object that needs to be provided to consume the call. However there is no way for a relayer to know what they are supposed to call to get the call to be consumed, since they should not have access to the `GatewayChannel` object itself. This creates the need to make the destination for each incoming call publicly known so that any relayer can query it.

### Transaction Object

The following objects are defined in the `axelar::discovery` module and can be used to register/decode transactions.

```rust
struct Function has store, copy, drop {
package_id: address,
module_name: String,
name: String,
}

/// Arguments are prefixed with:
/// - 0 for objects followed by exactly 32 bytes that cointain the object id
/// - 1 for pures followed by the bcs encoded form of the pure object
/// - 2 for the ApprovedCall object, followed by nothing (to be passed into the target function)
/// - 3 for the payload of the contract call (to be passed into the intermediate function)
/// - 4 for an argument returned from a previous move call, followed by a u8 specified which call to get the return of (0 for the first transaction AFTER the one that gets ApprovedCall out), and then another u8 specifying which argument to input.
struct MoveCall has store, copy, drop {
function: Function,
arguments: vector<vector<u8>>,
type_arguments: vector<String>,
}

public struct Transaction has store, copy, drop {
is_final: bool,
move_calls: vector<MoveCall>,
}
```

As seen above there are some options to allow for dynamic objects to be referenced that could not be known on chain, since they do not exist on chain (the `ApprovedCall` object, the payload of the incoming call and the returned arguments of a previous `MoveCall`).
8 changes: 5 additions & 3 deletions move/axelar_gateway/sources/auth.move
Original file line number Diff line number Diff line change
Expand Up @@ -165,10 +165,11 @@ use axelar_gateway::proof;

#[test_only]
public fun dummy(ctx: &mut TxContext): AxelarSigners {
let mut rng = sui::random::new_generator_for_testing();
AxelarSigners {
epoch: 0,
epoch_by_signers_hash: table::new(ctx),
domain_separator: bytes32::new(@0x1),
domain_separator: bytes32::from_bytes(rng.generate_bytes(32)),
minimum_rotation_delay: 1,
last_rotation_timestamp: 0,
previous_signers_retention: 3,
Expand Down Expand Up @@ -262,13 +263,14 @@ fun test_update_rotation_timestamp_insufficient_rotation_delay() {
#[test]
#[expected_failure(abort_code = EInvalidEpoch)]
fun test_validate_proof_invalid_epoch() {
let mut rng = sui::random::new_generator_for_testing();
let ctx = &mut sui::tx_context::dummy();
let mut self = new(ctx);
let proof = axelar_gateway::proof::create_for_testing(
axelar_gateway::weighted_signers::create_for_testing(
vector[],
0,
bytes32::new(@0x0),
bytes32::from_bytes(rng.generate_bytes(32)),
),
vector[],
);
Expand All @@ -281,7 +283,7 @@ fun test_validate_proof_invalid_epoch() {
self.epoch = 3;

self.validate_proof(
bytes32::new(@0x0),
bytes32::from_bytes(rng.generate_bytes(32)),
proof,
);

Expand Down
13 changes: 8 additions & 5 deletions move/axelar_gateway/sources/channel.move
Original file line number Diff line number Diff line change
Expand Up @@ -217,11 +217,12 @@ fun test_to_address() {

#[test]
fun test_create_approved_message() {
let mut rng = sui::random::new_generator_for_testing();
let input_source_chain = std::ascii::string(b"Source Chain");
let input_message_id = std::ascii::string(b"message id");
let input_source_address = std::ascii::string(b"Source Address");
let input_destination_id = @0x5678;
let input_payload = b"payload";
let input_destination_id = sui::address::from_bytes(rng.generate_bytes(32));
let input_payload = rng.generate_bytes(32);
let approved_message: ApprovedMessage = create_approved_message(
input_source_chain,
input_message_id,
Expand All @@ -246,14 +247,15 @@ fun test_create_approved_message() {

#[test]
fun test_consume_approved_message() {
let mut rng = sui::random::new_generator_for_testing();
let ctx = &mut sui::tx_context::dummy();
let channel: Channel = new(ctx);

let input_source_chain = std::ascii::string(b"Source Chain");
let input_message_id = std::ascii::string(b"message id");
let input_source_address = std::ascii::string(b"Source Address");
let input_destination_id = channel.to_address();
let input_payload = b"payload";
let input_payload = rng.generate_bytes(32);
let approved_message: ApprovedMessage = create_approved_message(
input_source_chain,
input_message_id,
Expand All @@ -280,14 +282,15 @@ fun test_consume_approved_message() {
#[test]
#[expected_failure(abort_code = EInvalidDestination)]
fun test_consume_approved_message_wrong_destination() {
let mut rng = sui::random::new_generator_for_testing();
let ctx = &mut sui::tx_context::dummy();
let channel: Channel = new(ctx);

let source_chain = std::ascii::string(b"Source Chain");
let message_id = std::ascii::string(b"message id");
let source_address = std::ascii::string(b"Source Address");
let destination_id = @0x5678;
let payload = b"payload";
let destination_id = sui::address::from_bytes(rng.generate_bytes(32));
let payload = rng.generate_bytes(32);

let approved_message = create_approved_message(
source_chain,
Expand Down
Loading

0 comments on commit 63075d6

Please sign in to comment.