Skip to content

Commit

Permalink
feat: add smart contract and script documentation
Browse files Browse the repository at this point in the history
  • Loading branch information
dmitriy-woof-software committed Nov 22, 2024
1 parent e1b9003 commit bcec7f6
Show file tree
Hide file tree
Showing 2 changed files with 202 additions and 0 deletions.
185 changes: 185 additions & 0 deletions docs/RewardsV2.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,185 @@
# Documentation for `CometRewardsV2` Contract Core Use Cases

The `CometRewardsV2` contract is designed to efficiently manage the distribution and claiming of rewards in the Compound ecosystem. It uses a Merkle tree structure to handle both уxisting users and new users.
Existing members are users who had already accrued rewards before the campaign began. Their accrued rewards up to a specific block (the start block) are represented in the Merkle tree.
New members are users who began accruing rewards after the campaign started. The same Merkle tree is reused for them, to ensure a unified rewards mechanism.
This design ensures scalability and cost-effectiveness.

---

## Core Use Cases

### 1. **Claiming Rewards for Existing Members**

- **Purpose**: Allow users who were already accrued rewards before the start of the campaign to claim their rewards.

- **Mechanism**:
- A **Merkle tree** is used to represent the accrued reward balances for all users at the time of a specific block (the campaign start block).
- Each user's accrued rewards are included in the tree as their "initial value."
- To claim rewards:
- The user provides a **Merkle proof** to verify their inclusion in the tree.
- The contract uses this proof to validate the user’s accrued rewards without storing all user data on-chain, saving significant gas costs.

- **Why Merkle Trees?**
- **Efficiency**: Storing reward balances for a potentially large user base directly on-chain would be prohibitively expensive.
- **Scalability**: As new campaigns are created regularly, it’s necessary to maintain a lightweight storage model. The Merkle tree allows only the root to be stored on-chain, with individual proofs generated off-chain.
- **Historical Integration**: This mechanism enables seamless distribution of rewards accrued before the campaign started.

---

### 2. **Claiming Rewards for New Members (`ForNewMember`)**

- **Purpose**: Allow users who start accruing rewards after the campaign begins to claim their rewards.

- **Mechanism**:
- The tree includes all existing users and additional boundary addresses (`0x0` and `address(MAX)`).
- Sorting by address allows new users to infer their position relative to neighbors in the tree, even if they weren’t included initially.
- New users can claim by:
1. Using the same Merkle proof mechanism to validate their position.
2. Verifying their potential neighbors and ensuring consistency in their claim.

- **Tree Design**:
- Every entry in the tree includes an **index**, enabling accurate proof verification.
- The tree is sorted to ensure every possible address (even those absent from the initial campaign) can find a valid claim path.

- **Advantages**:
- Eliminates the need for a separate mechanism or tree for new users.
- Maintains a lightweight and scalable rewards distribution system.


---

## Why Use Merkle Trees for Rewards Distribution?

1. **Historical Rewards Distribution**:
- The Compound protocol accrued rewards for users long before `CometRewardsV2` was deployed.
- To retroactively distribute these rewards, the accrued values at a specific block (campaign start) are used as the initial balances for existing members.

2. **Efficient Data Storage**:
- Storing all user balances on-chain is expensive and infeasible given the large user base.
- By storing only the Merkle root, the contract drastically reduces on-chain storage costs.

3. **Regular Campaigns**:
- The contract facilitates frequent campaign creation by only requiring the Merkle root for each campaign.
- Users can generate proofs off-chain to claim their rewards, minimizing computational overhead.

4. **Secure Verification**:
- The Merkle proof system ensures that only valid claims are accepted, with the proof structure preventing fraudulent claims.

5. **Dynamic Membership**:
- New users can easily be integrated into the rewards system using the same Merkle tree.
- Sorting by address and including boundary addresses ensures no gaps or ambiguities in user inclusion.

---

This approach combines the advantages of historical reward integration, cost-efficiency, and scalability, ensuring the protocol can handle rewards distribution effectively for both existing and new members.































# Example of a Correct Merkle Tree with Indices

### 1. **Sorted List of Addresses with Indices**

The tree uses a sorted list of addresses, including boundaries (`0x000...000` and `0xFFF...FFF`), with each entry including its index. Indices are assigned based on the sorted order:

| Index | Address | Accrued Value |
|-------|---------------|---------------|
| 0 | `0x000...000` | 0 (boundary) |
| 1 | `0x000...001` | 100 |
| 2 | `0x000...002` | 200 |
| 3 | `0x000...004` | 300 |
| 4 | `0x000...007` | 400 |
| 5 | `0xFFF...FFF` | 0 (boundary) |

### 2. **Tree Construction**
1. Each entry is represented as a **leaf node** with the structure:
`Hash(address, index, accrued_value)`
2. Pairs of nodes are hashed together iteratively to form parent nodes until a single root is reached.

---

### Corrected Example: Claiming for a New User (Using Neighbor Proofs)

#### **Tree Structure Overview**
The Merkle tree is constructed based on the sorted list of all addresses (including boundaries) and their accrued values at the start of the campaign. Each entry is hashed in the format:
`Hash(address, index, accrued_value)`.

The tree allows **new users** (who are not part of the initial list) to use **existing proofs** of their neighbors to validate their claims.

---

#### **New User Process**

1. **New User Address**: `0x000...005`
- The new user is not directly in the tree but is positioned logically between existing addresses:
- **Left Neighbor**: `0x000...004` (Index 3, Accrued Value 300)
- **Right Neighbor**: `0x000...007` (Index 4, Accrued Value 400)

2. **Validation Using Neighbor Proofs**:
- The new user does not generate their own proof.
- Instead, they use **proofs of their neighbors** (`0x000...004` and `0x000...007`) from the Merkle tree to prove their logical position.

#### **How the New User Claims**

The claim mechanism involves the following:

1. **Identify Neighbors**:
- The new user finds their position relative to the sorted list:
- **Left Neighbor**: `Hash(0x000...004, 3, 300)`
- **Right Neighbor**: `Hash(0x000...007, 4, 400)`

2. **Use Neighbor Proofs**:
- The new user utilizes the Merkle proofs for their **left and right neighbors**.
- These proofs include:
- The full path from the leaf node of each neighbor to the Merkle root.
- The hashes necessary to validate the inclusion of `0x000...004` and `0x000...007`.

3. **Verify Consistency**:
- The contract checks:
- That the new user logically falls between the provided neighbors.
- That their address does not conflict with the tree structure.
- This ensures no double claims or inconsistencies.

---

#### **Steps in the Contract**

1. The new user submits their address (`0x000...005`), neighbors (`0x000...004` and `0x000...007`), and their neighbors' proofs.
2. The contract:
- Validates the neighbors' proofs against the Merkle root.
- Confirms the user logically fits between the neighbors based on sorting.
- Ensures the user’s claim aligns with the expected rules for new users.

#### **No New Proofs**:
The new user does not generate a new proof.
Instead, they rely on existing neighbor proofs for validation.

This approach avoids the need to recompute or extend the Merkle tree. It ensures scalability and reduces gas costs for integrating new users.
17 changes: 17 additions & 0 deletions scripts/rewards_v2/README.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,17 @@
# Rewards V2: Proof Generation Script

The purpose of this script is to generate proofs for Rewards V2. Detailed information about the Rewards V2 implementation can be found in the [RewardsV2.md](../../docs/RewardsV2.md).

## Script Workflow

1. User Collection. Retrieve all users that interacted with the Comet starting from the Comet creation block up to a specified block.

2. Accrued Value Calculation. Simulate the accrue calls for all users using multicall to calculate accrued values as of the specified block.

3. Input Data Preparation. Prepare input data in the required format for constructing a sorted Merkle tree.

4. Merkle Tree Generation. Construct a sorted Merkle tree using the prepared input data.

5. File Generation and Storage. Generate the proof file and save it for later use in the [/campaigns](../../campaigns/) directory.

To validate the list of interacted addresses with Comet please use [Dune Query](https://dune.com/queries/4320237)

0 comments on commit bcec7f6

Please sign in to comment.