Skip to main content
Version: v2

Database Performance

Database Performance Optimization

Creating Custom Indices

Indices are essential for optimizing database performance, especially when dealing with large datasets. By creating indices, you can significantly speed up query times. Here's how to define custom indices in your schema.

Single Column Indices

Define an entity and use the @index directive on fields you wish to add an index to.

type MyEntity {
id: ID!
userAddress: String! @index
tokenAddress: String! @index
}

The fields marked with @index will create indices in your database, making queries on these fields much faster.

Composite Indices

You can also group fields into one composite index:

type MyEntity @index(fields: ["userAddress", "tokenAddress"]) {
id: ID!
userAddress: String!
tokenAddress: String!
}

This will create a composite index on both of these fields, which is particularly useful for queries that filter on both userAddress and tokenAddress.

Automatic Indices

Please note that all id fields and @derivedFrom fields already have indices automatically created, so there is no need to add a custom index on them.

Example

type Token
@index(fields: ["id", "tokenId"])
@index(fields: ["tokenId", "collection"]) {
id: ID!
tokenId: BigInt! @index
collection: NftCollection! @index
owner: User!
}

The above example shows how to create single field indices (@index) or multi-field indices (@index(fields: ["id", "tokenId"])).

When and Why to Use Indices

Single Column Indices

  • When to use: Use single column indices when you frequently query a table based on a single field.
  • Why to use: Single column indices improve the performance of queries filtering or sorting on a single column.

Composite Indices

  • When to use: Use composite indices when you frequently query a table based on multiple fields.
  • Why to use: Composite indices improve the performance of queries that filter or sort on multiple columns simultaneously. They are especially useful when the combined columns are frequently used together in queries.

Impact on Write Times and Storage

  • Write Times: Creating indices has a minor impact on write times. Each write operation needs to update the index, but this impact is generally negligible.
  • Storage: Indices use additional storage. The storage required depends on the number and type of indices created.

Optimizing Schema and Queries

Structuring Schema

  • Ensure fields frequently used in queries are indexed.
  • Use composite indices for queries filtering on multiple fields.
  • Avoid unnecessary indices on fields rarely used in queries.

Optimizing Hasura GraphQL Queries

  • Retrieve Changed Entities: When polling for updates, retrieve only changed entities to minimize data transfer and processing.
  • Use Timestamps: Utilize timestamps to fetch only the entities that have changed since the last query.

Example Query

query getUpdatedTokens($lastFetched: timestamptz!) {
Token(where: { updatedAt: { _gt: $lastFetched } }) {
id
tokenId
collection
owner
}
}

In this example, updatedAt is a timestamp field used to fetch tokens updated since the last fetch.