Skip to content

Commit

Permalink
feat(torii/graphql): use Connection abstraction to return data for …
Browse files Browse the repository at this point in the history
…ercBalance and ercTransfer query

commit-id:bc10539f
  • Loading branch information
lambda-0x committed Oct 13, 2024
1 parent 7035d54 commit 1e9a382
Show file tree
Hide file tree
Showing 4 changed files with 374 additions and 66 deletions.
2 changes: 2 additions & 0 deletions crates/torii/graphql/src/constants.rs
Original file line number Diff line number Diff line change
Expand Up @@ -9,6 +9,8 @@ pub const EVENT_MESSAGE_TABLE: &str = "event_messages";
pub const MODEL_TABLE: &str = "models";
pub const TRANSACTION_TABLE: &str = "transactions";
pub const METADATA_TABLE: &str = "metadata";
pub const ERC_BALANCE_TABLE: &str = "balances";
pub const ERC_TRANSFER_TABLE: &str = "erc_transfers";

pub const ID_COLUMN: &str = "id";
pub const EVENT_ID_COLUMN: &str = "event_id";
Expand Down
204 changes: 178 additions & 26 deletions crates/torii/graphql/src/object/erc/erc_balance.rs
Original file line number Diff line number Diff line change
@@ -1,15 +1,27 @@
use async_graphql::connection::PageInfo;
use async_graphql::dynamic::{Field, FieldFuture, InputValue, TypeRef};
use async_graphql::{Name, Value};
use convert_case::{Case, Casing};
use serde::Deserialize;
use sqlx::{FromRow, Pool, Sqlite, SqliteConnection};
use sqlx::sqlite::SqliteRow;
use sqlx::{FromRow, Pool, Row, Sqlite, SqliteConnection};
use starknet_crypto::Felt;
use torii_core::sql::utils::felt_to_sql_string;
use tracing::warn;

use crate::constants::{ERC_BALANCE_NAME, ERC_BALANCE_TYPE_NAME};
use super::handle_cursor;
use crate::constants::{
DEFAULT_LIMIT, ERC_BALANCE_NAME, ERC_BALANCE_TABLE, ERC_BALANCE_TYPE_NAME, ID_COLUMN,
};
use crate::mapping::ERC_BALANCE_TYPE_MAPPING;
use crate::object::connection::page_info::PageInfoObject;
use crate::object::connection::{
connection_arguments, cursor, parse_connection_arguments, ConnectionArguments,
};
use crate::object::{BasicObject, ResolvableObject};
use crate::query::data::count_rows;
use crate::query::filter::{Comparator, Filter, FilterValue};
use crate::query::order::{CursorDirection, Direction};
use crate::types::{TypeMapping, ValueMapping};
use crate::utils::extract;

Expand Down Expand Up @@ -38,41 +50,173 @@ impl ResolvableObject for ErcBalanceObject {
TypeRef::named_nn(TypeRef::STRING),
);

let field = Field::new(self.name().0, TypeRef::named_list(self.type_name()), move |ctx| {
FieldFuture::new(async move {
let mut conn = ctx.data::<Pool<Sqlite>>()?.acquire().await?;
let address = extract::<Felt>(
ctx.args.as_index_map(),
&account_address.to_case(Case::Camel),
)?;
let mut field = Field::new(
self.name().0,
TypeRef::named(format!("{}Connection", self.type_name())),
move |ctx| {
FieldFuture::new(async move {
let mut conn = ctx.data::<Pool<Sqlite>>()?.acquire().await?;
let connection = parse_connection_arguments(&ctx)?;
let address = extract::<Felt>(
ctx.args.as_index_map(),
&account_address.to_case(Case::Camel),
)?;

Check warning on line 63 in crates/torii/graphql/src/object/erc/erc_balance.rs

View check run for this annotation

Codecov / codecov/patch

crates/torii/graphql/src/object/erc/erc_balance.rs#L57-L63

Added lines #L57 - L63 were not covered by tests

let erc_balances = fetch_erc_balances(&mut conn, address).await?;
let filter = vec![Filter {
field: "account_address".to_string(),
comparator: Comparator::Eq,
value: FilterValue::String(felt_to_sql_string(&address)),
}];

Check warning on line 69 in crates/torii/graphql/src/object/erc/erc_balance.rs

View check run for this annotation

Codecov / codecov/patch

crates/torii/graphql/src/object/erc/erc_balance.rs#L65-L69

Added lines #L65 - L69 were not covered by tests

Ok(Some(Value::List(erc_balances)))
})
})
let total_count =
count_rows(&mut conn, ERC_BALANCE_TABLE, &None, &Some(filter)).await?;

Check warning on line 72 in crates/torii/graphql/src/object/erc/erc_balance.rs

View check run for this annotation

Codecov / codecov/patch

crates/torii/graphql/src/object/erc/erc_balance.rs#L71-L72

Added lines #L71 - L72 were not covered by tests

let (data, page_info) =
fetch_erc_balances2(&mut conn, address, &connection, total_count).await?;

Check warning on line 75 in crates/torii/graphql/src/object/erc/erc_balance.rs

View check run for this annotation

Codecov / codecov/patch

crates/torii/graphql/src/object/erc/erc_balance.rs#L74-L75

Added lines #L74 - L75 were not covered by tests

let results = erc_balance_connection_output(&data, total_count, page_info)?;

Check warning on line 77 in crates/torii/graphql/src/object/erc/erc_balance.rs

View check run for this annotation

Codecov / codecov/patch

crates/torii/graphql/src/object/erc/erc_balance.rs#L77

Added line #L77 was not covered by tests
// let erc_balances = fetch_erc_balances(&mut conn, address).await?;

Ok(Some(Value::Object(results)))
})

Check warning on line 81 in crates/torii/graphql/src/object/erc/erc_balance.rs

View check run for this annotation

Codecov / codecov/patch

crates/torii/graphql/src/object/erc/erc_balance.rs#L80-L81

Added lines #L80 - L81 were not covered by tests
},
)
.argument(argument);

field = connection_arguments(field);
vec![field]
}
}

async fn fetch_erc_balances(
async fn fetch_erc_balances2(

Check warning on line 91 in crates/torii/graphql/src/object/erc/erc_balance.rs

View check run for this annotation

Codecov / codecov/patch

crates/torii/graphql/src/object/erc/erc_balance.rs#L91

Added line #L91 was not covered by tests
conn: &mut SqliteConnection,
address: Felt,
) -> sqlx::Result<Vec<Value>> {
let query = "SELECT t.contract_address, t.name, t.symbol, t.decimals, b.balance, b.token_id, \
c.contract_type
FROM balances b
connection: &ConnectionArguments,
total_count: i64,
) -> sqlx::Result<(Vec<SqliteRow>, PageInfo)> {
let table_name = ERC_BALANCE_TABLE;
let id_column = format!("b.{}", ID_COLUMN);

let mut query = format!(
"SELECT b.id, t.contract_address, t.name, t.symbol, t.decimals, b.balance, b.token_id, \
c.contract_type
FROM {table_name} b

Check warning on line 103 in crates/torii/graphql/src/object/erc/erc_balance.rs

View check run for this annotation

Codecov / codecov/patch

crates/torii/graphql/src/object/erc/erc_balance.rs#L94-L103

Added lines #L94 - L103 were not covered by tests
JOIN tokens t ON b.token_id = t.id
JOIN contracts c ON t.contract_address = c.contract_address
WHERE b.account_address = ?";
JOIN contracts c ON t.contract_address = c.contract_address"
);
let mut conditions = vec!["b.account_address = ?".to_string()];

let mut cursor_param = &connection.after;
if let Some(after_cursor) = &connection.after {
conditions.push(handle_cursor(after_cursor, CursorDirection::After, ID_COLUMN)?);
}

Check warning on line 112 in crates/torii/graphql/src/object/erc/erc_balance.rs

View check run for this annotation

Codecov / codecov/patch

crates/torii/graphql/src/object/erc/erc_balance.rs#L105-L112

Added lines #L105 - L112 were not covered by tests

if let Some(before_cursor) = &connection.before {
cursor_param = &connection.before;
conditions.push(handle_cursor(before_cursor, CursorDirection::Before, ID_COLUMN)?);
}

Check warning on line 117 in crates/torii/graphql/src/object/erc/erc_balance.rs

View check run for this annotation

Codecov / codecov/patch

crates/torii/graphql/src/object/erc/erc_balance.rs#L114-L117

Added lines #L114 - L117 were not covered by tests

if !conditions.is_empty() {
query.push_str(&format!(" WHERE {}", conditions.join(" AND ")));
}

Check warning on line 121 in crates/torii/graphql/src/object/erc/erc_balance.rs

View check run for this annotation

Codecov / codecov/patch

crates/torii/graphql/src/object/erc/erc_balance.rs#L119-L121

Added lines #L119 - L121 were not covered by tests

let is_cursor_based = connection.first.or(connection.last).is_some() || cursor_param.is_some();

Check warning on line 123 in crates/torii/graphql/src/object/erc/erc_balance.rs

View check run for this annotation

Codecov / codecov/patch

crates/torii/graphql/src/object/erc/erc_balance.rs#L123

Added line #L123 was not covered by tests

let data_limit =
connection.first.or(connection.last).or(connection.limit).unwrap_or(DEFAULT_LIMIT);
let limit = if is_cursor_based {
match &cursor_param {
Some(_) => data_limit + 2,
None => data_limit + 1, // prev page does not exist

Check warning on line 130 in crates/torii/graphql/src/object/erc/erc_balance.rs

View check run for this annotation

Codecov / codecov/patch

crates/torii/graphql/src/object/erc/erc_balance.rs#L125-L130

Added lines #L125 - L130 were not covered by tests
}
} else {
data_limit

Check warning on line 133 in crates/torii/graphql/src/object/erc/erc_balance.rs

View check run for this annotation

Codecov / codecov/patch

crates/torii/graphql/src/object/erc/erc_balance.rs#L133

Added line #L133 was not covered by tests
};

let rows = sqlx::query(query).bind(felt_to_sql_string(&address)).fetch_all(conn).await?;
let order_direction = match (connection.first, connection.last) {
(Some(_), _) => Direction::Desc,
(_, Some(_)) => Direction::Asc,
_ => Direction::Desc,

Check warning on line 139 in crates/torii/graphql/src/object/erc/erc_balance.rs

View check run for this annotation

Codecov / codecov/patch

crates/torii/graphql/src/object/erc/erc_balance.rs#L136-L139

Added lines #L136 - L139 were not covered by tests
};

let mut erc_balances = Vec::new();
query.push_str(&format!(" ORDER BY {id_column} {} LIMIT {limit}", order_direction.as_ref()));

Check warning on line 142 in crates/torii/graphql/src/object/erc/erc_balance.rs

View check run for this annotation

Codecov / codecov/patch

crates/torii/graphql/src/object/erc/erc_balance.rs#L142

Added line #L142 was not covered by tests

if let Some(offset) = connection.offset {
query.push_str(&format!(" OFFSET {}", offset));
}

Check warning on line 146 in crates/torii/graphql/src/object/erc/erc_balance.rs

View check run for this annotation

Codecov / codecov/patch

crates/torii/graphql/src/object/erc/erc_balance.rs#L144-L146

Added lines #L144 - L146 were not covered by tests

let mut data = sqlx::query(&query).bind(felt_to_sql_string(&address)).fetch_all(conn).await?;
let mut page_info = PageInfo {
has_previous_page: false,
has_next_page: false,
start_cursor: None,
end_cursor: None,
};

if data.is_empty() {
Ok((data, page_info))
} else if is_cursor_based {
match cursor_param {
Some(cursor_query) => {
let first_cursor = cursor::encode(
&data[0].try_get::<String, &str>(&id_column)?,
&data[0].try_get_unchecked::<String, &str>(&id_column)?,

Check warning on line 163 in crates/torii/graphql/src/object/erc/erc_balance.rs

View check run for this annotation

Codecov / codecov/patch

crates/torii/graphql/src/object/erc/erc_balance.rs#L148-L163

Added lines #L148 - L163 were not covered by tests
);

if &first_cursor == cursor_query && data.len() != 1 {
data.remove(0);
page_info.has_previous_page = true;
} else {
data.pop();
}

Check warning on line 171 in crates/torii/graphql/src/object/erc/erc_balance.rs

View check run for this annotation

Codecov / codecov/patch

crates/torii/graphql/src/object/erc/erc_balance.rs#L166-L171

Added lines #L166 - L171 were not covered by tests

if data.len() as u64 == limit - 1 {
page_info.has_next_page = true;
data.pop();
}

Check warning on line 176 in crates/torii/graphql/src/object/erc/erc_balance.rs

View check run for this annotation

Codecov / codecov/patch

crates/torii/graphql/src/object/erc/erc_balance.rs#L173-L176

Added lines #L173 - L176 were not covered by tests
}
None => {
if data.len() as u64 == limit {
page_info.has_next_page = true;
data.pop();
}

Check warning on line 182 in crates/torii/graphql/src/object/erc/erc_balance.rs

View check run for this annotation

Codecov / codecov/patch

crates/torii/graphql/src/object/erc/erc_balance.rs#L179-L182

Added lines #L179 - L182 were not covered by tests
}
}

if !data.is_empty() {
page_info.start_cursor = Some(cursor::encode(
&data[0].try_get::<String, &str>(ID_COLUMN)?,
&data[0].try_get_unchecked::<String, &str>(ID_COLUMN)?,

Check warning on line 189 in crates/torii/graphql/src/object/erc/erc_balance.rs

View check run for this annotation

Codecov / codecov/patch

crates/torii/graphql/src/object/erc/erc_balance.rs#L186-L189

Added lines #L186 - L189 were not covered by tests
));
page_info.end_cursor = Some(cursor::encode(
&data[data.len() - 1].try_get::<String, &str>(ID_COLUMN)?,
&data[data.len() - 1].try_get_unchecked::<String, &str>(ID_COLUMN)?,

Check warning on line 193 in crates/torii/graphql/src/object/erc/erc_balance.rs

View check run for this annotation

Codecov / codecov/patch

crates/torii/graphql/src/object/erc/erc_balance.rs#L191-L193

Added lines #L191 - L193 were not covered by tests
));
}

Check warning on line 195 in crates/torii/graphql/src/object/erc/erc_balance.rs

View check run for this annotation

Codecov / codecov/patch

crates/torii/graphql/src/object/erc/erc_balance.rs#L195

Added line #L195 was not covered by tests

Ok((data, page_info))

Check warning on line 197 in crates/torii/graphql/src/object/erc/erc_balance.rs

View check run for this annotation

Codecov / codecov/patch

crates/torii/graphql/src/object/erc/erc_balance.rs#L197

Added line #L197 was not covered by tests
} else {
let offset = connection.offset.unwrap_or(0);
if 1 < offset && offset < total_count as u64 {
page_info.has_previous_page = true;
}
if limit + offset < total_count as u64 {
page_info.has_next_page = true;
}

Check warning on line 205 in crates/torii/graphql/src/object/erc/erc_balance.rs

View check run for this annotation

Codecov / codecov/patch

crates/torii/graphql/src/object/erc/erc_balance.rs#L199-L205

Added lines #L199 - L205 were not covered by tests

Ok((data, page_info))

Check warning on line 207 in crates/torii/graphql/src/object/erc/erc_balance.rs

View check run for this annotation

Codecov / codecov/patch

crates/torii/graphql/src/object/erc/erc_balance.rs#L207

Added line #L207 was not covered by tests
}
}

Check warning on line 209 in crates/torii/graphql/src/object/erc/erc_balance.rs

View check run for this annotation

Codecov / codecov/patch

crates/torii/graphql/src/object/erc/erc_balance.rs#L209

Added line #L209 was not covered by tests

for row in rows {
let row = BalanceQueryResultRaw::from_row(&row)?;
fn erc_balance_connection_output(
data: &[SqliteRow],
total_count: i64,
page_info: PageInfo,
) -> sqlx::Result<ValueMapping> {
let mut edges = Vec::new();
for row in data {
let row = BalanceQueryResultRaw::from_row(row)?;
let cursor = cursor::encode(&row.id, &row.id);

Check warning on line 219 in crates/torii/graphql/src/object/erc/erc_balance.rs

View check run for this annotation

Codecov / codecov/patch

crates/torii/graphql/src/object/erc/erc_balance.rs#L211-L219

Added lines #L211 - L219 were not covered by tests

let balance_value = match row.contract_type.to_lowercase().as_str() {
"erc20" => {
Expand Down Expand Up @@ -116,10 +260,17 @@ async fn fetch_erc_balances(
}
};

erc_balances.push(balance_value);
edges.push(Value::Object(ValueMapping::from([
(Name::new("node"), balance_value),
(Name::new("cursor"), Value::String(cursor)),
])));

Check warning on line 266 in crates/torii/graphql/src/object/erc/erc_balance.rs

View check run for this annotation

Codecov / codecov/patch

crates/torii/graphql/src/object/erc/erc_balance.rs#L263-L266

Added lines #L263 - L266 were not covered by tests
}

Ok(erc_balances)
Ok(ValueMapping::from([
(Name::new("totalCount"), Value::from(total_count)),
(Name::new("edges"), Value::List(edges)),
(Name::new("pageInfo"), PageInfoObject::value(page_info)),
]))

Check warning on line 273 in crates/torii/graphql/src/object/erc/erc_balance.rs

View check run for this annotation

Codecov / codecov/patch

crates/torii/graphql/src/object/erc/erc_balance.rs#L269-L273

Added lines #L269 - L273 were not covered by tests
}

// TODO: This would be required when subscriptions are needed
Expand All @@ -133,6 +284,7 @@ async fn fetch_erc_balances(
#[derive(FromRow, Deserialize, Debug, Clone)]
#[serde(rename_all = "camelCase")]
struct BalanceQueryResultRaw {
pub id: String,
pub contract_address: String,
pub name: String,
pub symbol: String,
Expand Down
Loading

0 comments on commit 1e9a382

Please sign in to comment.