use crate::client::{
schema::{
schema,
Address,
AssetId,
ConversionError,
Nonce,
PageInfo,
UtxoId,
U32,
U64,
},
PageDirection,
PaginatedResult,
PaginationRequest,
};
use itertools::Itertools;
use std::str::FromStr;
#[derive(cynic::QueryVariables, Debug)]
pub struct CoinByIdArgs {
pub utxo_id: UtxoId,
}
#[derive(cynic::QueryFragment, Debug)]
#[cynic(
schema_path = "./assets/schema.sdl",
graphql_type = "Query",
variables = "CoinByIdArgs"
)]
pub struct CoinByIdQuery {
#[arguments(utxoId: $utxo_id)]
pub coin: Option<Coin>,
}
#[derive(cynic::InputObject, Clone, Debug)]
#[cynic(schema_path = "./assets/schema.sdl")]
pub struct CoinFilterInput {
pub owner: Address,
pub asset_id: Option<AssetId>,
}
#[derive(cynic::QueryVariables, Debug)]
pub struct CoinsConnectionArgs {
filter: CoinFilterInput,
pub after: Option<String>,
pub before: Option<String>,
pub first: Option<i32>,
pub last: Option<i32>,
}
impl From<(Address, AssetId, PaginationRequest<String>)> for CoinsConnectionArgs {
fn from(r: (Address, AssetId, PaginationRequest<String>)) -> Self {
match r.2.direction {
PageDirection::Forward => CoinsConnectionArgs {
filter: CoinFilterInput {
owner: r.0,
asset_id: Some(r.1),
},
after: r.2.cursor,
before: None,
first: Some(r.2.results as i32),
last: None,
},
PageDirection::Backward => CoinsConnectionArgs {
filter: CoinFilterInput {
owner: r.0,
asset_id: Some(r.1),
},
after: None,
before: r.2.cursor,
first: None,
last: Some(r.2.results as i32),
},
}
}
}
#[derive(cynic::QueryFragment, Debug)]
#[cynic(
schema_path = "./assets/schema.sdl",
graphql_type = "Query",
variables = "CoinsConnectionArgs"
)]
pub struct CoinsQuery {
#[arguments(filter: $filter, after: $after, before: $before, first: $first, last: $last)]
pub coins: CoinConnection,
}
#[derive(cynic::QueryFragment, Debug)]
#[cynic(schema_path = "./assets/schema.sdl")]
pub struct CoinConnection {
pub edges: Vec<CoinEdge>,
pub page_info: PageInfo,
}
impl From<CoinConnection> for PaginatedResult<Coin, String> {
fn from(conn: CoinConnection) -> Self {
PaginatedResult {
cursor: conn.page_info.end_cursor,
has_next_page: conn.page_info.has_next_page,
has_previous_page: conn.page_info.has_previous_page,
results: conn.edges.into_iter().map(|e| e.node).collect(),
}
}
}
#[derive(cynic::QueryFragment, Debug)]
#[cynic(schema_path = "./assets/schema.sdl")]
pub struct CoinEdge {
pub cursor: String,
pub node: Coin,
}
#[derive(cynic::QueryFragment, Debug)]
#[cynic(schema_path = "./assets/schema.sdl")]
pub struct Coin {
pub amount: U64,
pub block_created: U32,
pub asset_id: AssetId,
pub utxo_id: UtxoId,
pub maturity: U32,
pub owner: Address,
}
#[derive(cynic::QueryFragment, Debug)]
#[cynic(schema_path = "./assets/schema.sdl", graphql_type = "Coin")]
pub struct CoinIdFragment {
pub utxo_id: UtxoId,
}
#[derive(cynic::InputObject, Clone, Debug)]
#[cynic(schema_path = "./assets/schema.sdl")]
pub struct ExcludeInput {
utxos: Vec<UtxoId>,
messages: Vec<Nonce>,
}
impl ExcludeInput {
pub fn from_tuple(tuple: (Vec<&str>, Vec<&str>)) -> Result<Self, ConversionError> {
let utxos = tuple.0.into_iter().map(UtxoId::from_str).try_collect()?;
let messages = tuple.1.into_iter().map(Nonce::from_str).try_collect()?;
Ok(Self { utxos, messages })
}
}
#[derive(cynic::InputObject, Clone, Debug)]
#[cynic(schema_path = "./assets/schema.sdl")]
pub struct SpendQueryElementInput {
pub asset_id: AssetId,
pub amount: U64,
pub max: Option<U64>,
}
#[derive(cynic::QueryFragment, Debug)]
#[cynic(schema_path = "./assets/schema.sdl")]
pub struct MessageCoin {
pub amount: U64,
pub sender: Address,
pub recipient: Address,
pub nonce: Nonce,
pub da_height: U64,
}
#[derive(cynic::InlineFragments, Debug)]
#[cynic(schema_path = "./assets/schema.sdl")]
pub enum CoinType {
Coin(Coin),
MessageCoin(MessageCoin),
#[cynic(fallback)]
Unknown,
}
impl CoinType {
pub fn amount(&self) -> u64 {
match self {
CoinType::Coin(c) => c.amount.0,
CoinType::MessageCoin(m) => m.amount.0,
CoinType::Unknown => 0,
}
}
}
#[derive(cynic::QueryVariables, Debug)]
pub struct CoinsToSpendArgs {
owner: Address,
query_per_asset: Vec<SpendQueryElementInput>,
excluded_ids: Option<ExcludeInput>,
}
pub(crate) type CoinsToSpendArgsTuple =
(Address, Vec<SpendQueryElementInput>, Option<ExcludeInput>);
impl From<CoinsToSpendArgsTuple> for CoinsToSpendArgs {
fn from(r: CoinsToSpendArgsTuple) -> Self {
CoinsToSpendArgs {
owner: r.0,
query_per_asset: r.1,
excluded_ids: r.2,
}
}
}
#[derive(cynic::QueryFragment, Debug)]
#[cynic(
schema_path = "./assets/schema.sdl",
graphql_type = "Query",
variables = "CoinsToSpendArgs"
)]
pub struct CoinsToSpendQuery {
#[arguments(owner: $owner, queryPerAsset: $query_per_asset, excludedIds: $excluded_ids)]
pub coins_to_spend: Vec<Vec<CoinType>>,
}
#[cfg(test)]
mod tests {
use super::*;
#[test]
fn coin_by_id_query_gql_output() {
use cynic::QueryBuilder;
let operation = CoinByIdQuery::build(CoinByIdArgs {
utxo_id: UtxoId::default(),
});
insta::assert_snapshot!(operation.query)
}
#[test]
fn coins_connection_query_gql_output() {
use cynic::QueryBuilder;
let operation = CoinsQuery::build(CoinsConnectionArgs {
filter: CoinFilterInput {
owner: Address::default(),
asset_id: Some(AssetId::default()),
},
after: None,
before: None,
first: None,
last: None,
});
insta::assert_snapshot!(operation.query)
}
}