mrls 0.1.8

Unofficial Rust client for the Moralis Web3 API - tokens, NFTs, wallets, DeFi, and market data
Documentation
//! `DeFi` API client

use super::types::{DefiPositionsResponse, DefiSummary, PairAddress, PairPrice, PairReserves};
use crate::client::Client;
use crate::error::Result;
use serde::Serialize;

/// Query parameters for `DeFi` endpoints
#[derive(Debug, Default, Serialize)]
pub struct DefiQuery {
    #[serde(skip_serializing_if = "Option::is_none")]
    pub chain: Option<String>,
    #[serde(skip_serializing_if = "Option::is_none")]
    pub cursor: Option<String>,
    #[serde(skip_serializing_if = "Option::is_none")]
    pub limit: Option<i32>,
    #[serde(skip_serializing_if = "Option::is_none")]
    pub exchange: Option<String>,
    #[serde(skip_serializing_if = "Option::is_none")]
    pub to_block: Option<String>,
}

impl DefiQuery {
    #[must_use]
    pub fn new() -> Self {
        Self::default()
    }

    #[must_use]
    pub fn chain(mut self, chain: impl Into<String>) -> Self {
        self.chain = Some(chain.into());
        self
    }

    #[must_use]
    pub fn cursor(mut self, cursor: impl Into<String>) -> Self {
        self.cursor = Some(cursor.into());
        self
    }

    #[must_use]
    pub fn limit(mut self, limit: i32) -> Self {
        self.limit = Some(limit);
        self
    }

    #[must_use]
    pub fn exchange(mut self, exchange: impl Into<String>) -> Self {
        self.exchange = Some(exchange.into());
        self
    }
}

/// API for `DeFi` operations
pub struct DefiApi<'a> {
    client: &'a Client,
}

impl<'a> DefiApi<'a> {
    #[must_use]
    pub fn new(client: &'a Client) -> Self {
        Self { client }
    }

    /// Get price between two tokens in a pair
    pub async fn get_pair_price(
        &self,
        token0: &str,
        token1: &str,
        query: Option<&DefiQuery>,
    ) -> Result<PairPrice> {
        let path = format!("/{token0}/{token1}/price");
        if let Some(q) = query {
            self.client.get_with_query(&path, q).await
        } else {
            self.client.get(&path).await
        }
    }

    /// Get reserves for a pair
    pub async fn get_pair_reserves(
        &self,
        pair_address: &str,
        chain: Option<&str>,
    ) -> Result<PairReserves> {
        let path = format!("/{pair_address}/reserves");
        if let Some(chain) = chain {
            let query = DefiQuery::new().chain(chain);
            self.client.get_with_query(&path, &query).await
        } else {
            self.client.get(&path).await
        }
    }

    /// Get pair address for two tokens
    pub async fn get_pair_address(
        &self,
        token0: &str,
        token1: &str,
        query: Option<&DefiQuery>,
    ) -> Result<PairAddress> {
        let path = format!("/{token0}/{token1}/pairAddress");
        if let Some(q) = query {
            self.client.get_with_query(&path, q).await
        } else {
            self.client.get(&path).await
        }
    }

    /// Get `DeFi` summary for a wallet
    pub async fn get_wallet_defi_summary(&self, address: &str) -> Result<DefiSummary> {
        let path = format!("/wallets/{address}/defi/summary");
        self.client.get(&path).await
    }

    /// Get all `DeFi` positions for a wallet
    pub async fn get_wallet_defi_positions(
        &self,
        address: &str,
        query: Option<&DefiQuery>,
    ) -> Result<DefiPositionsResponse> {
        let path = format!("/wallets/{address}/defi/positions");
        if let Some(q) = query {
            self.client.get_with_query(&path, q).await
        } else {
            self.client.get(&path).await
        }
    }

    /// Get `DeFi` positions for a specific protocol
    pub async fn get_wallet_protocol_positions(
        &self,
        address: &str,
        protocol: &str,
        query: Option<&DefiQuery>,
    ) -> Result<DefiPositionsResponse> {
        let path = format!("/wallets/{address}/defi/{protocol}/positions");
        if let Some(q) = query {
            self.client.get_with_query(&path, q).await
        } else {
            self.client.get(&path).await
        }
    }
}