Optimize Flux queries

Optimize your Flux queries to reduce their memory and compute (CPU) requirements.

Start queries with pushdowns

Pushdowns are functions or function combinations that push data operations to the underlying data source rather than operating on data in memory. Start queries with pushdowns to improve query performance. Once a non-pushdown function runs, Flux pulls data into memory and runs all subsequent operations there.

Pushdown functions and function combinations

Most pushdowns are supported when querying an InfluxDB 2.2 or InfluxDB Cloud data source. As shown in the following table, a handful of pushdowns are not supported in InfluxDB 2.2.

FunctionsInfluxDB 2.2InfluxDB Cloud
count()
drop()
duplicate()
filter() *
fill()
first()
keep()
last()
max()
mean()
min()
range()
rename()
sum()
window()
Function combinations
group() |> count()
group() |> first()
group() |> last()
group() |> max()
group() |> min()
group() |> sum()
sort() |> limit()
window() |> count()
window() |> first()
window() |> last()
window() |> max()
window() |> min()
window() |> sum()

* filter() only pushes down when all parameter values are static. See Avoid processing filters inline.

Use pushdown functions and function combinations at the beginning of your query. Once a non-pushdown function runs, Flux pulls data into memory and runs all subsequent operations there.

Pushdown functions in use
  1. from(bucket: "example-bucket")
  2. |> range(start: -1h) //
  3. |> filter(fn: (r) => r.sensor == "abc123") //
  4. |> group(columns: ["_field", "host"]) // Pushed to the data source
  5. |> aggregateWindow(every: 5m, fn: max) //
  6. |> filter(fn: (r) => r._value >= 90.0) //
  7. |> top(n: 10) // Run in memory

Avoid processing filters inline

Avoid using mathematic operations or string manipulation inline to define data filters. Processing filter values inline prevents filter() from pushing its operation down to the underlying data source, so data returned by the previous function loads into memory. This often results in a significant performance hit.

For example, the following query uses dashboard variables and string concatenation to define a region to filter by. Because filter() uses string concatenation inline, it can’t push its operation to the underlying data source and loads all data returned from range() into memory.

  1. from(bucket: "example-bucket")
  2. |> range(start: -1h)
  3. |> filter(fn: (r) => r.region == v.provider + v.region)

To dynamically set filters and maintain the pushdown ability of the filter() function, use variables to define filter values outside of filter():

  1. region = v.provider + v.region
  2. from(bucket: "example-bucket")
  3. |> range(start: -1h)
  4. |> filter(fn: (r) => r.region == region)

Avoid short window durations

Windowing (grouping data based on time intervals) is commonly used to aggregate and downsample data. Increase performance by avoiding short window durations. More windows require more compute power to evaluate which window each row should be assigned to. Reasonable window durations depend on the total time range queried.

Use “heavy” functions sparingly

The following functions use more memory or CPU than others. Consider their necessity in your data processing before using them:

We’re continually optimizing Flux and this list may not represent its current state.

Use set() instead of map() when possible

set(), experimental.set(), and map can each set columns value in data, however set functions have performance advantages over map().

Use the following guidelines to determine which to use:

  • If setting a column value to a predefined, static value, use set() or experimental.set().
  • If dynamically setting a column value using existing row data, use map().

Set a column value to a static value

The following queries are functionally the same, but using set() is more performant than using map().

  1. data
  2. |> map(fn: (r) => ({ r with foo: "bar" }))
  3. // Recommended
  4. data
  5. |> set(key: "foo", value: "bar")

Dynamically set a column value using existing row data

  1. data
  2. |> map(fn: (r) => ({ r with foo: r.bar }))

Balance time range and data precision

To ensure queries are performant, balance the time range and the precision of your data. For example, if you query data stored every second and request six months worth of data, results would include ≈15.5 million points per series. Depending on the number of series returned after filter()(cardinality), this can quickly become many billions of points. Flux must store these points in memory to generate a response. Use pushdowns to optimize how many points are stored in memory.

To query data over large periods of time, create a task to downsample data, and then query the downsampled data instead.

Measure query performance with Flux profilers

Use the Flux Profiler package to measure query performance and append performance metrics to your query output. The following Flux profilers are available:

  • query: provides statistics about the execution of an entire Flux script.
  • operator: provides statistics about each operation in a query.

Import the profiler package and enable profilers with the profile.enabledProfilers option.

  1. import "profiler"
  2. option profiler.enabledProfilers = ["query", "operator"]
  3. // Query to profile

For more information about Flux profilers, see the Flux Profiler package.