A real-time blockchain indexer for the Penumbra network that collects raw block data and provides an API for querying the latest blocks.
This application connects to a Penumbra node, continuously synchronizes the latest blocks, stores them in a PostgreSQL database, and provides a REST API for querying the data.
-
Block Synchronizer
- Connects to Penumbra RPC node
- Fetches latest blocks in real-time
- Handles reconnection and error recovery
- Manages block data validation
-
Database Layer
- PostgreSQL database for block storage
- Maintains latest 10 blocks
- Automatic cleanup of old blocks
- Optimized queries for block retrieval
-
API Server
- RESTful endpoints for data access
- JSON response format
- CORS support for frontend integration
- Error handling and status codes
- Language: Rust
- Database: PostgreSQL 15
- Dependencies:
tokio
- Async runtimeaxum
- Web frameworksqlx
- Database ORMreqwest
- HTTP clientserde
- Serializationchrono
- DateTime handling
Initially, we planned to use gRPC for communication with the Penumbra network. However, we encountered several issues:
- Dependency Errors: Multiple compilation issues with dependencies, even when following the official documentation.
- Buf Compatibility: Despite using
buf
for protobuf compilation, errors persisted, making development slow and unreliable. - Build Complexity: The additional complexity of managing gRPC dependencies and ensuring cross-platform compatibility introduced unnecessary overhead.
- Simplicity & Maintainability: By using RESTful APIs, we reduce the dependency footprint and make it easier for new developers to contribute to the project without dealing with gRPC-specific configurations.
For these reasons, we opted for a REST API with WebSockets as a future improvement for real-time updates.
CREATE TABLE blocks (
height BIGINT PRIMARY KEY,
time TIMESTAMP WITH TIME ZONE NOT NULL,
hash TEXT NOT NULL,
proposer_address TEXT NOT NULL,
tx_count INTEGER NOT NULL,
previous_block_hash TEXT,
data JSONB NOT NULL,
created_at TIMESTAMP WITH TIME ZONE NOT NULL DEFAULT CURRENT_TIMESTAMP
);
Returns the latest 10 blocks from the chain.
Response Format:
{
"blocks": [
{
"height": number,
"time": string (ISO datetime),
"hash": string,
"proposer_address": string,
"tx_count": number,
"previous_block_hash": string,
"data": object,
"created_at": string (ISO datetime)
}
],
"total_count": number
}
- Docker
- Docker Compose
DB_URL=postgres://indexer:indexer@db/indexer
RPC_URL=http://grpc.penumbra.silentvalidator.com:26657
API_PORT=3000
- Clone the repository:
git clone https://github.com/JumpiiX/penumbra-indexer
cd penumbra-indexer
- Start the application:
docker compose up --build
The API will be available at http://localhost:3000/api/blocks
-
Database Container
- PostgreSQL 15
- Persistent volume for data storage
- Health checks configured
-
Indexer Container
- Rust application
- Automatic reconnection handling
- Real-time block synchronization
src/
├── main.rs # Application entry point
├── client.rs # Penumbra RPC client
├── db/
│ └── mod.rs # Database operations
├── api/
│ ├── mod.rs # API setup
│ └── routes.rs # API endpoints
└── models/
└── mod.rs # Data structures
The application implements comprehensive error handling:
- Database connection errors
- RPC node connectivity issues
- Invalid block data
- API error responses
The application provides logging for:
- Block synchronization status
- Database operations
- API requests
- Error conditions
- Add block search functionality
- Implement transaction indexing
- Add validator statistics
- Create metrics endpoint
- Add WebSocket support for real-time updates