Getting Started
QueryIndex Supply is a hosted HTTP API for running SQL queries on Ethereum Event Logs.
You can use this API from your backend or from your user's browser.
Here is an example query
curl -G https://api.indexsupply.net/query \
--data-urlencode 'chain=8453' \
--data-urlencode 'query=select "from", "to", tokens from transfer limit 1' \
--data-urlencode 'event_signatures=Transfer(address indexed from, address indexed to, uint tokens)' \
| jq .
And the response
{
"block_height": 18479546,
"result": [
[
[
"from",
"to",
"tokens"
],
[
"0x0000000000000000000000000000000000000000",
"0xdaabdaac8073a7dabdc96f6909e8476ab4001b34",
"0"
]
]
]
}
Chains
Here are the currently supported chains.
Name | Id |
---|---|
Arbitrum One | 42161 |
Base | 8453 |
Base Sepolia | 84532 |
Forma | 984122 |
Gnosis | 100 |
Main | 1 |
Odyssey | 911867 |
Polygon Amoy | 80002 |
World Chain | 480 |
World Chain Sepolia | 481 |
Zora | 7777777 |
For POST
requests, use the Chain: 8453
header. For GET
requests use the ?chain=8453
query param.
Email support@indexsupply.com to request new chains.
Queries
The API offers 3 basic types of queries: Single, Batch, and Live.
Single
A single query is executed against the latest state of Index Supply's indexer.
Batch
Batch queries are useful when you have several queries that you would like to run within the same database transaction. This allows you to get a consistent view of the chain across multiple queries. The response includes the block height at which the queries were run. You can use this value to start your live queries.
Live
A live query takes a query and a block height and will send HTTP SSE events as new events matching the query are indexed.
It is common to do a single (or batch) query on page load and then subscribe to updates once the page state has been initialized.
Reorgs
In the case of a chain reorg, clients will receive a block height that is lower than previously received. Clients should keep a single value pointer to the latest block height and when a new block is lower, the client should discard the entire state and start over. This should be fast, bug free, and consistent with crash-only software design principles!
Response
Regardless of the kind of query (ie get, post, single, batch) there is a single response object. The response is JSON and includes the block height at which the query(s) were executed and a 3-dimensional array.
{
block_height: {},
result: [[[]]]
}
The first, outer array dimension relates to the number of queries submitted (one for single and many for batch). In the case of GET /query
and GET /query-live
this outer array will always have length=1
(meaning the result is: result[0]
).
The second array dimension represents the number of rows returned from the query. This array can be empty (length=0
) in the case that the query returned no rows.
If a query did return a set of rows, then the second array will always contain at least 2 items. The first is an array of the column names and the rest are arrays of column values.
The third array dimension represents column values. In the case of the column names, this will be an array of strings. In the case of column values, it will be an array with the following types:
ABI Type | JSON Type |
---|---|
bool | bool |
bytesN | hexadecimal string |
string | string |
intN | decimal string |
uintN | decimal string |
Arrays of these types will be a JSON array of the type.
All inner (3rd dimension) arrays will have the same length.
GET /query
Request
GET /query?chain={}&query={}&event_signatures={}
Query Parameters
chain
. See chains for possible values.query
. A SQL query referencing tables and columns from theevent_signatures
. See SQL for more details on the query language.event_signatures
. A single human readable event signature
GET /query-live
The request parameters for /query-live
is identical to GET /query
.
The response is a standard response object but delivered via HTTP SSE. The SSE protocol will keep the connection open indefinitely and each new block will trigger a new event. Events are plain text, prefixed with data:
and separated by a \n\n
.
Request
GET /query-live?chain={}&query={}&event_signatures={}
POST /query
Request
POST /query
Chain: 84532
[
{
"query": "select a from foo",
"event_signatures": ["Foo(uint a)"]
}
]
Multiple requests can be batched by adding objects to the request array
POST /query
Chain: 84532
[
{
"query": "select a from foo",
"event_signatures": ["Foo(uint a)"]
},
{
"query": "select a from bar",
"event_signatures": ["Bar(uint a)"]
}
]
Here is a curl example of a batch request
curl -X POST https://api.indexsupply.net/query \
-H "Content-Type: application/json" \
-H "Chain: 8453" \
-d '[
{
"event_signatures": ["Transfer(address indexed from, address indexed to, uint256 value)"],
"query": "select tx_hash from transfer limit 1"
},
{
"event_signatures": ["Transfer(address indexed from, address indexed to, uint256 indexed value)"],
"query": "select tx_hash from transfer limit 1"
}
]'
And the response
{
"block_height": 22803982,
"result": [
[
[
"tx_hash"
],
[
"0x8bd891398f3745cc8a6b90f0df78a41cdfcaf2745534da6a6034d198841026f4"
]
],
[
[
"tx_hash"
],
[
"0x8bd891398f3745cc8a6b90f0df78a41cdfcaf2745534da6a6034d198841026f4"
]
]
]
}
SQL
When you provide an event signature Foo(uint indexed bar, uint baz)
you effectively have a table named foo
with a numeric columns named bar
and baz
that you can query:
select baz from foo where bar = 1
EVM Columns
In addition to event data, there are other EVM columns available:
Column | Type | Description |
---|---|---|
address | bytea | contract address emitting the event |
block_num | numeric | |
log_idx | numeric | |
tx_hash | bytea |
These can be used with the event data. For example:
select block_num, log_idx, baz
from foo
where address = 0x0000000000000000000000000000000000000000
and bar = 1
SQL Details
Index Supply supports a subset of the Postgres SQL language. Here is a brief overview of the supported syntax:
SELECT select_list
FROM from_item
WHERE condition
GROUP BY grouping_column_reference [, …]
HAVING group_condition
LIMIT count
OFFSET start
where select_list is one of: * | [[expression [AS output_name]], …]
* Project all column references for all from_items
[[expression [AS output_name]], …]
Comma-separated list of value expressions. Value expression
can be one of: column reference or aggregate function
If a non-column-reference value expression is used in the select list,
it conceptually adds a new virtual column to the returned table.
The value expression is evaluated once for each result row, with the
row's values substituted for any column references.
If a column name reference is used and if more than one table
has a column of the same name, the column name reference must
be qualified with the table name. Eg table.column
where aggregate function can be one of:
sum(), count(), avg(), min(), max()
where from_item can be one of:
[[event_name [AS name]], …]
[
[event_name [AS name]]
join_type from_item
ON join_condition
]
where event_name is an ascii string representing the
name of the Ethereum Event. For example: "transfer" for
`Transfer(address indexed from, address indexed to, uint tokens)`
If more than one table is specified then the tables are CROSS JOIN-ed.
A WHERE clause can be used to reduce the number of
returned rows in the CROSS JOIN.
where join_type is: { [INNER] | { LEFT | RIGHT | FULL } [OUTER] }
where join_condition is a boolean expression
where group_column_reference is a column name reference
group_condition filters group rows created by GROUP BY
Expressions may include scalar sub-queries or table expression
sub-queries when combined with the EXISTS, NOT EXISTS,
IN, and NOT IN operators. Other operators include:
^
*
/
%
+
-
<
>
=
<=
>=
<>
IS
IS NULL
IS NOT NULL
NOT
AND
OR