sui_indexer/apis/
coin_api.rsuse crate::indexer_reader::IndexerReader;
use async_trait::async_trait;
use jsonrpsee::core::RpcResult;
use jsonrpsee::RpcModule;
use sui_json_rpc::coin_api::{parse_to_struct_tag, parse_to_type_tag};
use sui_json_rpc::error::SuiRpcInputError;
use sui_json_rpc::SuiRpcModule;
use sui_json_rpc_api::{cap_page_limit, CoinReadApiServer};
use sui_json_rpc_types::{Balance, CoinPage, Page, SuiCoinMetadata};
use sui_open_rpc::Module;
use sui_types::balance::Supply;
use sui_types::base_types::{ObjectID, SuiAddress};
use sui_types::gas_coin::{GAS, TOTAL_SUPPLY_MIST};
pub(crate) struct CoinReadApi {
inner: IndexerReader,
}
impl CoinReadApi {
pub fn new(inner: IndexerReader) -> Self {
Self { inner }
}
}
#[async_trait]
impl CoinReadApiServer for CoinReadApi {
async fn get_coins(
&self,
owner: SuiAddress,
coin_type: Option<String>,
cursor: Option<String>,
limit: Option<usize>,
) -> RpcResult<CoinPage> {
let limit = cap_page_limit(limit);
if limit == 0 {
return Ok(CoinPage::empty());
}
let coin_type =
parse_to_type_tag(coin_type)?.to_canonical_string(true);
let cursor = match cursor {
Some(c) => c
.parse()
.map_err(|e| SuiRpcInputError::GenericInvalid(format!("invalid cursor: {e}")))?,
None => ObjectID::ZERO,
};
let mut results = self
.inner
.get_owned_coins(owner, Some(coin_type), cursor, limit + 1)
.await?;
let has_next_page = results.len() > limit;
results.truncate(limit);
let next_cursor = results.last().map(|o| o.coin_object_id.to_string());
Ok(Page {
data: results,
next_cursor,
has_next_page,
})
}
async fn get_all_coins(
&self,
owner: SuiAddress,
cursor: Option<String>,
limit: Option<usize>,
) -> RpcResult<CoinPage> {
let limit = cap_page_limit(limit);
if limit == 0 {
return Ok(CoinPage::empty());
}
let cursor = match cursor {
Some(c) => c
.parse()
.map_err(|e| SuiRpcInputError::GenericInvalid(format!("invalid cursor: {e}")))?,
None => ObjectID::ZERO,
};
let mut results = self
.inner
.get_owned_coins(owner, None, cursor, limit + 1)
.await?;
let has_next_page = results.len() > limit;
results.truncate(limit);
let next_cursor = results.last().map(|o| o.coin_object_id.to_string());
Ok(Page {
data: results,
next_cursor,
has_next_page,
})
}
async fn get_balance(
&self,
owner: SuiAddress,
coin_type: Option<String>,
) -> RpcResult<Balance> {
let coin_type =
parse_to_type_tag(coin_type)?.to_canonical_string(true);
let mut results = self
.inner
.get_coin_balances(owner, Some(coin_type.clone()))
.await?;
if results.is_empty() {
return Ok(Balance::zero(coin_type));
}
Ok(results.swap_remove(0))
}
async fn get_all_balances(&self, owner: SuiAddress) -> RpcResult<Vec<Balance>> {
self.inner
.get_coin_balances(owner, None)
.await
.map_err(Into::into)
}
async fn get_coin_metadata(&self, coin_type: String) -> RpcResult<Option<SuiCoinMetadata>> {
let coin_struct = parse_to_struct_tag(&coin_type)?;
self.inner
.get_coin_metadata(coin_struct)
.await
.map_err(Into::into)
}
async fn get_total_supply(&self, coin_type: String) -> RpcResult<Supply> {
let coin_struct = parse_to_struct_tag(&coin_type)?;
if GAS::is_gas(&coin_struct) {
Ok(Supply {
value: TOTAL_SUPPLY_MIST,
})
} else {
self.inner
.get_total_supply(coin_struct)
.await
.map_err(Into::into)
}
}
}
impl SuiRpcModule for CoinReadApi {
fn rpc(self) -> RpcModule<Self> {
self.into_rpc()
}
fn rpc_doc_module() -> Module {
sui_json_rpc_api::CoinReadApiOpenRpc::module_doc()
}
}