-
Notifications
You must be signed in to change notification settings - Fork 189
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
fix(torii): handle failed metadata fetch and empty metadata in db #2755
Conversation
40b6ead
to
0d7338e
Compare
WalkthroughOhayo, sensei! This pull request introduces several enhancements to the handling of ERC721 tokens within the Changes
Possibly related PRs
Suggested reviewers
🪧 TipsChatThere are 3 ways to chat with CodeRabbit:
Note: Be mindful of the bot's finite context window. It's strongly recommended to break down tasks such as reading entire modules into smaller chunks. For a focused discussion, use review comments to chat about specific files and their changes, instead of using the PR comments. CodeRabbit Commands (Invoked using PR comments)
Other keywords and placeholders
Documentation and Community
|
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Actionable comments posted: 0
🧹 Outside diff range and nitpick comments (1)
crates/torii/core/src/executor/erc.rs (1)
245-256
: Excellent error handling improvement, sensei!The new error handling gracefully handles metadata fetch failures by:
- Returning an empty string instead of failing
- Properly logging warnings for debugging
- Allowing token registration to proceed
Consider adding metrics/monitoring for failed metadata fetches to track the health of external metadata services.
Consider adding the following metrics:
- Counter for failed metadata fetches
- Histogram for metadata fetch latency
- Counter for empty metadata fallbacks
📜 Review details
Configuration used: .coderabbit.yaml
Review profile: CHILL
📒 Files selected for processing (4)
crates/torii/core/src/executor/erc.rs
(1 hunks)crates/torii/core/src/executor/mod.rs
(0 hunks)crates/torii/graphql/src/object/erc/token_balance.rs
(3 hunks)crates/torii/graphql/src/object/erc/token_transfer.rs
(3 hunks)
💤 Files with no reviewable changes (1)
- crates/torii/core/src/executor/mod.rs
🔇 Additional comments (5)
crates/torii/graphql/src/object/erc/token_balance.rs (3)
106-110
: Ohayo! LGTM - SQL conditions for metadata validation
The added SQL conditions properly ensure that only tokens with valid metadata are returned.
303-303
: LGTM - Struct update aligns with SQL changes
The metadata field type change from Option<String>
to String
correctly reflects that metadata will always be present due to the SQL conditions.
241-243
: Metadata parsing looks good, sensei!
The assumption that metadata is always JSON is now safe due to the SQL conditions. However, let's verify that all existing metadata in the database is valid JSON.
✅ Verification successful
Metadata parsing is safe, sensei!
Based on the codebase analysis, the assumption that metadata is always JSON is valid because:
- The metadata is properly handled across the codebase with proper JSON parsing and error handling
- The SQL schema shows metadata is stored in a dedicated column with proper constraints
- The token metadata is fetched and validated before storage, with special handling for edge cases (e.g., control characters filtering in Loot Survivor NFT)
The expect("metadata is always json")
is justified as the codebase ensures JSON validity before storage.
🏁 Scripts executed
The following scripts were executed for the analysis:
Script:
#!/bin/bash
# Verify all metadata in the database is valid JSON
echo "SELECT metadata FROM tokens WHERE metadata IS NOT NULL AND metadata != ''" | sqlite3 -json . | \
jq -r '.[] | .metadata' | \
while read -r metadata; do
if ! echo "$metadata" | jq . >/dev/null 2>&1; then
echo "Invalid JSON found: $metadata"
fi
done
Length of output: 249
Script:
#!/bin/bash
# Let's check how metadata is handled in the codebase
rg -A 5 "metadata.*json" --type rust
# Also check for any SQL schema definitions or migrations related to metadata
fd -e sql | xargs cat
# Check for any test cases handling metadata
rg -A 5 "metadata.*test" --type rust
Length of output: 21803
crates/torii/graphql/src/object/erc/token_transfer.rs (2)
122-126
: Ohayo! Consistent metadata validation across token types
The SQL conditions mirror those in token_balance.rs, maintaining a consistent approach to metadata validation.
270-272
: Consistent metadata handling, sensei!
The metadata handling changes align with those in token_balance.rs, maintaining consistency across the codebase.
Codecov ReportAttention: Patch coverage is
Additional details and impacted files@@ Coverage Diff @@
## main #2755 +/- ##
=======================================
Coverage 56.02% 56.03%
=======================================
Files 427 427
Lines 54589 54591 +2
=======================================
+ Hits 30586 30588 +2
Misses 24003 24003 ☔ View full report in Codecov by Sentry. |
"t.metadata IS NOT NULL".to_string(), | ||
"t.metadata != ''".to_string(), |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Do we want to ignore the tokens without metadata? We may return them and the front will see that they are missing, or do we have an other reason?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
there are few cases like nft got burned in which case the metadata would no longer be present, but we still need them in db since the token id is foreign key in other tables.
and also in some cases contracts don't follow the spec properly or the json metadata is broken, etc... even in those cases we still want indexing to happen normally and we would just ignore those tokens
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
But only metadata is ignored, or the whole token? The whole token, right?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
yeah whole token, because without metadata there is no attributes, images, etc... so i dont think UI can do much with the token anyways
commit-id:5474e77e
0d7338e
to
266c8cc
Compare
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Actionable comments posted: 2
📜 Review details
Configuration used: .coderabbit.yaml
Review profile: CHILL
📒 Files selected for processing (4)
crates/torii/core/src/executor/erc.rs
(1 hunks)crates/torii/core/src/executor/mod.rs
(0 hunks)crates/torii/graphql/src/object/erc/token_balance.rs
(3 hunks)crates/torii/graphql/src/object/erc/token_transfer.rs
(3 hunks)
💤 Files with no reviewable changes (1)
- crates/torii/core/src/executor/mod.rs
🚧 Files skipped from review as they are similar to previous changes (1)
- crates/torii/core/src/executor/erc.rs
🔇 Additional comments (4)
crates/torii/graphql/src/object/erc/token_balance.rs (2)
106-109
:
Ohayo sensei! Consider revising the metadata condition.
The SQL condition allows NULL metadata, but the struct BalanceQueryResultRaw
expects metadata to always be present. This could lead to runtime errors if a NULL value is encountered.
Consider either:
- Updating the SQL condition to ensure metadata is always present:
- "(t.metadata IS NULL OR length(t.metadata) > 0)".to_string(),
+ "(t.metadata IS NOT NULL AND length(t.metadata) > 0)".to_string(),
- Or keeping metadata as
Option<String>
in the struct if NULL values are expected.
Likely invalid or redundant comment.
302-302
: Verify metadata handling consistency.
The change from Option<String>
to String
requires that metadata is always present and valid. This needs to be enforced consistently across:
- SQL queries
- JSON parsing
- Data insertion logic
✅ Verification successful
Ohayo sensei! Based on the findings, I can now provide a final response:
Metadata field change is safe and consistent with implementation
The change from Option<String>
to String
for metadata is properly enforced by the system:
- The SQL schema shows metadata is handled at insertion time in
crates/torii/core/src/executor/erc.rs
- The token insertion query includes metadata as a required field
- No UPDATE queries were found that could leave metadata in an invalid state
The change aligns with the system's design where metadata is always provided during token creation.
🏁 Scripts executed
The following scripts were executed for the analysis:
Script:
#!/bin/bash
# Check for potential NULL metadata in the database
echo "Checking for NULL metadata in tokens table..."
sqlite3 .torii/db.sqlite3 "SELECT COUNT(*) FROM tokens WHERE metadata IS NULL;"
# Check for invalid JSON in metadata
echo "Checking for invalid JSON in metadata..."
sqlite3 .torii/db.sqlite3 "SELECT id, metadata FROM tokens WHERE json_valid(metadata) = 0;"
Length of output: 563
Script:
#!/bin/bash
# Let's check the SQL schema and queries in the codebase
rg -A 5 "CREATE TABLE.*tokens"
# Check how metadata is handled in SQL queries
rg "metadata.*tokens"
# Check metadata handling in the code
ast-grep --pattern 'metadata: $_'
# Check for Option<String> patterns
rg "Option<String>"
Length of output: 8412
Script:
#!/bin/bash
# Let's check how metadata is handled in token_balance.rs and related files
rg -A 5 "metadata.*tokens" crates/torii/graphql/src/object/erc/
# Check SQL queries that insert or update metadata
rg "INSERT INTO tokens.*metadata"
rg "UPDATE tokens.*metadata"
# Check the token balance implementation
ast-grep --pattern 'impl $_ for TokenBalance'
Length of output: 1179
crates/torii/graphql/src/object/erc/token_transfer.rs (2)
122-125
:
Ohayo sensei! Similar metadata condition issue as in token_balance.rs.
The SQL condition allows NULL metadata, but the struct TransferQueryResultRaw
expects metadata to always be present.
Apply the same fix as suggested for token_balance.rs:
- "(t.metadata IS NULL OR length(t.metadata) > 0)".to_string(),
+ "(t.metadata IS NOT NULL AND length(t.metadata) > 0)".to_string(),
269-271
:
Same JSON parsing issue as in token_balance.rs.
Using .expect()
for JSON parsing could cause runtime panics.
Apply the same error handling fix as suggested for token_balance.rs.
let metadata_str = row.metadata; | ||
let metadata: serde_json::Value = | ||
serde_json::from_str(metadata_str).expect("metadata is always json"); | ||
serde_json::from_str(&metadata_str).expect("metadata is always json"); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Consider using proper error handling for JSON parsing.
Using .expect()
could cause runtime panics if the metadata is not valid JSON. Even if metadata is present, it might not be well-formed JSON.
Consider handling the JSON parsing error gracefully:
- let metadata: serde_json::Value =
- serde_json::from_str(&metadata_str).expect("metadata is always json");
+ let metadata: serde_json::Value = match serde_json::from_str(&metadata_str) {
+ Ok(value) => value,
+ Err(e) => {
+ warn!("Failed to parse metadata JSON: {}", e);
+ continue;
+ }
+ };
Committable suggestion skipped: line range outside the PR's diff.
@@ -339,7 +337,7 @@ struct TransferQueryResultRaw { | |||
pub symbol: String, | |||
pub decimals: u8, | |||
pub contract_type: String, | |||
pub metadata: Option<String>, | |||
pub metadata: String, |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
🛠️ Refactor suggestion
Add tests for metadata handling.
The change to non-optional metadata field requires comprehensive testing to ensure robustness.
Consider adding tests for:
- Valid metadata handling
- Invalid JSON metadata handling
- Edge cases (empty strings, very large metadata)
Would you like me to help create a test suite for these scenarios?
Summary by CodeRabbit
New Features
Bug Fixes
Refactor