Skip to main content

Active Liquidity

Introduction

This guide will cover how to fetch and compute the active liquidity in the specific Tick ranges of a pool. It is based on the Liquidity Density example and can be seen used in production, albeit in a more sophisticated way, in the Uniswap Analytics website.

info

If you need a briefer on the SDK and to learn more about how these guides connect to the examples repository, please visit our background page!

In this guide, we will use the V3 subgraph to fetch all ticks from theGraph and compute the active liquidity our Pool can use at each Tick. We then use recharts to draw a chart that visualizes our Pool's liqudity density.

This guide will cover:

  1. Getting the tickSpacing and currently active Tick from the Pool
  2. Calculating active liquidity from net liquidity
  3. Drawing a chart from the Tick data

This guide will not cover:

  • Specifics of working with the recharts library. You can read more about that here.

At the end of the guide, we should be able to visualize the liquidity of any V3 Pool.

Understanding Active Liquidity

To visualize the distribution of active liquidity in our Pool, we want to draw our Chart around the currently active Tick. For that we have to first understand:

  • What is an initialized Tick?
  • What is the current Tick?

Initialized Ticks

When providing liquidity for a pool, the LP decides the price range in which the liquidity should be provided, and the amount of liquidity to be provided. The pool understands the position as liquidity between the lower and upper Tick. The Tick Index in this context is a representation of the price between the Pool's assets.

Looking at this visualization of multiple positions in a V3 Pool, we can see that the liquidity available for a swap does not change inside a position, but when crossing into the next position. This is what the Initialized Ticks of a Pool represent - they are a representation of the start or end of one or more positions.

LiquidityNet1

When entering or leaving a position, its liquidity is added or removed from the active liquidity available for a Swap. The initialized Ticks store this change in available liquidity in the liquidityNet field. The change is always stored in relation to the currently active Tick - the current price. When the price crosses an initialized Tick, it gets updated and liqudity that was previously added when crossing the Tick would now be removed and vice versa.

The liquidityGross value represents the gross value of liquidity referencing the tick. This is important for the edge case that one position ends at a Tick and a second position with exactly the same liquidity value would start at the Tick. In this case liquidityNet would be 0 but liquidityGross would still have a value, which ensures that the Tick is not deleted from the Pool.

To visualize liquidity in a graph, we will only need to consider the changes, so it's sufficient to fetch the Ticks with liquidityNet not 0.

Fetching initialized Ticks

To fetch all ticks of our Pool, we will use the Uniswap V3 graph. To visualize active liquidity, we need the tickIdx, the liquidityGross and the liquidityNet.

We define our GraphQL query and send a POST request to the V3 subgraph API endpoint:

axios.post(
"https://api.thegraph.com/subgraphs/name/uniswap/uniswap-v3",
{"query": `{ ticks(
where: {poolAddress: "${poolAddress.toLowerCase()}", liquidityNet_not: "0"}
first: 1000,
skip: ${skip},
orderBy: tickIdx,
orderDirection: asc
) {
tickIdx
liquidityGross
liquidityNet
}
}`
},
{
headers: {
"Content-Type": "application/json"
}
}
)

We only fetch the ticks that have liquidity, and we convert the poolAddress to lower case for the subgraph to work with. To make sure the Ticks are ordered correctly, we also define the order direction in the query.

note

GraphQL is only able to fetch 1000 records at a time. If a pool has more than 1000 initialized ticks, multiple calls are necessary to get all of them.

The ticks we got from theGraph have this format:

interface GraphTick {
tickIdx: string
liquidityGross: string
liquidityNet: string
}

Current Tick

The current Tick of the Pool represents the current Price after the last swap. Considering that the initialized Ticks only represent positions, we see that it is not necessarily one of the initialized Ticks but can be at any point in between them. The active liqudity at the current Price is also stored in the smart contract - we already fetched it with the liquidity function in the previous guide.

Tickspacing

Only the Ticks with indices that are divisible with 0 remainder by the tickspacing of a Pool are initializable. This is a convention defined by the protocol to save gas. The Tickspacing of the Pool is dependent on the Fee Tier. Pools with lower fees are meant to be used for more stable Token Pairs and allow for more granularity in where LPs position their liquidity.

We can get the tickSpacing from the TICK_SPACINGS enum exposed by the v3-sdk:

import { TICK_SPACINGS }

const tickSpacing = TICK_SPACINGS[fee]

Alternatively, if we have already constructed a Pool object, we could just call Pool.tickSpacing().

Putting it all together

For the purpose of visualizing the liquidity density of the Pool, it rarely makes sense to display the full Tick Range of the Pool, as the vast majority of liquidity will be focused in a narrow price range.

Instead, we will display a sensible number of Ticks around the current price.

Calculating active liquidity

We know the spacing between Ticks and the Initialized Ticks where active liquidity changes. All we have to do is start calculating from the current Tick and iterate outwards.

The code mentioned in the following snippets can be found in active-liquidity.ts.

To draw our chart we want a data structure that looks something like this:

interface TickProcessed {
tickIdx: number,
liquidityActive: JSBI,
liquidityNet: JSBI,
price0: string,
price1: string,
isCurrent: boolean
}

To access the initialized Ticks directly from their Tick Index, we store them in a Record:

const tickIdxToTickDictionary: Record<string, GraphTick> = Object.fromEntries(
ticks.map((graphTick) => [graphTick.tickIdx, graphTick])
)

The ticks variable in this code snippet is the result we got from the V3 Subgraph earlier.

We want to mark the Tick closest to the current Price and we want to be able to display the prices at a Tick to the user. We calculate the initializable Tick closest to the current price and create the active Tick that we start from:

import { tickToPrice } from '@uniswap/v3-sdk'

const activeTickIdx = Math.floor(pool.tickCurrent / tickSpacing) * tickSpacing

const activeTickProcessed: TickProcessed = {
tickIdx: activeTickIdx,
liquidityActive: pool.liquidity,
liquidityNet: JSBI.BigInt(0),
price0: tickToPrice(tokenA, tokenB, activeTickIdx).toFixed(6),
price1: tickToPrice(tokenB, tokenA, activeTickIdx).toFixed(6),
isCurrent: true
}

Here we also calculate the price of the tokens from the tickIdx, the v3-sdk exports a handy utility function for that, tickToPrice. We store the Price as a string as we won't make any further calculations in this example. We will instead use it to display prices in the tooltip of our chart. Notice how the price0 is the Price of tokenA in terms of tokenB and the price1 is the Price of tokenB in terms of tokenA at the specified Tick.

If the current Tick is initialized, we also need to set the liquidityNet to correctly handle moving out of the position:

const currentTickInitialized = tickIdxToTickDictionary[activeTickIdx]
if (currentTickInitialized !== undefined) {
activeTickProcessed.liquidityNet = JSBI.BigInt(currentTickInitialized.liquidityNet)
}

We now start iterating outwards from the active Tick and compute the active liquidity for each Tick we want to display. The processed Tick is then saved in an Array of TickProcessed. We choose an arbitrary number of Ticks we want to display, for this example we calculate 100 Ticks in each direction.

import { TickMath, tickToPrice } from '@uniswap/v3-sdk'

let previousTickProcessed = {
...activeTickProcessed
}

processedTicks: TickProcessed[] = []

for (let i = 0; i < 100; i++) {
const currentTickIdx = previousTickProcessed.tickIdx + tickSpacing

if (currentTickIdx > TickMath.MAX_TICK) {
break
}

const currentTickProcessed = {
liquidityActive: previousTickProcessed.liquidityActive,
tickIdx: currentTickIdx,
liquidityNet: JSBI.BigInt(0),
price0: tickToPrice(token0, token1, currentTickIdx),
price1: tickToPrice(token1, token0, currentTickIdx),
isActive: false
}

...
}

We calculate one Tick at a time, and we need to make sure our Tick stays inside the possible price range by checking against TickMath.MAX_TICK. Again, we check if our current Tick is initialized and if so, recalculate the active liquidity:

for (let i = 0; i < 100; i++) {

...

const currentTickInitialized = tickIdxToTickDictionary[currentTickIdx]

if (currentTickInitialized !== undefined) {
currentTickProcessed.liquidityNet = JSBI.BigInt(currentTickInitialized.liquidityNet)
currentTickProcessed.liquidityActive = JSBI.add(
previousTickProcessed.liquidityActive,
JSBI.BigInt(currentTickInitialized.liquidityNet)
)
}

processedTicks.push(currentTickProcessed)
previousTickProcessed = currentTickProcessed
}

After we are done calculating the next 100 Ticks after the current Tick, we iterate in the opposite direction for the previous Ticks. Iterating downwards, we need to subtract the net liquidity where we added it when iterating upwards. You can find a full code example in the Uniswap Example repository.

We are finally able to combine the previous, active and subsequent Ticks:

const allProcessedTicks = previousTicks.concat(activeTickProcessed).concat(subsequentTicks)

Drawing the Chart

We are done with our calculations and move on to displaying the data. Recharts is not able to handle JSBI, so we need to convert the Array we created to a format it can handle:

const chartTicks: TicksChart[] = allProcessedTicks.map((tickProcessed) => {
return {...processedTick, liquidityActiveChart: parseFloat(tickProcessed.liquidityActive.toString())}
})

The loss of precision will not be visually noticeable in the chart and we are still able to display the exact number in a Tooltip if we wish to. Liquidity is stored in a uint128 format onchain, so the maximum loss of precision will be far smaller than the number of decimals of almost any ERC20 Token.

Finally, we draw the Chart:

<ResponsiveContainer width="80%" height={400}>
<BarChart>
<XAxis />
<YAxis />
<Bar dataKey="liquidityActiveChart" fill="#2172E5">
{chartTicks.map((entry, index) => {
return (
<Cell
key={`cell-${index}`}
fill={entry.isActive ? '#F51E87' : '#2172E5'}
/>
)
})}
</Bar>
</BarChart>
</ResponsiveContainer>

In a real application, you will probably want to format the chart properly and display additional information for users. Check out the full code example to this guide and the official recharts documentation.

You can also take a look at the Uniswap Info repository to see a similar chart used in production.

Locked Liquidity

If you run the example, you will notice that the chart also displays a custom tooltip with additional information that we didn't touch on in this example. The total locked liqudity in the tooltip represents the sum of positions in the currency locked at the selected Tick. It is calculated as the maximum token output of a swap when crossing to the next Tick. The V3 pool here is initialized with only the liquidity of the current Tick.

Depending on your use case, it may make sense to display this value. You can find the full code in the code example.

Next Steps

Now that you are familiar with liquidity data, consider checking out our next guide on using Uniswap as a Price Oracle.