Skip to content
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

Use indexation cache to satisfy "coins to spend" queries #2463

Open
wants to merge 258 commits into
base: master
Choose a base branch
from

Conversation

rafal-ch
Copy link
Contributor

@rafal-ch rafal-ch commented Nov 28, 2024

Closes #2391

This PR includes all changes from the Part 1 PR, making it deprecated.

Description

Changes in this PR:

The new CoinsToSpend index

  • This is the database that stores all coins to spend sorted by the amounts (i.e. largest-by-value coins first)
  • The key consists of several parts
    • Retryable flag - to distinguish between retryable messages and other coins
    • Address (owner)
    • AssetID
    • Amount - as "big-endian" bytes to leverage the RocksDB key sorting capabilities
    • Foreign Key - this are bytes of the key from either the Messages or Coins on-chain databases
      • for messages this is a 32-bytes Nonce
      • for coins this is a 34-bytes UtxoId
  • The value is an instance of IndexedCoinType enum, so we know which on-chain database to query when returning the actual coins
  • This index is updated when executor events are processed
  • When querying for "coins to spend" the following algorithm is applied:
    • First, we get as many "big" coins as required to satisfy double the amount from the query (respecting max and excluded params)
    • If we have enough coins, but there are still some "slots" in the query left (because we selected less coins than max) we fill the remaining slots with a random number of "dust" coins
    • If it happens that the value of selected "dust coins" is able to cover the value of some of the already selected "big coins", we remove the latter from the response
    • If at any step we encounter a problem (reading from database, integer conversions, etc.) we bail with an appropriate error

Changes to CoinsQueryError type

  • The MaxCoinsReached variant has been removed because in the new algorithm we never query for more coins than the specified max, hence, without additional effort, we are not able to tell whether the query could be satisfied if user provided bigger max
  • The InsufficientCoins has been renamed to InsufficientCoinsForTheMax and it now contains the additional max field

Off-chain database metadata

  • The metadata for off-chain database now contain the additional IndexationKind - CoinsToSpend

Refactoring

  • The indexation.rs module was split into separate files, each per indexation type + errors + some utils.

Other

  • Integration tests have to be updated to not expect the exact number of coins to spend in the response (currently, due to randomness, we're not deterministic in this regard)
  • The number of excluded ids in the coinsToSpend GraphQL query is now limited to the maximum number of inputs allowed in transaction.

Before requesting review

  • I have reviewed the code myself
  • I have created follow-up issues caused by this PR and linked them here

Follow-up issues

@rafal-ch rafal-ch requested a review from xgreenx December 13, 2024 12:54
@netrome netrome self-requested a review December 16, 2024 15:14
Copy link
Contributor

@netrome netrome left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Nice stuff so far. I have only done a relatively shallow sweep and will get back again during the week for a deeper look. I have a few nits and suggestions so far, but nothing major.

I'm not a fan of the lack of explicit error handling in crates/fuel-core/src/graphql_api/storage/coins.rs but other than that I think the code looks generally really well-written.

crates/fuel-core/src/coins_query.rs Show resolved Hide resolved
crates/fuel-core/src/graphql_api.rs Show resolved Hide resolved
crates/fuel-core/src/graphql_api/indexation/balances.rs Outdated Show resolved Hide resolved
crates/fuel-core/src/graphql_api/indexation/balances.rs Outdated Show resolved Hide resolved
crates/fuel-core/src/graphql_api/indexation/balances.rs Outdated Show resolved Hide resolved
crates/fuel-core/src/graphql_api/storage/coins.rs Outdated Show resolved Hide resolved
crates/fuel-core/src/graphql_api/storage/coins.rs Outdated Show resolved Hide resolved
crates/fuel-core/src/graphql_api/storage/coins.rs Outdated Show resolved Hide resolved
crates/fuel-core/src/graphql_api/storage/coins.rs Outdated Show resolved Hide resolved
crates/fuel-core/src/graphql_api/storage/coins.rs Outdated Show resolved Hide resolved
rafal-ch added a commit that referenced this pull request Jan 2, 2025
Closes #2474

## Description
This PR changes the following:
1. Adds new test in `tests/tests/relayer.rs` - to ensure that only
**non**-retryable messages contribute to the balance returned from
`balance()`, `balances()` and `coins_to_spend()`[^1] endpoints
2. Modifies the `balance()` test in `tests/tests/balances.rs` to also
include some retryable message which is expected **not** to contribute
to the total balance.
   * for `balances()` and `coins_to_spend()` it was already the case

1<sup>st</sup> test checks the behavior of the actual blockchain
operations while the 2<sup>nd</sup> tests the proper initialization of
the balances index at genesis.

## Checklist
- [X] Breaking changes are clearly marked as such in the PR description
and changelog
- [X] New behavior is reflected in tests

### Before requesting review
- [X] I have reviewed the code myself

[^1]: currently, `coins_to_spend()` is not using indexation, this will
change when [this PR](#2463)
is merged.

---------

Co-authored-by: Green Baneling <[email protected]>
Copy link
Contributor

@netrome netrome left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Apologies for taking longer than expected. While most of this looks good to me, I have a few questions of some things that puzzle me. Moreover, there are two concrete things I'd like to see updated before approving:

First, we should update the description and name of the random_improve function. Second, I'd like to see an integration test that shows how this algorithm behaves end to end.

crates/fuel-core/src/coins_query.rs Show resolved Hide resolved
crates/fuel-core/src/coins_query.rs Show resolved Hide resolved
crates/fuel-core/src/coins_query.rs Show resolved Hide resolved
tests/tests/coins.rs Show resolved Hide resolved
netrome
netrome previously approved these changes Jan 8, 2025
Copy link
Contributor

@netrome netrome left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Thanks for updating. I would like to see the "random_improve" function renamed, perhaps just to "random" but I can live with the old name as well.

@rafal-ch
Copy link
Contributor Author

rafal-ch commented Jan 9, 2025

Thanks for updating. I would like to see the "random_improve" function renamed, perhaps just to "random" but I can live with the old name as well.

My understanding was that the "random improve" is actually the name of the algorithm. The linked article literally mentions this name. Cardano seems to be using the exact same name as well.

// We aim to reduce dust creation by targeting twice the required amount for selection,
// inspired by the random-improve approach. This increases the likelihood of generating
// useful change outputs for future transactions, minimizing unusable dust outputs.
// See also "let upper_target = target.saturating_mul(2);" in "fn random_improve()".
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Very interesting !

let storage = block_st_transaction.storage::<CoinsToSpendIndex>();
let maybe_old_value = storage.replace(&key, &IndexedCoinType::Coin)?;
if maybe_old_value.is_some() {
return Err(IndexationError::CoinToSpendAlreadyIndexed {
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

If we return an error here the coin will still be inserted in the transaction and not revert. If we drop the transaction in an outer function it might not be an issue but maybe the dev of the outer function isn't aware of this. Same for add_messages

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
Projects
None yet
Development

Successfully merging this pull request may close these issues.

Optimize GraphQL "coins to spend" query
5 participants