Using Subgraphs
RPC endpoints are only able to return a single piece of information at a time, such as the name of a single lock, or the expiration date of a single key... etc.
Retrieving multiple attributes of an object or a list of objects is generally costly and not recommended. For that, subgraphs are a better solution. The way to think about Subgraphs is as "views" of the data stored on chains that is organized and indexed using different dimensions. They use the popular GraphQL API query language to make it trivial to quickly retrieve large amounts of data.
In this tutorial, we will write a function that uses Unlock's subgraphs to retrieve a list of locks managed by a given address. We actually use a similar approach to load a user's lock on our dashboard.
Building the request​
The first step is to build the request itself. For this, The Graph offers a convenient Playground tool. Let's look at an example with our xDAI subgraph (all networks share the same schema).
In the left column, you can see a query builder that you can use to customize your queries. The right column provides a convenient way to inspect the schema and auto-complete your queries.
In order to get the locks managed by a given address, we build the following request:
{
locks(orderBy: createdAtBlock, orderDirection: desc, where:{
lockManagers_contains:
[ "0xDD8e2548da5A992A63aE5520C6bC92c37a2Bcc44" ]
}) {
id
address
name
lockManagers
price
tokenAddress
totalSupply
expirationDuration
maxNumberOfKeys
version
}
}
We retrieve the locks sorted by the creation block in a descending order (first in the list is the most recent) where the lockManagers
field contains our given address. For each lock, we get their address, name, price, token (if it is an ERC20 lock), duration, as well as total supply and maximum number of memberships.
Sending the request​
There are exists multiple GraphQL libraries in many languages. Here we will focus on the most basic approach: using JavaScript's fetch function that is available in any web browser environment, but also in node.js using node-fetch
.
// We build the query, with a variable $owner
const query = `query Locks($owner: String!) {
lock(orderBy: createdAtBlock, orderDirection: desc,
where:{
lockManagers_contains:
[ $owner ]
})
{
id
address
name
lockManagers
price
tokenAddress
expirationDuration
version
}
}
}`;
// We set the variable's value to the the user's address
const variables = { owner: userAddress };
// We send a POST request to the subgraph endpoint of our choice (change if using a different network!)
// The body of the request must include a stringified version of and object built with the query and variables
const result = await fetch(
"https://api.thegraph.com/subgraphs/name/unlock-protocol/unlock",
{
method: "POST",
headers: {
"Content-Type": "application/json",
Accept: "application/json",
},
body: JSON.stringify({
query,
variables,
}),
}
).then((r) => r.json());
The result
variable will be populated with the result of the query. It will be an array, with all of the information that was retrieved, and matching the query's format. More specifically, here it includes the following: a data
object that includes a single property: lockManagers
and a list of matching locks!
{
"data": {
"locks": [
{
"id": "0x3ba39a6185ac5d35927e6166255bebee9f61112d",
"address": "0x3ba39a6185ac5d35927e6166255bebee9f61112d",
"name": "First Goerli Lock",
"lockManagers": ["0xdd8e2548da5a992a63ae5520c6bc92c37a2bcc44"],
"price": "70000000000000000",
"tokenAddress": "0x0000000000000000000000000000000000000000",
"expirationDuration": "115792089237316195423570985008687907853269984665640564039457584007913129639935",
"version": "10"
},
{
"id": "0x874161a65ab3341b958c815cc933b9868ac4790e",
"address": "0x874161a65ab3341b958c815cc933b9868ac4790e",
"name": "Second Goerli lock",
"lockManagers": ["0xdd8e2548da5a992a63ae5520c6bc92c37a2bcc44"],
"price": "10000000000000000",
"tokenAddress": "0x0000000000000000000000000000000000000000",
"expirationDuration": "2592000",
"version": "10"
}
]
}
}
Next steps​
GraphQL lets developers build complex queries to retrieve data from the blockchain that can be used to populate fields in your web3 application!