Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Checkpoint is a library for indexing data from Starknet events and making it accessible through GraphQL. Checkpoint is inspired by The Graph and focused on providing similar functionality for Starknet.
This diagram gives more detail about the flow of data once a Checkpoint instance has started processing Starknet transactions:
As highlighted in the flow diagram above:
A decentralized application adds data to Starknet through a transaction on a smart contract.
The smart contract emits one or more events while processing the transaction. Checkpoint continually scans Starknet for new blocks and checks if it contains events for your configured contracts they may contain.
The decentralized application queries the Checkpoint GraphQL API. Checkpoint translates the GraphQL queries into SQL queries to fetch this entity data from the database. The decentralized application displays this data in a rich UI for end-users, which they can also use to issue new transactions on Starknet, and the cycle repeats.
Checkpoint is an NPM package that can be installed through the following command:
Follow our handy guides to get started with Checkpoint as quickly as possible:
Learn the fundamentals of Checkpoint to get a deeper understanding of how it works:
Checkpoint learns what and how to index Starknet data based on configuration parameters, known as the . Checkpoint configuration defines the smart contracts of interest and the relevant events to be tracked. The logic for how to map events to data that Checkpoint will store in its database is defined in user-defined Javascript functions called . Checkpoint then exposes the stored data to the public through a GraphQL API.
Checkpoint calls the data writer function for the respective event. are responsible for writing entity objects to the database.
For each block where Checkpoint encounters relevant events, it creates a record in the database to keep track of it. These records can be queried through the GraphQL API using the _checkpoints
query. You can read more about how to do that .
Checkpoint's time travel queries feature allows to access historical data at a specific block number. By specifying a block, you can retrieve data from that exact point in the past.
In this guide, we will set up a project that uses Checkpoint to expose data.
To successfully, follow this guide and run the template project, you'll need the following dependencies set up on your computer:
Node.js (>=14.x.x)
Yarn
PostgreSQL (or Docker to quickly run an image).
First, install Node dependencies by running the following command in the project's directory:
Checkpoint projects (and by extension this template) require a PostgreSQL database connection to store indexed data. If you have a PostgreSQL server running, then create a copy of the .env.example
file and name .env
. Then update the DATABASE_URL
value in the .env
file to match the connection string to your database.
Next, start the server by running:
This will start the indexing process and also startup the GraphQL server for querying indexed information.
Now that the server is running, open up your browser and visit http://localhost:3000. This will show a GraphiQL interface to explore the exposed data.
Try executing the following query:
You should get a response similar to the one in this image:
The response to your queries depends on how much on-chain data Checkpoint has indexed. If the server has run long enough, you'll get more results. Checkpoint exposes a _metadata
query, which can be used to track the latest block it has indexed. Example of this is:
Here is a quick description of important files within the template project and what each does.
You are encouraged to try modifying the template project to understand how Checkpoint works and quickly get started using it to index your contracts data on Starknet.
First, clone the Checkpoint starter template project that can be found at: .
This starter project defines a Post
entity in its src/schema.gql
schema file and is setup to write data based on that structure. Read more on how Entity schemas are used to generate GraphQL queries .
: Defines the Post
entity Checkpoint uses to generate API queries.
: Defines the object used to initialize Checkpoint and it contains the details of a Poster contracts deployed on Starknet mainnet and Starknet sepolia networks.
: Defines the responsible for writing Post
data whenever Checkpoint encounters a new_post
event.
: This is the entrypoint, that initializes Checkpoint, configures indexers for each network and exposes the GraphQL API.
: Defines blocks where we are sure the contracts events exists. This is used to seed checkpoint inside the index.ts file to speed up Checkpoints indexing.
It's possible to create one-to-one relations in schema. Those relations will be then available for querying via GraphQL.
To define one-to-one relation nested entity must have id
field that is either String!
or ID!
. In parent entity create field (with any name) that is of nested entity's type (can be nullable or required).
When creating entities in writer set Proposal
's space
field to the value of Space
's id
field.
It's now possible to nest space entity when querying proposals. You can still filter proposals by space (limited to Space
's id
currently).
In many cases there are contracts that should be tracked but their addresses are not known right away so they can't be configured in static config. Templates make it possible to define templates for such contracts that can be dynamically turned into sources whenever writer becomes aware of new contract's existence - for example when new contract has been deployed through factory.
Sample config might look like this:
After template is defined it can be executed in any writer to start tracking specific contract address for events defined in that template:
Once template has been executed all events defined in that template will be tracked for given contract address.
Checkpoint initially supported indexing Starknet contracts only, but it can also index Ethereum contracts now.
Usage with Ethereum is very similar to usage with Starknet, differences are:
Use full signature for event name in config (ProposalUpdated(uint256,(address,bytes),string)
instead of ProposalUpdated
)
Your writers should be using evm.Writer
instead of starknet.Writer
.
You should create indexer using new evm.EvmIndexer
instead of new starknet.StarknetIndexer
.
Templates are defined in and each template has a name that is later used when executing it.
Checkpoint can work with any network and chain as long as there is implemented for it. Currently there is official support for Starknet and Ethereum (or any other EVM network), but external providers are supported as well.
Checkpoint provides support for BigInt
, Decimal
, BigDecimal
. It's also possible to define custom (or define existing) decimal types.
BigInt and BigDecimal types are mapped to following PostgreSQL types by default:
BigInt
-> bigint
Decimal
-> decimal(10, 2)
BigDecimal
-> decimal(20, 8)
To use default (or custom) BigInt and BigDecimal type it needs to be predeclared in GraphQL schema using scalar
keyword. Once it's declared it can be used as field's type.
It's possible to define new or to redefine existing decimal types using overrides. This file should be called overrides.json and stored in your source directory.
You can then pass those to Checkpoint via options:
Checkpoint is a Node.js library and running the following NPM commands will install it:
Two arguments can be used to initialize a Checkpoint instance. These are:
GraphQL schema (required)
Mainnet: 0x0654e9232d5f402829755029901f69c32b423ded0f8c081e416e3b24f5a7a46e
Sepolia: 0x03aa7630a4f9c5108bf3cd1910c7d45404cba865fc0fc0756bf9eedc073a98a9
With new versions of Checkpoint contracts on different networks can be indexed and queried from single Checkpoint instance.
In this case we want to index Poster contract on both Starknet mainnet and Starknet sepolia so we define two configurations.
The start
block number is set to 639485
because our mainnet contract was deployed at that block. This will mean Checkpoint starts scanning from that block as opposed to starting at block 0.
Checkpoint requires a set of defined GraphQL Schema Objects. These schema objects will be used to create the database tables for indexing records and also generate graphql queries for accessing the indexed data.
For this guide, we will want to track a Post
entity and have it exposed via the graphql API. This entity can be defined as the following schema file:
When updating your schema you should run following script to generate ORM models:
Data writers are JavaScript functions that get invoked by Checkpoint when it discovers a block containing this event. A Data writer is responsible for writing records to the database. These records will eventually be exposed via Checkpoint's GraphQL endpoint.
We have defined data writer function in our Checkpoint configuration for new_post
event called handleNewPost
.
Let's create these data writer functions:
With the above code snippet, we have a data writer that writes new posts to the PostgreSQL database.
Finally, we can initialize a checkpoint instance with our arguments like these:
Next, we start up checkpoint's indexer like this:
The above code will start the checkpoint indexer, and it will begin processing each Starknet block. When a relevant contract event is found, it gets passed to the data writer for that event.
Once Checkpoint has run for a while and has written some data to its database, you can query this data using the generated GraphQL API.
You can mount the query endpoint on any port you like using the graphql
handler exported by Checkpoint's object. Like this:
This will mount a GraphQL endpoint that can be accessed at http://localhost:3000/graphql (and a GraphiQL interface at http://localhost:3000/graphql when visited from the browser).
Checkpoint exposes two types of queries:
These enables users to fetch information about indexer.
For starters, you can visit http://localhost:3000/graphql in your browser and try running the sample query generated in the graphiql UI.
At this juncture, you should have Checkpoint running, indexing your contracts data (on multiple networks) and serving this indexed data via graphql.
(optional)
A Checkpoint configuration is an object that defines the smart contract addresses and their respective events. For this guide, we will be indexing the list of posts and authors for this . A copy of this contract is deployed on following Starknet networks:
To successfully track the addresses of authors, you'll need to listen to events from the contract. Therefore, a valid checkpoint configuration for the above requirements will be:
fn
value is the name of the data writer function to be invoked when new_post
events are encountered. Read more about Checkpoint configuration .
Checkpoint will use the above entity (Post)
to generate a PostgreSQL database table named posts
with columns matching the defined fields. It will also generate a list of GraphQL queries to enable querying indexed data. Read more about how queries are generated .
You can view a more comprehensive data writer example in our checkpoint-template codebase .
Next up, you can explore our template repository to get up and running quickly.
You can also explore some Checkpoint or move on to the next guide.
Checkpoint uses a configuration object to determine which networks and contract information it will be indexing.
Checkpoint supports Express.js by default, but it's also possible to use Checkpoint with ApolloServer. Assuming ApolloServer is already installed in your Checkpoint project you can start ApolloServer using Checkpoint's schema with following code:
Currently only Express.js is officially supported. ApolloServer compatibility is only provided as alternative for user's convenience, but it's not guaranteed that it will support all ApolloServer's features.
When initializing Checkpoint library there is an optional parameter that can be passed to configure some extra behavior of the library.
The type definition for the option object is:
This option can be provided as the final argument to the Checkpoint constructor like this:
See more about the configuration options below.
By default, when a Checkpoint
object is started, it looks up the DATABASE_URL
environment variable, but with dbConnection
option parameter, you can specify a different connection string within the codebase itself and this value will override the DATABASE_URL
value in the environment when connecting to the database.
There are six (6) log levels currently supported by Checkpoint, these are:
In a non-production environment, you can set the prettifyLogs
option to true
and this will output a pretty version
You can seed Checkpoint with a list of blocks, and Checkpoint will start by scanning this list of blocks first and invokes the appropriate data writers for any events founds. Once Checkpoint checks through the list of seeded blocks, it continues sequential scanning from the next block.
There is a seedCheckpoints
method defined on Checkpoint instances:
This seedCheckpoints
method should be called before starting the indexer, and the method can be called with as many contracts and blocks you already have.\
You can set up a process (or external service) that periodically uses the _checkpoints
query to fetch the latest blocks and export them for archiving or sharing with another instance of Checkpoint.
Using our Poster contract examples from above, to fetch all blocks where Checkpoint has encountered an event, you can run the following query:
Used to redefine available in the schema.
For example, in the checkpoint template , seed blocks for the Poster contracts are defined in the `checkpoints.json` file as:
And this array is used as an argument when like:
For each block where Checkpoint encounters relevant events, it creates a record in the database to keep track of it. These records can be queried through the GraphQL API using the _checkpoints
query. You can read more about how to do that .
The above query will fetch the subsequent 100 blocks after the `221984` block where Checkpoint encounters an events for the contracts address (0x04d10712e72b971262f5df09506bbdbdd7f729724030fa909e8c8e7ac2fd0012
).
Checkpoint introduces a robust ORM to streamline your interactions with the database.
Checkpoint enables data manipulation in your blockchain database without having to construct SQL queries directly. It features a capability to generate model objects from a GraphQL schema, and this guide will lead you through the process of creating and saving a model object into your database.
The first step is to generate the models from your GraphQL schema. This is done with the checkpoint generate
command. Run it in your project environment like this:
After running this command, your models will be created based on your GraphQL schema.
To use the generated models in your scripts, you need to import them. This is done with an import
statement at the top of your file. For instance, if you've generated a Post
model, you can import it like this:
This allows you to use the Post
class to create new posts and save them to the database.
Each model is defined in a GraphQL schema file, such as schema.gql
. Every model object will have a unique id
along with other properties. Here's an example of a Post
model:
Please note, this Post
model is merely an example. Your actual model will have properties specific to your needs.
To create a new instance of your model, you instantiate it with the id
and indexer name. Here's how you'd create a new Post
:
You then set the other properties:
Once the instance is defined, you can load an entity and edit it using loadEntity
like that :
Once the instance is fully defined, you can save it into your database using the save
method. This operation is asynchronous, so you should use await
:
Checkpoint requires a set of defined GraphQL Schema Objects. The schema objects will be used to create a database structure for indexing this data and also generating GraphQL Queries for accessing the indexed data.
In Checkpoint terms, these schema objects are called Entities
, an Entity can be defined as a GraphQL Object with a unique name and an id
field.
For example:
Vote
and User
are valid entities because their names are unique (within the schema) and contain an id
field.
Checkpoint generates two Query fields for each of the defined entities. One for querying a single entity record by its id and the second for querying multiple records of an entity.
For example, using the earlier defined User
entity, Checkpoint will generate two query fields like:
Things to note:
The name of the single record entity query is derived from the entity's name lowercased.
The name of the multi-record entity query is derived from the entity's name lowercased with an s
suffix.
The generated Where*
types fields for multi-record are derived based on original fields in the entity, and non-null values are treated a AND
where filters when being executed against the database.
Data writers are callback functions that Checkpoint invokes when the event of a contract is discovered at a particular block.
Here is a video presentation of Checkpoint by at StarkWare Sessions 23.
Currently, Checkpoint exposes the following internal data queries:
These are used to query single or multiple metadata records. Metadata records are key-value pairs of data used by Checkpoint internally to describe the state of its process. These queries are defined as:
For starters, you can execute the following query to see a list of all metadata values exposed by Checkpoint:
Internally, Checkpoint keeps track of blocks where contracts event are found. These records are usually used by Checkpoint to speed up re-indexing when restarted. The _checkpoint(s)
queries provide a way to query these blocks. The results can be exported and used to seed another Checkpoint instance running on another machine.
These queries are defined as:
For example, you can run the following query to fetch all blocks where the event of a particular contract can be found:
Checkpoint also exposes some queries to inspect the internal state of how the indexer is running. Typically, internal queries are usually prefixed with an underscore (_
) and are structured in a similar way to how are generated.
Snapshot X, the onchain voting protocol by Snapshot, leverages Checkpoint to efficiently build and expose its GraphQL API, which is sourced from data in its Starknet and EVM contracts. You can find the code here:
Data writers are callback functions that Checkpoint invokes when the event of a contract is discovered at a particular block.