Blnk is optimized out of the box for most workloads. This guide is for teams processing millions of records who want to squeeze out extra performance when searching directly from the DB.
Default indexes
These fields are already indexed and optimized:
| Table | Indexed fields |
|---|
transactions | transaction_id, reference, status, currency, source, destination, parent_transaction, created_at, meta_data (GIN) |
balances | balance_id, ledger_id, identity_id, indicator, currency, created_at |
ledgers | ledger_id, created_at |
identity | identity_id, country, created_at |
accounts | account_id, ledger_id, identity_id, balance_id, created_at |
You can sort and filter efficiently on any of these indexed fields without additional configuration.
Use Typesense for large-scale queries
For complex searches, filtering across millions of records, or when you need sub-millisecond response times, use Blnk’s Typesense integration instead of database queries.
Typesense is optimized for:
- Full-text search across any field
- Faceted filtering with instant results
- Sorting large datasets without custom indexes
Adding custom indexes
If you’re not using Typesense and frequently sort or filter by a specific field at scale, add an index.
Create single column indexes
Create indexes for frequently filtered or sorted fields:-- Sort transactions by amount
CREATE INDEX CONCURRENTLY IF NOT EXISTS idx_transactions_amount
ON blnk.transactions (amount);
-- Sort balances by current balance
CREATE INDEX CONCURRENTLY IF NOT EXISTS idx_balances_balance
ON blnk.balances (balance);
Create composite indexes
For queries that filter and sort on multiple fields, composite indexes are more efficient:-- Filter + sort combo (most efficient for this pattern)
CREATE INDEX CONCURRENTLY IF NOT EXISTS idx_transactions_status_amount
ON blnk.transactions (status, amount);
Create metadata indexes
If you frequently filter on a specific metadata key, index that key directly:-- Index a specific metadata key
CREATE INDEX CONCURRENTLY IF NOT EXISTS idx_transactions_customer_type
ON blnk.transactions ((meta_data->>'customer_type'));
The meta_data field already has a GIN index for general JSONB queries, but indexing specific keys provides better performance for frequent filters on those keys.
Use CONCURRENTLY when creating indexes on production databases to avoid locking tables. This allows the index to be built without blocking reads and writes.
Debugging slow queries
Use EXPLAIN ANALYZE to see how Postgres executes a query:
EXPLAIN ANALYZE
SELECT *
FROM blnk.transactions
WHERE status = 'APPLIED'
ORDER BY amount DESC
LIMIT 50;
What to look for:
Seq Scan on large tables: strong signal you need an index
- High actual time values: indicates slow query execution
- Large row counts being filtered down late: suggests missing indexes
QUERY PLAN
─────────────────────────────────────────────────────────────────────────────
Limit (cost=0.00..1234.56 rows=50 width=200) (actual time=0.123..45.678 rows=50 loops=1)
-> Seq Scan on transactions (cost=0.00..12345.67 rows=50000 width=200) (actual time=0.120..45.600 rows=50000 loops=1)
Filter: (status = 'APPLIED'::text)
Rows Removed by Filter: 100000
Planning Time: 0.123 ms
Execution Time: 45.789 ms
In this example, a sequential scan is being performed, which suggests adding an index on status would improve performance.
Best practices
-
Monitor query performance: Use
EXPLAIN ANALYZE regularly to identify slow queries.
-
Index strategically: Only add indexes for fields you filter or sort on frequently. Too many indexes can slow down writes.
-
Use composite indexes: When filtering and sorting on multiple fields together, create composite indexes.
-
Consider Typesense: For complex queries or very large datasets, Typesense often provides better performance than database indexes.
-
Test in production-like environments: Index performance can vary based on data distribution and query patterns.
Need help?
We are very happy to help you make the most of Blnk, regardless of whether it is your first time or you are switching from another tool.
To ask questions or discuss issues, please contact us or join our Discord community.