pool-sync-mantle

Crates.iopool-sync-mantle
lib.rspool-sync-mantle
version1.1.0
created_at2025-08-25 16:48:28.449788+00
updated_at2025-08-27 13:13:02.945922+00
descriptionA comprehensive library for synchronizing DeFi pools (UniswapV3, Agni, MerchantMoe) on Mantle blockchain with intelligent caching and rate limiting
homepagehttps://github.com/Whisker17/PoolSync_Mantle
repositoryhttps://github.com/Whisker17/PoolSync_Mantle
max_upload_size
id1809812
size346,898
(Whisker17)

documentation

README

PoolSync For Mantle

PoolSync is a comprehensive utility crate for efficiently synchronizing DeFi pools from multiple protocols on the Mantle blockchain. This crate streamlines the process of pool synchronization with support for both V2 and V3 style AMMs, intelligent caching, and rate limiting, eliminating the need for repetitive boilerplate code in DeFi projects targeting the Mantle ecosystem.

Features

  • 🏊‍♂️ Multi-Protocol Support - UniswapV3, Agni (V3-style), and MerchantMoe (V2-style)
  • 🚀 Smart Caching System - Persistent cache for each protocol with automatic management
  • Dual-Node Architecture - Archive + Full node setup for optimal performance and cost efficiency
  • 🔄 Rate Limiting - Built-in rate limiting for public endpoints
  • 📦 Flexible Configuration - Builder pattern with block range support
  • 🔍 Automatic Token Resolution - Fetches token names and metadata automatically
  • 🛠️ Extensible Architecture - Easy to add new protocols with trait-based design
  • 📊 Progress Tracking - Built-in logging and progress indicators

Installation

Add this to your Cargo.toml:

[dependencies]
pool-sync-mantle = "1.0.0"
anyhow = "1.0.82"
tokio = {version = "1.37.0", features = ["rt-multi-thread", "macros"]}
dotenv = "0.15.0"
env_logger = "0.11.4"

Configure your .env with both a full node and an archive node. The archive endpoint must be an archive node, while the full node can be either type. This dual-node design optimizes costs - use a paid archive endpoint for the initial intensive sync, then let the full node handle ongoing synchronization. After initial sync, all data is cached locally, dramatically reducing endpoint strain.

FULL = "full node endpoint"
ARCHIVE = "archive node endpoint"

Supported Protocols

Mantle Network

  • Uniswap V3 - Advanced AMM with concentrated liquidity and tick-based pricing
  • Agni Finance - V3-style AMM (Uniswap V3 fork) with concentrated liquidity
  • Merchant Moe - V2-style constant product AMM with pair-based liquidity

Example Usage

Basic Pool Synchronization

use anyhow::Result;
use pool_sync_mantle::{Chain, PoolSync, PoolType, PoolInfo};
use env_logger;

#[tokio::main]
async fn main() -> Result<()> {
    env_logger::init();
    
    // Example: Sync multiple protocol pools on Mantle
    let start_block = 80000400; // Starting block for Mantle
    let end_block = 80803400;   // Ending block (testing small range)
    
    println!("Syncing DeFi pools on Mantle from block {} to {}...", start_block, end_block);
    
    // Configure and build the PoolSync instance
    let pool_sync = PoolSync::builder()
        .add_pool(PoolType::UniswapV3)     // Add Uniswap V3 pools
        .add_pool(PoolType::Agni)          // Add Agni pools  
        .add_pool(PoolType::MerchantMoe)   // Add MerchantMoe V2 pools
        .chain(Chain::Mantle)              // Target Mantle network
        .rate_limit(1000)                  // Set rate limit (ms between requests)
        .block_range(start_block, end_block) // Set block range
        .build()?;

    // Synchronize pools
    let (pools, last_synced_block) = pool_sync.sync_pools().await?;
    println!(
        "Sync completed! Synced {} pools, last synced block: {}",
        pools.len(),
        last_synced_block
    );

    // Display information about some pools
    println!("\nFirst 5 pools information:");
    for (i, pool) in pools.iter().take(5).enumerate() {
        println!(
            "  {}. Address: {:?}, Token0: {}, Token1: {}", 
            i + 1,
            pool.address(), 
            pool.token0_name(), 
            pool.token1_name()
        );
    }

    Ok(())
}

Quick Start Example

use anyhow::Result;
use pool_sync_mantle::{PoolSync, PoolType, Chain, PoolInfo};

#[tokio::main]
async fn main() -> Result<()> {
    // Quick sync without specifying block range (uses cache)
    let pool_sync = PoolSync::builder()
        .add_pool(PoolType::UniswapV3)
        .chain(Chain::Mantle)
        .rate_limit(500)
        .build()?;

    let (pools, last_synced_block) = pool_sync.sync_pools().await?;
    
    println!("Found {} UniswapV3 pools on Mantle", pools.len());
    println!("Last synced block: {}", last_synced_block);
    
    // Filter high-fee pools
    let high_fee_pools: Vec<_> = pools.iter()
        .filter(|p| p.fee() >= 3000)  // 0.3% fee or higher
        .collect();
    
    println!("High-fee pools: {}", high_fee_pools.len());
    Ok(())
}

Architecture

Caching System

PoolSync uses an intelligent caching system that creates separate cache files for each protocol:

  • cache/Mantle_UniswapV3_cache.json
  • cache/Mantle_Agni_cache.json
  • cache/Mantle_MerchantMoe_cache.json

The cache stores the last synced block number and pool data, enabling efficient incremental updates on subsequent runs.

Pool Structures

The library supports both V2 and V3 style pools through a unified interface:

  • V3 Pools (UniswapV3, Agni): Concentrated liquidity with tick-based pricing
  • V2 Pools (MerchantMoe): Constant product formula with pair-based liquidity

All pools implement the PoolInfo trait providing consistent access to:

  • Pool and token addresses
  • Token names, decimals, and symbols
  • Pool type identification
  • Fee information
  • Stability flags

Modular Fetcher System

Each protocol implements the PoolFetcher trait, providing:

  • Factory address for pool discovery
  • Event signature for pool creation events
  • Pool data parsing and structure creation
  • Chain-specific configuration

Advanced Usage

Performance Optimization

use anyhow::Result;
use pool_sync_mantle::{PoolSync, PoolType, Chain, PoolInfo};

#[tokio::main]
async fn main() -> Result<()> {
    // Optimized configuration for large-scale syncing
    let pool_sync = PoolSync::builder()
        .add_pool(PoolType::UniswapV3)
        .add_pool(PoolType::Agni)
        .chain(Chain::Mantle)
        .rate_limit(250)                    // Faster rate for paid endpoints
        .block_range(80000000, 80100000)    // Sync specific large range
        .build()?;

    println!("Starting optimized sync...");
    let (pools, last_block) = pool_sync.sync_pools().await?;
    
    // Analyze pool distribution by protocol
    let uniswap_pools = pools.iter().filter(|p| matches!(p.pool_type(), PoolType::UniswapV3)).count();
    let agni_pools = pools.iter().filter(|p| matches!(p.pool_type(), PoolType::Agni)).count();
    
    println!("Sync completed up to block {}", last_block);
    println!("UniswapV3 pools: {}, Agni pools: {}", uniswap_pools, agni_pools);
    
    // Find pools with specific token pairs
    let usdc_pools: Vec<_> = pools.iter()
        .filter(|p| p.token0_name().contains("USDC") || p.token1_name().contains("USDC"))
        .collect();
        
    println!("Found {} pools with USDC", usdc_pools.len());
    Ok(())
}

Working with Different Pool Types

use pool_sync_mantle::{Pool, PoolType};

fn analyze_pool(pool: &Pool) {
    match pool {
        Pool::UniswapV3(v3_pool) => {
            println!("V3 Pool: {} (tick spacing: {})", 
                     v3_pool.address, v3_pool.tick_spacing);
        },
        Pool::Agni(agni_pool) => {
            println!("Agni Pool: {} (fee: {})", 
                     agni_pool.address, agni_pool.fee);
        },
        Pool::MerchantMoe(v2_pool) => {
            println!("V2 Pool: {} (stable: {})", 
                     v2_pool.address, v2_pool.stable);
        }
    }
}

Adding New Protocols

This codebase focuses specifically on Mantle network. To add a new DEX protocol:

1. Add Pool ABI

Add the factory ABI to src/pools/abis/YourProtocol.json

2. Create Pool Fetcher

Create src/pools/pool_fetchers/yourprotocol/yourprotocol_fetcher.rs:

use crate::pools::PoolFetcher;
use alloy::primitives::Address;

pub struct YourProtocolFetcher;

impl PoolFetcher for YourProtocolFetcher {
    fn pool_type(&self) -> PoolType {
        PoolType::YourProtocol
    }
    
    fn factory_address(&self, chain: Chain) -> Address {
        match chain {
            Chain::Mantle => "your_factory_address".parse().unwrap(),
        }
    }
    
    // Implement other required methods...
}

3. Update Pool Types

Add your protocol to src/pools/mod.rs:

#[derive(Debug, Clone, Copy, PartialEq, Eq, Hash, Serialize, Deserialize)]
pub enum PoolType {
    UniswapV3,
    MerchantMoe,
    Agni,
    YourProtocol,  // Add here
}

4. Register in Builder

Update src/builder.rs to include your fetcher

5. Add to Chain Support

Update src/chain.rs to map your protocol to Mantle

Refer to existing implementations (Agni, MerchantMoe) as examples.

Troubleshooting

Common Issues

Cache Corruption: Delete the cache files in cache/ directory to force a full resync

Rate Limiting: Increase the rate limit value or upgrade to a paid RPC endpoint

Memory Usage: For large block ranges, consider syncing in smaller chunks

Archive Node Requirement: The initial sync requires an archive node for historical data access

Roadmap

  • Enhanced Protocol Support - Add more Mantle-native DEXs
  • Database Integration - Optional PostgreSQL/SQLite backend
  • Real-time Updates - WebSocket subscriptions for live pool updates
  • Cross-chain Support - Extend beyond Mantle to other networks
  • Pool Analytics - Built-in TVL, volume, and liquidity analysis
  • Batch Operations - Bulk token resolution and metadata fetching

Contributing

Contributions are welcome! Please feel free to submit a Pull Request. For major changes, please open an issue first to discuss what you would like to change.

Acknowledgments

Took inspiration from amm-rs - an excellent AMM library worth checking out!

Commit count: 11

cargo fmt