PIVOT keyword

PIVOT allows you to pivot rows into a columns. This can be useful when you want to ingest narrow-schema data, and then pivot it into a wide-schema.

This syntax is supported within SELECT queries.

Syntax

Flow chart showing the syntax of the PIVOT keyword

Mechanics

The PIVOT keyword comes after a general table select.

There are two components:

Aggregate Columns

These columns appear immediately after the PIVOT keyword. These are aggregates that will be calculated for each of the Pivot columns. These are the values that will be placed in the output columns.

Pivot Columns

These columns appear after the FOR keyword, and define the filtering and final column names. The aggregate functions will be run for each of these.

Single aggregate and pivot

basic PIVOTDemo this query
(trades LIMIT 1000) 
PIVOT (
avg(price)
FOR symbol IN ('BTC-USD', 'ETH-USD')
);
BTC-USDETH-USD
39282.2007365439062616.588454404948

This query calculates an average price based on filtering rows that contain the symbols defined in the queries.

In short, this shifts the symbol names into the column position, and fills it with the corresponding aggregate value.

An equivalent non-pivot query might look like this:

SELECT
avg(CASE WHEN symbol = 'BTC-USD' THEN price END) AS 'BTC-USD',
avg(CASE WHEN symbol = 'ETH-USD' THEN price END) AS 'ETH-USD'
FROM trades

Multiple aggregates, single pivot

multiple aggregatesDemo this query
(trades LIMIT 1000) 
PIVOT (
avg(price),
count(price)
FOR symbol IN ('BTC-USD', 'ETH-USD')
);
BTC-USD_avgBTC-USD_countETH-USD_avgETH-USD_count
39282.2007365439063532616.588454404948647

In this case, the aggregate functions are applied to each of the filtered symbols, so the final output has 2×2=42 \times 2 = 4 columns.

Single aggregate, multiple pivots

multiple pivotsDemo this query
(trades LIMIT 1000) 
PIVOT (
avg(price)
FOR symbol IN ('BTC-USD', 'ETH-USD')
side IN ('buy', 'sell')
);
BTC-USD_buyBTC-USD_sellETH-USD_buyETH-USD_sell
39286.99746113989439276.414687500032616.8504132231392616.253626760561

In this case, the aggregate function is applied to each of the symbols, combinatorially.

Therefore, the output dataset is 1x(2x2)=41 x (2 x 2) = 4 columns.

Multiple aggregates, multiple pivots

multiple aggregates and pivotsDemo this query
(trades LIMIT 1000)
PIVOT (
avg(price),
count(price)
FOR symbol IN ('BTC-USD', 'ETH-USD')
side IN ('buy', 'sell')
);
BTC-USD_buy_avgBTC-USD_buy_countBTC-USD_sell_avgBTC-USD_sell_countETH-USD_buy_avgETH-USD_buy_countETH-USD_sell_avgETH-USD_sell_count
39286.99746113989419339276.414687500031602616.8504132231393632616.253626760561284

Each of the aggregates is applied to each combination of pivot columns. Therefore, the output column count is 2x2x2=82 x 2 x 2 = 8.

Aliasing aggregate columns

If you wish to control the column output name, or need to override it to avoid duplicate issues, you can set an alias.

aggregate with aliasDemo this query
(trades LIMIT 1000)
PIVOT (
avg(price) as average_price
FOR symbol IN ('BTC-USD', 'ETH-USD')
);
BTC-USD_average_priceETH-USD_average_price
39282.2007365439062616.588454404948

With GROUP BY

You can add an explicit group by to the PIVOT clause to modify the output result set.

Consider this basic case, where we are just taking an average price:

pivot without explicit group byDemo this query
(trades LIMIT 1000) 
PIVOT (
avg(price)
FOR symbol IN ('BTC-USD')
);
BTC-USD
39282.200736543906

Perhaps we actually want to run this for both buy and sell sides? In earlier examples, we demonstrated how you can do this with multiple output columns:

multiple pivots without explicit group byDemo this query
(trades LIMIT 1000) 
PIVOT (
avg(price)
FOR symbol IN ('BTC-USD')
side IN ('buy', 'sell')
);
BTC-USD_buyBTC-USD_sell
39286.99746113989439276.41468750003

But perhaps we'd rather just have a side and BTC-USD column, with two rows in the output?

pivot with explicit group byDemo this query
(trades LIMIT 1000) 
PIVOT (
avg(price)
FOR symbol IN ('BTC-USD')
GROUP BY side
);
sideBTC-USD
buy39286.997461139894
sell39276.41468750003

You can imagine that the above query is equivalent to:

above without using pivotDemo this query
SELECT side,
avg(price) as 'BTC-USD'
FROM (trades LIMIT 1000)
WHERE symbol = 'BTC-USD'
GROUP BY side, symbol

This then scales up as you add more clauses to the PIVOT:

explicit group by and multiple clausesDemo this query
(trades LIMIT 1000) 
PIVOT (
avg(price),
count(price)
FOR symbol IN ('BTC-USD', 'ETH-USD')
GROUP BY side
);
sideBTC-USD_avgBTC-USD_countETH-USD_avgETH-USD_count
sell39276.414687500031602616.253626760561284
buy39286.9974611398941932616.850413223139363

With ORDER BY

We can add an ORDER BY clause to sort the final result set by a column. For example, if we wanted to guarantee the ordering to be the buy row, then sell:

explicit group by and order byDemo this query
(trades LIMIT 1000) 
PIVOT (
avg(price),
count(price)
FOR symbol IN ('BTC-USD', 'ETH-USD')
GROUP BY side
ORDER BY side
);
sideBTC-USD_avgBTC-USD_countETH-USD_avgETH-USD_count
buy39286.9974611398941932616.850413223139363
sell39276.414687500031602616.253626760561284

With LIMIT

Additionally, you can tag a LIMIT on the query. So we could take the above result set and select just the first row.

explicit group by and order by and limitDemo this query
(trades LIMIT 1000) 
PIVOT (
avg(price),
count(price)
FOR symbol IN ('BTC-USD', 'ETH-USD')
GROUP BY side
ORDER BY side
LIMIT 1
);
sideBTC-USD_avgBTC-USD_countETH-USD_avgETH-USD_count
buy39286.9974611398941932616.850413223139363