Install
This page covers installing a Shinzo Indexer with Docker or from source. To complete an indexer setup, you must also register it with the Shinzo Network (see Registration).
Hardware recommendations
| Component | Minimum | Recommended |
|---|---|---|
| CPU | 8 vCPUs | 16 vCPUs |
| Memory (RAM) | 16 GB | 32–64 GB |
| Storage | 3 TB NVMe | 4+ TB NVMe |
| OS | Ubuntu 24.04 | Ubuntu 26.04 |
Using Docker
These steps use Docker to run the Shinzo Indexer. To build the indexer from source, see Building from source below.
Prerequisites
- Docker.
- Access to an Ethereum execution node that exposes JSON-RPC and WebSocket. The indexer does not run a node for you, it just reads from one. This can be a node you run yourself, a node co-located with your validator, or a managed provider.
- A browser wallet setup. This wallet does not need to hold any funds.
Steps
-
Pull the pre-built indexer image from the Shinzo container registry.
docker pull ghcr.io/shinzonetwork/shinzo-indexer-client:standardoutputstandard: Pulling from shinzonetwork/shinzo-indexer-client 2521f1b70bf8: Pull complete 2c845527b24c: Pull complete [...] Digest: sha256:a272b09607e6f3f07399d72d019f058919ba2854469835b80478fd75799fa0fd Status: Downloaded newer image for ghcr.io/shinzonetwork/shinzo-indexer-client:standard -
Gather your Geth node's:
- RPC URL
- WebSocket URL
- API key (if set)
-
Start the indexer by filling in your details and running:
docker run --rm \ -e GETH_RPC_URL={{ YOUR RPC URL }}\ -e GETH_WS_URL={{ YOUR WEBSOCKET URL }}\ -e GETH_API_KEY={{ YOUR API KEY (OPTIONAL) }} \ -e INDEXER_START_HEIGHT=0 \ -e DEFRADB_KEYRING_SECRET=devnet-secret \ -e DEFRADB_PLAYGROUND=true \ -e DEFRADB_P2P_ENABLED=true \ -e DEFRADB_P2P_LISTEN_ADDR=/ip4/0.0.0.0/tcp/9171 \ -e LOGGER_DEBUG=true \ -p 9181:9181 \ -p 9171:9171 \ -p 8080:8080 \ ghcr.io/shinzonetwork/shinzo-indexer-client:standard
You should see the indexer connect to Geth and start collecting and committing blocks:
2026-05-11T10:59:54.762Z INFO Committed block 25071330 (ID: bae-235bbc36-32ff-5fb0-8361-6c4dc3d6aeb9)
2026-05-11T10:59:54.902Z DEBUG HTTP response: 200 OK (Content-Length: )
2026-05-11T10:59:54.902Z DEBUG HTTP request successful, status: 200 OK
2026-05-11T10:59:55.272Z DEBUG HTTP Request: POST http://35.193.228.182:8080
2026-05-11T10:59:55.272Z DEBUG Setting x-goog-api-key header: df7f****e6db
2026-05-11T10:59:55.272Z DEBUG Request headers: Content-Type=application/json, User-Agent=
2026-05-11T10:59:55.409Z DEBUG HTTP response: 200 OK (Content-Length: )
2026-05-11T10:59:55.409Z DEBUG HTTP request successful, status: 200 OK
Eventually your indexer will catch up with the validator node and start waiting for new blocks rather than pulling historical data:
2026-05-11T11:05:09.338Z DEBUG HTTP response: 200 OK (Content-Length: )
2026-05-11T11:05:09.338Z DEBUG HTTP request successful, status: 200 OK
2026-05-11T11:05:09.338Z INFO Block 25071451 not available yet, waiting...
Registration
Once the indexer is running, register it with the Shinzo Network. See Registration for details.
Building from source
You can also build the indexer binary from source instead of using Docker.
Prerequisites
- Go 1.26 or later.
- Git.
- Access to an Ethereum execution node (same as the Docker install method).
Steps
-
Clone the repository and install the Go dependencies.
git clone https://github.com/shinzonetwork/shinzo-indexer-client.git cd shinzo-indexer-client go mod download -
Create a
.envfile with your node details and indexer settings.cat > .env << EOF GETH_RPC_URL=<your-rpc-url> GETH_WS_URL=<your-ws-url> GETH_API_KEY=<your-api-key> DEFRADB_KEYRING_SECRET=<your-keyring-secret> DEFRADB_PLAYGROUND=true DEFRADB_P2P_ENABLED=true DEFRADB_P2P_LISTEN_ADDR=/ip4/0.0.0.0/tcp/9171 INDEXER_START_HEIGHT=0 LOGGER_DEBUG=true EOF -
Build the binary.
make build -
Run the indexer.
make start
The included config.yaml works for most local development. You typically only need to change peer settings or storage paths for advanced setups. Environment variables in .env override values in config.yaml.
Registration
Once your indexer is running, register it with the Shinzo Network. See Registration for details.
Do you need an API key?
It depends on where your Geth node is.
If the indexer and the Geth node are on the same private network (both on VMs in the same VPC, for example) you probably don't need one. Geth has no authentication by default. Leave GETH_API_KEY empty and point GETH_RPC_URL at the node's internal IP or hostname.
If you are connecting to an externally hosted node, authentication is almost always required. Two common cases:
- GCP Blockchain Node Engine (
blockchainnodeengine.com) expects the API key in theX-goog-api-keyheader. - A self-hosted node behind a reverse proxy (e.g. nginx) uses whatever header the operator configures.
X-Api-Keyis common.
The indexer picks the right header automatically based on the URL.
Exposed ports
The following ports must be exposed and available on the machine.
| Port | Service |
|---|---|
8080 | Health endpoint (/health), metrics (/metrics), and registration ('/registration'). |
9171 | DefraDB P2P. |
9181 | DefraDB GraphQL API. |
Troubleshooting
Permission denied on .defra/keys
The data directories are owned by root but the container runs as UID 1001. Stop the container, fix the ownership, then start again:
docker-compose -f ~/docker-compose.yml stop
chown -R 1001:1001 ~/data/defradb ~/data/lens
docker-compose -f ~/docker-compose.yml start
Failed to load existing DefraDB identity
DEFRADB_KEYRING_SECRET has changed since the first run. Restore the original value in your compose file and restart.
WebSocket unavailable, will use HTTP-only mode
The indexer falls back to HTTP polling. Check that GETH_WS_URL is correct and the port is reachable. HTTP-only mode works but is slightly slower.