Pipeline aggregations

Pipeline aggregations

Pipeline aggregations work on the outputs produced from other aggregations rather than from document sets, adding information to the output tree. There are many different types of pipeline aggregation, each computing different information from other aggregations, but these types can be broken down into two families:

Parent

A family of pipeline aggregations that is provided with the output of its parent aggregation and is able to compute new buckets or new aggregations to add to existing buckets.

Sibling

Pipeline aggregations that are provided with the output of a sibling aggregation and are able to compute a new aggregation which will be at the same level as the sibling aggregation.

Pipeline aggregations can reference the aggregations they need to perform their computation by using the buckets_path parameter to indicate the paths to the required metrics. The syntax for defining these paths can be found in the buckets_path Syntax section below.

Pipeline aggregations cannot have sub-aggregations but depending on the type it can reference another pipeline in the buckets_path allowing pipeline aggregations to be chained. For example, you can chain together two derivatives to calculate the second derivative (i.e. a derivative of a derivative).

Because pipeline aggregations only add to the output, when chaining pipeline aggregations the output of each pipeline aggregation will be included in the final output.

buckets_path Syntax

Most pipeline aggregations require another aggregation as their input. The input aggregation is defined via the buckets_path parameter, which follows a specific format:

  1. AGG_SEPARATOR = `>` ;
  2. METRIC_SEPARATOR = `.` ;
  3. AGG_NAME = <the name of the aggregation> ;
  4. METRIC = <the name of the metric (in case of multi-value metrics aggregation)> ;
  5. MULTIBUCKET_KEY = `[<KEY_NAME>]`
  6. PATH = <AGG_NAME><MULTIBUCKET_KEY>? (<AGG_SEPARATOR>, <AGG_NAME> )* ( <METRIC_SEPARATOR>, <METRIC> ) ;

For example, the path "my_bucket>my_stats.avg" will path to the avg value in the "my_stats" metric, which is contained in the "my_bucket" bucket aggregation.

Here are some more examples:

  • multi_bucket["foo"]>single_bucket>multi_metric.avg will go to the avg metric in the "multi_metric" agg under the single bucket "single_bucket" within the "foo" bucket of the "multi_bucket" multi-bucket aggregation.
  • agg1["foo"]._count will get the _count metric for the "foo" bucket in the multi-bucket aggregation "multi_bucket"

Paths are relative from the position of the pipeline aggregation; they are not absolute paths, and the path cannot go back “up” the aggregation tree. For example, this derivative is embedded inside a date_histogram and refers to a “sibling” metric "the_sum":

  1. resp = client.search(
  2. aggs={
  3. "my_date_histo": {
  4. "date_histogram": {
  5. "field": "timestamp",
  6. "calendar_interval": "day"
  7. },
  8. "aggs": {
  9. "the_sum": {
  10. "sum": {
  11. "field": "lemmings"
  12. }
  13. },
  14. "the_deriv": {
  15. "derivative": {
  16. "buckets_path": "the_sum"
  17. }
  18. }
  19. }
  20. }
  21. },
  22. )
  23. print(resp)
  1. response = client.search(
  2. body: {
  3. aggregations: {
  4. my_date_histo: {
  5. date_histogram: {
  6. field: 'timestamp',
  7. calendar_interval: 'day'
  8. },
  9. aggregations: {
  10. the_sum: {
  11. sum: {
  12. field: 'lemmings'
  13. }
  14. },
  15. the_deriv: {
  16. derivative: {
  17. buckets_path: 'the_sum'
  18. }
  19. }
  20. }
  21. }
  22. }
  23. }
  24. )
  25. puts response
  1. const response = await client.search({
  2. aggs: {
  3. my_date_histo: {
  4. date_histogram: {
  5. field: "timestamp",
  6. calendar_interval: "day",
  7. },
  8. aggs: {
  9. the_sum: {
  10. sum: {
  11. field: "lemmings",
  12. },
  13. },
  14. the_deriv: {
  15. derivative: {
  16. buckets_path: "the_sum",
  17. },
  18. },
  19. },
  20. },
  21. },
  22. });
  23. console.log(response);
  1. POST /_search
  2. {
  3. "aggs": {
  4. "my_date_histo": {
  5. "date_histogram": {
  6. "field": "timestamp",
  7. "calendar_interval": "day"
  8. },
  9. "aggs": {
  10. "the_sum": {
  11. "sum": { "field": "lemmings" }
  12. },
  13. "the_deriv": {
  14. "derivative": { "buckets_path": "the_sum" }
  15. }
  16. }
  17. }
  18. }
  19. }

The metric is called “the_sum”

The buckets_path refers to the metric via a relative path “the_sum”

buckets_path is also used for Sibling pipeline aggregations, where the aggregation is “next” to a series of buckets instead of embedded “inside” them. For example, the max_bucket aggregation uses the buckets_path to specify a metric embedded inside a sibling aggregation:

  1. resp = client.search(
  2. aggs={
  3. "sales_per_month": {
  4. "date_histogram": {
  5. "field": "date",
  6. "calendar_interval": "month"
  7. },
  8. "aggs": {
  9. "sales": {
  10. "sum": {
  11. "field": "price"
  12. }
  13. }
  14. }
  15. },
  16. "max_monthly_sales": {
  17. "max_bucket": {
  18. "buckets_path": "sales_per_month>sales"
  19. }
  20. }
  21. },
  22. )
  23. print(resp)
  1. response = client.search(
  2. body: {
  3. aggregations: {
  4. sales_per_month: {
  5. date_histogram: {
  6. field: 'date',
  7. calendar_interval: 'month'
  8. },
  9. aggregations: {
  10. sales: {
  11. sum: {
  12. field: 'price'
  13. }
  14. }
  15. }
  16. },
  17. max_monthly_sales: {
  18. max_bucket: {
  19. buckets_path: 'sales_per_month>sales'
  20. }
  21. }
  22. }
  23. }
  24. )
  25. puts response
  1. const response = await client.search({
  2. aggs: {
  3. sales_per_month: {
  4. date_histogram: {
  5. field: "date",
  6. calendar_interval: "month",
  7. },
  8. aggs: {
  9. sales: {
  10. sum: {
  11. field: "price",
  12. },
  13. },
  14. },
  15. },
  16. max_monthly_sales: {
  17. max_bucket: {
  18. buckets_path: "sales_per_month>sales",
  19. },
  20. },
  21. },
  22. });
  23. console.log(response);
  1. POST /_search
  2. {
  3. "aggs": {
  4. "sales_per_month": {
  5. "date_histogram": {
  6. "field": "date",
  7. "calendar_interval": "month"
  8. },
  9. "aggs": {
  10. "sales": {
  11. "sum": {
  12. "field": "price"
  13. }
  14. }
  15. }
  16. },
  17. "max_monthly_sales": {
  18. "max_bucket": {
  19. "buckets_path": "sales_per_month>sales"
  20. }
  21. }
  22. }
  23. }

buckets_path instructs this max_bucket aggregation that we want the maximum value of the sales aggregation in the sales_per_month date histogram.

If a Sibling pipeline agg references a multi-bucket aggregation, such as a terms agg, it also has the option to select specific keys from the multi-bucket. For example, a bucket_script could select two specific buckets (via their bucket keys) to perform the calculation:

  1. resp = client.search(
  2. aggs={
  3. "sales_per_month": {
  4. "date_histogram": {
  5. "field": "date",
  6. "calendar_interval": "month"
  7. },
  8. "aggs": {
  9. "sale_type": {
  10. "terms": {
  11. "field": "type"
  12. },
  13. "aggs": {
  14. "sales": {
  15. "sum": {
  16. "field": "price"
  17. }
  18. }
  19. }
  20. },
  21. "hat_vs_bag_ratio": {
  22. "bucket_script": {
  23. "buckets_path": {
  24. "hats": "sale_type['hat']>sales",
  25. "bags": "sale_type['bag']>sales"
  26. },
  27. "script": "params.hats / params.bags"
  28. }
  29. }
  30. }
  31. }
  32. },
  33. )
  34. print(resp)
  1. response = client.search(
  2. body: {
  3. aggregations: {
  4. sales_per_month: {
  5. date_histogram: {
  6. field: 'date',
  7. calendar_interval: 'month'
  8. },
  9. aggregations: {
  10. sale_type: {
  11. terms: {
  12. field: 'type'
  13. },
  14. aggregations: {
  15. sales: {
  16. sum: {
  17. field: 'price'
  18. }
  19. }
  20. }
  21. },
  22. hat_vs_bag_ratio: {
  23. bucket_script: {
  24. buckets_path: {
  25. hats: "sale_type['hat']>sales",
  26. bags: "sale_type['bag']>sales"
  27. },
  28. script: 'params.hats / params.bags'
  29. }
  30. }
  31. }
  32. }
  33. }
  34. }
  35. )
  36. puts response
  1. const response = await client.search({
  2. aggs: {
  3. sales_per_month: {
  4. date_histogram: {
  5. field: "date",
  6. calendar_interval: "month",
  7. },
  8. aggs: {
  9. sale_type: {
  10. terms: {
  11. field: "type",
  12. },
  13. aggs: {
  14. sales: {
  15. sum: {
  16. field: "price",
  17. },
  18. },
  19. },
  20. },
  21. hat_vs_bag_ratio: {
  22. bucket_script: {
  23. buckets_path: {
  24. hats: "sale_type['hat']>sales",
  25. bags: "sale_type['bag']>sales",
  26. },
  27. script: "params.hats / params.bags",
  28. },
  29. },
  30. },
  31. },
  32. },
  33. });
  34. console.log(response);
  1. POST /_search
  2. {
  3. "aggs": {
  4. "sales_per_month": {
  5. "date_histogram": {
  6. "field": "date",
  7. "calendar_interval": "month"
  8. },
  9. "aggs": {
  10. "sale_type": {
  11. "terms": {
  12. "field": "type"
  13. },
  14. "aggs": {
  15. "sales": {
  16. "sum": {
  17. "field": "price"
  18. }
  19. }
  20. }
  21. },
  22. "hat_vs_bag_ratio": {
  23. "bucket_script": {
  24. "buckets_path": {
  25. "hats": "sale_type['hat']>sales",
  26. "bags": "sale_type['bag']>sales"
  27. },
  28. "script": "params.hats / params.bags"
  29. }
  30. }
  31. }
  32. }
  33. }
  34. }

buckets_path selects the hats and bags buckets (via [‘hat’]/[‘bag’]`) to use in the script specifically, instead of fetching all the buckets from sale_type aggregation

Special Paths

Instead of pathing to a metric, buckets_path can use a special "_count" path. This instructs the pipeline aggregation to use the document count as its input. For example, a derivative can be calculated on the document count of each bucket, instead of a specific metric:

  1. resp = client.search(
  2. aggs={
  3. "my_date_histo": {
  4. "date_histogram": {
  5. "field": "timestamp",
  6. "calendar_interval": "day"
  7. },
  8. "aggs": {
  9. "the_deriv": {
  10. "derivative": {
  11. "buckets_path": "_count"
  12. }
  13. }
  14. }
  15. }
  16. },
  17. )
  18. print(resp)
  1. response = client.search(
  2. body: {
  3. aggregations: {
  4. my_date_histo: {
  5. date_histogram: {
  6. field: 'timestamp',
  7. calendar_interval: 'day'
  8. },
  9. aggregations: {
  10. the_deriv: {
  11. derivative: {
  12. buckets_path: '_count'
  13. }
  14. }
  15. }
  16. }
  17. }
  18. }
  19. )
  20. puts response
  1. const response = await client.search({
  2. aggs: {
  3. my_date_histo: {
  4. date_histogram: {
  5. field: "timestamp",
  6. calendar_interval: "day",
  7. },
  8. aggs: {
  9. the_deriv: {
  10. derivative: {
  11. buckets_path: "_count",
  12. },
  13. },
  14. },
  15. },
  16. },
  17. });
  18. console.log(response);
  1. POST /_search
  2. {
  3. "aggs": {
  4. "my_date_histo": {
  5. "date_histogram": {
  6. "field": "timestamp",
  7. "calendar_interval": "day"
  8. },
  9. "aggs": {
  10. "the_deriv": {
  11. "derivative": { "buckets_path": "_count" }
  12. }
  13. }
  14. }
  15. }
  16. }

By using _count instead of a metric name, we can calculate the derivative of document counts in the histogram

The buckets_path can also use "_bucket_count" and path to a multi-bucket aggregation to use the number of buckets returned by that aggregation in the pipeline aggregation instead of a metric. For example, a bucket_selector can be used here to filter out buckets which contain no buckets for an inner terms aggregation:

  1. resp = client.search(
  2. index="sales",
  3. size=0,
  4. aggs={
  5. "histo": {
  6. "date_histogram": {
  7. "field": "date",
  8. "calendar_interval": "day"
  9. },
  10. "aggs": {
  11. "categories": {
  12. "terms": {
  13. "field": "category"
  14. }
  15. },
  16. "min_bucket_selector": {
  17. "bucket_selector": {
  18. "buckets_path": {
  19. "count": "categories._bucket_count"
  20. },
  21. "script": {
  22. "source": "params.count != 0"
  23. }
  24. }
  25. }
  26. }
  27. }
  28. },
  29. )
  30. print(resp)
  1. response = client.search(
  2. index: 'sales',
  3. body: {
  4. size: 0,
  5. aggregations: {
  6. histo: {
  7. date_histogram: {
  8. field: 'date',
  9. calendar_interval: 'day'
  10. },
  11. aggregations: {
  12. categories: {
  13. terms: {
  14. field: 'category'
  15. }
  16. },
  17. min_bucket_selector: {
  18. bucket_selector: {
  19. buckets_path: {
  20. count: 'categories._bucket_count'
  21. },
  22. script: {
  23. source: 'params.count != 0'
  24. }
  25. }
  26. }
  27. }
  28. }
  29. }
  30. }
  31. )
  32. puts response
  1. const response = await client.search({
  2. index: "sales",
  3. size: 0,
  4. aggs: {
  5. histo: {
  6. date_histogram: {
  7. field: "date",
  8. calendar_interval: "day",
  9. },
  10. aggs: {
  11. categories: {
  12. terms: {
  13. field: "category",
  14. },
  15. },
  16. min_bucket_selector: {
  17. bucket_selector: {
  18. buckets_path: {
  19. count: "categories._bucket_count",
  20. },
  21. script: {
  22. source: "params.count != 0",
  23. },
  24. },
  25. },
  26. },
  27. },
  28. },
  29. });
  30. console.log(response);
  1. POST /sales/_search
  2. {
  3. "size": 0,
  4. "aggs": {
  5. "histo": {
  6. "date_histogram": {
  7. "field": "date",
  8. "calendar_interval": "day"
  9. },
  10. "aggs": {
  11. "categories": {
  12. "terms": {
  13. "field": "category"
  14. }
  15. },
  16. "min_bucket_selector": {
  17. "bucket_selector": {
  18. "buckets_path": {
  19. "count": "categories._bucket_count"
  20. },
  21. "script": {
  22. "source": "params.count != 0"
  23. }
  24. }
  25. }
  26. }
  27. }
  28. }
  29. }

By using _bucket_count instead of a metric name, we can filter out histo buckets where they contain no buckets for the categories aggregation

Dealing with dots in agg names

An alternate syntax is supported to cope with aggregations or metrics which have dots in the name, such as the 99.9th percentile. This metric may be referred to as:

  1. "buckets_path": "my_percentile[99.9]"

Dealing with gaps in the data

Data in the real world is often noisy and sometimes contains gaps — places where data simply doesn’t exist. This can occur for a variety of reasons, the most common being:

  • Documents falling into a bucket do not contain a required field
  • There are no documents matching the query for one or more buckets
  • The metric being calculated is unable to generate a value, likely because another dependent bucket is missing a value. Some pipeline aggregations have specific requirements that must be met (e.g. a derivative cannot calculate a metric for the first value because there is no previous value, HoltWinters moving average need “warmup” data to begin calculating, etc)

Gap policies are a mechanism to inform the pipeline aggregation about the desired behavior when “gappy” or missing data is encountered. All pipeline aggregations accept the gap_policy parameter. There are currently two gap policies to choose from:

skip

This option treats missing data as if the bucket does not exist. It will skip the bucket and continue calculating using the next available value.

insert_zeros

This option will replace missing values with a zero (0) and pipeline aggregation computation will proceed as normal.

keep_values

This option is similar to skip, except if the metric provides a non-null, non-NaN value this value is used, otherwise the empty bucket is skipped.