- Introduction
- ChainIndexer Config
- Upgrade
- Backfill
- Regular Checks
- Downgrade Steps
- Terminology
- Appendix
This document is for externally-available, high-performance RPC providers and for node operators who use or expose the Ethereum and/or events APIs. It walks through the configuration changes, migration flow and operations/maintenance work needed to enable, backfill and maintain the ChainIndexer
. The justification for and benefits of the ChainIndexer
are documented here.
The ChainIndexer is now also required if you enable:
- Ethereum (
eth_*
) APIs using theEnableEthRPC
Lotus configuration option OR - ActorEvent APIs using the
EnableActorEventsAPI
Lotus configuration option
Note: If you are a Storage Provider or node operator who does not serve public RPC requests or does not need Ethereum or Event APIs (i.e, if Fevm.EnableEthRPC = false
and Events.EnableActorEventsAPI = false
, their default values), you can skip this document as the ChainIndexer
is already disabled by default.
The following must be enabled on an Lotus node before starting as they are disabled by default:
[Fevm]
# Enable the ETH RPC APIs.
# This is not required for ChainIndexer support, but ChainIndexer is required if you enable this.
EnableEthRPC = true
[Events]
# Enable the Actor Events APIs.
# This is not required for ChainIndexer support, but ChainIndexer is required if you enable this.
EnableActorEventsAPI = true
[ChainIndexer]
# Enable the ChainIndexer, which is required for the ETH RPC APIs and Actor Events APIs.
# If they are enabled, but the ChainIndexer is not, Lotus will exit during startup.
# (ChainIndexer needs to be explicitly enabled to signal to the node operator the extra
# supporting functionality that will now be running.)
EnableIndexer = true
You can learn more about these configuration options and other configuration options available for the ChainIndexer
here.
The ChainIndexer
includes a garbage collection (GC) mechanism to manage the amount of historical data retained. See the ChainIndexer size requirements.
By default, GC is disabled to preserve all indexed data.
To configure GC, use the GCRetentionEpochs
parameter in the ChainIndexer
section of your config.
The ChainIndexer periodically runs GC if GCRetentionEpochs
is > 0 and removes indexed data for epochs older than (current_head_height - GCRetentionEpochs)
.
[ChainIndexer]
GCRetentionEpochs = X # Replace X with your desired value
- Setting
GCRetentionEpochs
to 0 (default) disables GC. - Any positive value enables GC and determines the number of epochs of historical data to retain.
-
Archival Nodes: Keep GC disabled (
GCRetentionEpochs
= 0) to retain all indexed data. -
Non-Archival Nodes: Set
GCRetentionEpochs
to match the amount of chain state your node retains
Example: if your node is configured to retain Filecoin chain state with a Splitstore Hotstore that approximates 2 days of epochs, set GCRetentionEpochs
to at least retentionDays * epochsPerDay = 2 * 2880 = 5760
).
Warning: Setting this value below the chain state retention period may degrade RPC performance and reliability because the ChainIndexer will lack data for epochs still present in the chain state.
Note: Chainstore.Splitstore
is configured in terms of bytes (not epochs) and ChainIndexer.GCRetentionEpochs
is in terms of epochs (not bytes). For the purposes of this discussion, we're assuming operators have determined Chainstore.Splitstore.HotStoreMaxSpaceTarget
and Chainstore.Splitstore.HotStoreMaxSpaceThreshold
values that approximate a certain number days of storage in the Splitstore Hotstore. The guidance here is to make sure this approximation exceeds ChainIndexer.GCRetentionEpochs
.
Note: The following config options no longer exist in Lotus and have been removed in favor of the ChainIndexer config options explained above. They can be removed when upgrading to Lotus v1.31.0.
[Fevm]
EthTxHashMappingLifetimeDays
[Events]
DisableRealTimeFilterAPI
DisableHistoricFilterAPI
DatabasePath
[Index]
EnableMsgIndex
The Fevm.Events
options were marked as deprecated in Lotus 1.26, having been moved to the new top-level Events
section, and have now been removed with Lotus 1.31.
Fevm.Events.DatabasePath
(no replacement available)Fevm.Events.DisableRealTimeFilterAPI
(no replacement available)Fevm.Events.DisableHistoricFilterAPI
(no replacement available)Fevm.Events.FilterTTL
(useEvents.FilterTTL
intead)Fevm.Events.MaxFilters
(useEvents.MaxFilters
instead)Fevm.Events.MaxFilterResults
(useEvents.MaxFilterResults
instead)Fevm.Events.MaxFilterHeightRange
(useEvents.MaxFilterHeightRange
instead)
One can upgrade/downgrade between pre-ChainIndexer and with-ChainIndexer Lotus versions without conflict because they persist state to different directories and don't rely on each other. No backup is necessary (but extra backups don't hurt). There is still a backfilling step though when downgrading.
These upgrade steps assume one has multiple nodes in their fleet and can afford to have a node not handling traffic, potentially for days per backfill timing below.
One should also check to ensure they have sufficient disk space.
- This upgrade path assumes one has an existing node with existing
LOTUS_PATH
chain state they want to keep using and they don't want to import chain state from a snapshot. A prime example is an existing archival node. - Perform the preparation steps before proceeding.
- See here for the snapshot upgrade path.
- Route traffic away from an initial node
- Example: prevent a load balancer from routing traffic to a designated node.
- Stop the designated Lotus Node
- Stop the designated Lotus node before starting the upgrade and backfill process.
- Update Configuration
- Modify the Lotus configuration to enable the
ChainIndexer
as described in theChainIndexer Config
section above.
- Modify the Lotus configuration to enable the
- Restart Lotus Node
- Restart the Lotus node with the new configuration.
- The
ChainIndexer
will begin indexing real-time chain state changes immediately in the${LOTUS_PATH}/chainindex
directory. - However, it will not automatically index any historical chain state (i.e., any previously existing chain state prior to the upgrade).
- Backfill
- See the "Backfill" section below.
- This could potentially take days per Backfill Timing.
- Ensure node health
- Perform whatever steps are usually done to validate a node's health before handling traffic (e.g., log scans, smoke tests)
- Route traffic to the backfilled node that is now using ChainIndexer
- Ensure equal or better correctness and performance
- ChainIndexer-using nodes should have full correctness and better performance when compared to pre-ChainIndexer nodes.
Part 3 below is going to use the backfilled chainindex.db
from above with other nodes so they don't have to undergo as long of a backfill process. That said, this backfilled chaindex.db
shouldn't be done while the updated-and-backfilled node is running. Options include:
- Stop the updated-and-backfilled node before copying it.
cp ${LOTUS_PATH}/chainindex/chainindex.db /copy/destination/path/chainindex.db
- While the node is running, use the
sqlite3
CLI utility (which should be at least version 3.37) to clone it.
sqlite3 ${LOTUS_PATH}/chainindex/chainindex.db '.clone /copy/destination/path/chainindex.db'
Both of these will result in a file/copy/destination/path/chainindex.db
that can be copied around in part 4 below.
Now that one has a ${LOTUS_PATH}/chainindex/chainindex.db
from a trusted node, it can be copied to additional nodes to expedite bootstrapping.
- Route traffic away from the next node to upgrade
- Stop the Lotus Node
- Update Configuration
- Modify the Lotus configuration to enable the
ChainIndexer
as described in theChainIndexer Config
section above.
- Modify the Lotus configuration to enable the
- Copy
/copy/destination/path/chainindex.db
from the trusted node in part 2 above - Restart Lotus Node
- Restart your Lotus node with the new configuration.
- The
ChainIndexer
will begin indexing real-time chain state changes immediately in the${LOTUS_PATH}/chainindex
directory. - However, it will not automatically index the chain state from where the copied-in
chainindex.db
ends. This will need to be done manually.
- Backfill the small data gap from after the copied-in
chainindex.db
- See the
Backfill
section below. - This should be quick since this gaps is presumably on the order of epoch minutes, hours, or days rather than months.
- See the
- Ensure node health
- Perform whatever steps are usually done to validate a node's health before handling traffic (e.g., log scans, smoke tests)
- Route traffic to this newly upgraded ChainIndexer-enabled node
- Repeat for other nodes that need to upgrade
It's recommended to keep the pre-ChainIndexer indexing database directory (${LOTUS_PATH}/sqlite
) around until you've confirmed you don't need to downgrade. After sustained successful operations after the upgrade, the pre-ChainIndexer database directory can be removed to reclaim disk space.
Note: this upgrade path assumes one is starting a fresh node and importing chain state with a snapshot (i.e., lotus daemon --import-snapshot
). A prime example is an operator adding another node to their fleet that has limited history. If not using a snapshot, see the "upgrade with existing chain state" path.
- Review the preparation steps
- The disk space and upgrade times will be much smaller than the "upgrade with existing chain state" path assuming this is a non-archival node that is only indexing a limited number of days of epochs.
- Ensure the node is stopped and won't take any traffic initially upon starting
- Example: prevent a load balancer from routing traffic to the node.
- Update Configuration
- Modify the Lotus configuration to enable the
ChainIndexer
as described in theChainIndexer Config
section above.
- Modify the Lotus configuration to enable the
- Start lotus with the snapshot import
lotus daemon --import-snapshot
- Wait for the Lotus daemon to sync
- As the Lotus daemon syncs the chain, the ChainIndexer will automatically index the synced messages, but it will not automatically sync ETH RPC events and transactions.
- Backfill so ETH RPC events and transactions are indexed as well
- See the "Backfill" section below.
- This will look something like
lotus index validate-backfill --from <head_epoch> --to <epoch_corresponding_with_how_much_state_in_past_want_to_index> --backfill
- Example: if the current head is epoch 4360000 and one wants to index a day's worth of epochs (2880), then they'd use
--from 4360000 --to 4357120
- Example: if the current head is epoch 4360000 and one wants to index a day's worth of epochs (2880), then they'd use
- Ensure node health
- Perform whatever steps are usually done to validate a node's health before handling traffic (e.g., log scans, smoke tests)
- Route traffic to the backfilled node that is now using ChainIndexer
There is no automated migration from pre-ChainIndexer indices to the ChainIndex. Instead one needs to index historical chain state (i.e., backfill), if RPC access to that historical state is required. (If curious, read why.)
Backfilling the new ChainIndexer
was benchmarked to take approximately ~12 hours per month of epochs on a sample archival node doing no other work. Your results will vary depending on hardware, network, and competing processes. This means if one is upgrading a FEVM archival node, they should plan on the node being out of production service for ~10 days. Additional nodes to update don't need to go through the same time-intensive process though. They can get a ${LOTUS_PATH}/chainindex/chainindex.db
copied from a trusted node per the upgrade steps.
As of 202410, ChainIndexer will accumulate approximately ~340 MiB per day of data, or 10 GiB per month (see here).
The lotus index validate-backfill
command is a tool for validating and optionally backfilling the chain index over a range of epochs since calling the ChainValidateIndex
API for a single epoch at a time can be cumbersome, especially when backfilling or validating the index over a range of historical epochs, such as during a backfill. This tool wraps the ChainValidateIndex
API to efficiently process multiple epochs.
Note: This command can only be run when the Lotus daemon is already running with the ChainIndexer
enabled as it depends on the ChainValidateIndex
RPC API.
lotus index validate-backfill --from <start_epoch> --to <end_epoch> [--backfill] [--log-good]
The command validates the chain index entries for each epoch in the specified range, checking for missing or inconsistent entries (i.e. the indexed data does not match the actual chain state). If --backfill
is enabled (which it is by default), it will attempt to backfill any missing entries using the ChainValidateIndex
API.
You can learn about how to use the tool with lotus index validate-backfill -h
.
Note: If you are using a non-standard Lotus repo directory then you can run the command with lotus -repo /path/to/lotus/repo chainindex validate-backfill ...
, or by setting the LOTUS_REPO
environment variable.
During normal operation, it is possible, but not strictly necessary, to run periodic checks on the index to ensure it remains consistent with the chain state. The ChainIndexer is designed to be resilient and consistent, but unconsidered edge-cases, or bugs, could cause the index to become inconsistent.
The lotus index validate-backfill
command can be used to validate the index over a range of epochs and can be run periodically via cron, systemd timers, or some other means, to ensure the index remains consistent. An example bash script one could use to validate the index over the last 24 hours every 24 hours is provided below:
#!/bin/bash
LOGFILE="/var/log/lotus_chainindex_validate.log"
current_date=$(date '+%Y-%m-%d %H:%M:%S')
# Configurable setting for backfill option, set to 'false' to simply report errors as we should
# not expect regular errors in the index.
BACKFILL_OPTION=false
# Path to the lotus binary
LOTUS_BIN_PATH="/path/to/lotus"
# Get the current chain head epoch number
start_epoch=$(lotus chain head --height)
# Subtract 1 to account for deferred execution
start_epoch=$((start_epoch - 1))
# Define the number of epochs for validation, set to 3000 to validate the last 24 hours plus some buffer
epochs_to_validate=3000
# Calculate the end epoch
end_epoch=$((start_epoch - epochs_to_validate + 1))
# Run the Lotus chainindex validate-backfill command
validation_output=$("$LOTUS_BIN_PATH" chainindex validate-backfill --from="$start_epoch" --to="$end_epoch" --backfill="$BACKFILL_OPTION" --quiet 2>&1)
# Check the exit status of the command to determine if errors occurred
if [ $? -ne 0 ]; then
# Log the error with a timestamp
{
echo "[$current_date] Validation error:"
echo "$validation_output"
} >> "$LOGFILE"
else
echo "[$current_date] Validation completed successfully." >> "$LOGFILE"
fi
Note that this script simply logs any errors that occur during the validation process. It is up to the operator to determine the appropriate response to any errors that occur, including reporting potential bugs to Lotus maintainers. A further enhancement could be to send an alert to an operator if an error occurs.
In case you need to downgrade to the previous indexing system, follow these steps:
- Prevent the node from receiving traffic.
- Stop your Lotus node.
- Download or build a Lotus binary for the rollback version which has the implementation of the old
EthTxHashLookup
,MsgIndex
, andEventIndex
indices. - Ensure that you've set the correct config for the existing
EthTxHashLookup
,MsgIndex
, andEventIndex
indices in theconfig.toml
file. - Restart your Lotus node.
- Backfill the
EthTxHashLookup
,MsgIndex
, andEventIndex
indices using thelotus-shed index backfill-*
CLI tooling available in the previous indexing system for the range of epochs between the upgrade toChainIndexer
and the rollback ofChainIndexer
. - Route traffic back to the node.
- This corresponds to the indexing system used in Lotus versions before v1.31.0.
- It has been replaced by the ChainIndexer.
- It was composed of three indexers using three separate databases:
EthTxHashLookup
,MsgIndex
, andEventIndex
. - It persisted state to the removed option for
Events.DatabasePath
, which defaulted to${LOTUS_PATH}/sqlite
. - It had CLI backfill tooling:
lotus-shed index backfill-*
- This corresponds to the indexing system used in Lotus versions v1.31.0 onwards.
- It replaced the previous indexing system.
- It is composed of a single indexer,
ChainIndexer
, using a single database for transactions, messages, and events. - It persists state to
${LOTUS_PATH}/chainindex
. - It has this CLI backfill tooling:
lotus index validate-backfill
- Storage requirements: See the backfill disk space requirements.
- Backfil times: See the backfill timing.
Why isn't there an automated migration from the previous indexing system to the ChainIndexer indexing system?
The decision to not invest here ultimately comes down to the development-time cost vs. benefit ratio.
For archival nodes, we don't have the confidence that the previous indexing system has the correct data to bootstrap from. In 2024, Lotus maintainers have fixed multiple bugs in the previous indexing system, but they still see reports of missing data, mismatched event index counts, etc. Investing here in a migration isn't guaranteed to yield a correct index. As a result, one would still need to perform the backfill steps to validate and correct the data anyway. While this should be faster having partially correct data than no data, it would still require an archival node to take an outage on the order of days which isn't good enough.
The schemas of the old fragmented Indices don't naturally map to the schema of the ChainIndexer. There would be additional data wrangling work to ultimately get this right.
Backfilling is a one time cost. If an operator provider is running multiple nodes, they only need to do it on one node and can then simply copy over the Index to the other node per the upgrade steps. The new chainindex.db
copy can also be shared among operators if there is a trust relationship.
Note that this lack of an automated migration is primarily a concern for the relatively small-in-number archival nodes. It isn't as much of a concern for snapshot-synced nodes. For snapshot-synced nodes with only a portion of the chain state because they only serve queries going back a few days can expect the backfill take closer to an hour per backfill timing.
Please refer to the Lotus API documentation for detailed documentation of the ChainValidateIndex
JSON RPC API.
The ChainValidateIndex
JSON RPC API serves a dual purpose: it validates/diagnoses the integrity of the index at a specific epoch (i.e., it ensures consistency between indexed data and actual chain state), while also providing the option to backfill the ChainIndexer
if it does not have data for the specified epoch.
The ChainValidateIndex
RPC API is available for use once the Lotus daemon has started with ChainIndexer
enabled.
Here are some examples of how to use the ChainValidateIndex
JSON RPC API for validating/ backfilling the index:
- Validating the index for an epoch that is a NULL round:
curl -X POST -H "Content-Type: application/json" --data '{
"jsonrpc": "2.0",
"method": "Filecoin.ChainValidateIndex",
"params": [1954383, false],
"id": 1
}' http://localhost:1234/rpc/v1 | jq .
{
"id": 1,
"jsonrpc": "2.0",
"result": {
"TipSetKey": [],
"Height": 1954383,
"IndexedMessagesCount": 0,
"IndexedEventsCount": 0,
"Backfilled": false,
"IsNullRound": true
}
}
- Validating the Index for an epoch for which the Indexer has missing data with backfilling disabled:
curl -X POST -H "Content-Type: application/json" --data '{
"jsonrpc": "2.0",
"method": "Filecoin.ChainValidateIndex",
"params": [1995103, false],
"id": 1
}' http://localhost:1234/rpc/v1 | jq .
{
"error": {
"code": 1,
"message": "missing tipset at height 1995103 in the chain index, set backfill flag to true to fix"
},
"id": 1,
"jsonrpc": "2.0"
}
- Validating the Index for an epoch for which the Indexer has missing data with backfilling enabled:
curl -X POST -H "Content-Type: application/json" --data '{
"jsonrpc": "2.0",
"method": "Filecoin.ChainValidateIndex",
"params": [1995103, true],
"id": 1
}' http://localhost:1234/rpc/v1 | jq .
{
"id": 1,
"jsonrpc": "2.0",
"result": {
"TipSetKey": [
{
"/": "bafy2bzacebvzbpbdwxsclwyorlzclv6cbsvcbtq34sajow2sn7mnksy3wehew"
},
{
"/": "bafy2bzacedgei4ve3spkfp3oou5oajwd5cogn7lljsuvoj644fgj3gv7luamu"
},
{
"/": "bafy2bzacebbpcnjoi46obpaheylyxfy5y2lrtdsyglqw3hx2qg64quip5u76s"
}
],
"Height": 1995103,
"IndexedMessagesCount": 0,
"IndexedEventsCount": 0,
"Backfilled": true,
"IsNullRound": false
}
}