Auto-interval date histogram aggregation

Auto-interval date histogram aggregation

A multi-bucket aggregation similar to the Date histogram except instead of providing an interval to use as the width of each bucket, a target number of buckets is provided indicating the number of buckets needed and the interval of the buckets is automatically chosen to best achieve that target. The number of buckets returned will always be less than or equal to this target number.

The buckets field is optional, and will default to 10 buckets if not specified.

Requesting a target of 10 buckets.

  1. resp = client.search(
  2. index="sales",
  3. size="0",
  4. aggs={
  5. "sales_over_time": {
  6. "auto_date_histogram": {
  7. "field": "date",
  8. "buckets": 10
  9. }
  10. }
  11. },
  12. )
  13. print(resp)
  1. response = client.search(
  2. index: 'sales',
  3. size: 0,
  4. body: {
  5. aggregations: {
  6. sales_over_time: {
  7. auto_date_histogram: {
  8. field: 'date',
  9. buckets: 10
  10. }
  11. }
  12. }
  13. }
  14. )
  15. puts response
  1. const response = await client.search({
  2. index: "sales",
  3. size: 0,
  4. aggs: {
  5. sales_over_time: {
  6. auto_date_histogram: {
  7. field: "date",
  8. buckets: 10,
  9. },
  10. },
  11. },
  12. });
  13. console.log(response);
  1. POST /sales/_search?size=0
  2. {
  3. "aggs": {
  4. "sales_over_time": {
  5. "auto_date_histogram": {
  6. "field": "date",
  7. "buckets": 10
  8. }
  9. }
  10. }
  11. }

Keys

Internally, a date is represented as a 64 bit number representing a timestamp in milliseconds-since-the-epoch. These timestamps are returned as the bucket keys. The key_as_string is the same timestamp converted to a formatted date string using the format specified with the format parameter:

If no format is specified, then it will use the first date format specified in the field mapping.

  1. resp = client.search(
  2. index="sales",
  3. size="0",
  4. aggs={
  5. "sales_over_time": {
  6. "auto_date_histogram": {
  7. "field": "date",
  8. "buckets": 5,
  9. "format": "yyyy-MM-dd"
  10. }
  11. }
  12. },
  13. )
  14. print(resp)
  1. response = client.search(
  2. index: 'sales',
  3. size: 0,
  4. body: {
  5. aggregations: {
  6. sales_over_time: {
  7. auto_date_histogram: {
  8. field: 'date',
  9. buckets: 5,
  10. format: 'yyyy-MM-dd'
  11. }
  12. }
  13. }
  14. }
  15. )
  16. puts response
  1. const response = await client.search({
  2. index: "sales",
  3. size: 0,
  4. aggs: {
  5. sales_over_time: {
  6. auto_date_histogram: {
  7. field: "date",
  8. buckets: 5,
  9. format: "yyyy-MM-dd",
  10. },
  11. },
  12. },
  13. });
  14. console.log(response);
  1. POST /sales/_search?size=0
  2. {
  3. "aggs": {
  4. "sales_over_time": {
  5. "auto_date_histogram": {
  6. "field": "date",
  7. "buckets": 5,
  8. "format": "yyyy-MM-dd"
  9. }
  10. }
  11. }
  12. }

Supports expressive date format pattern

Response:

  1. {
  2. ...
  3. "aggregations": {
  4. "sales_over_time": {
  5. "buckets": [
  6. {
  7. "key_as_string": "2015-01-01",
  8. "key": 1420070400000,
  9. "doc_count": 3
  10. },
  11. {
  12. "key_as_string": "2015-02-01",
  13. "key": 1422748800000,
  14. "doc_count": 2
  15. },
  16. {
  17. "key_as_string": "2015-03-01",
  18. "key": 1425168000000,
  19. "doc_count": 2
  20. }
  21. ],
  22. "interval": "1M"
  23. }
  24. }
  25. }

Intervals

The interval of the returned buckets is selected based on the data collected by the aggregation so that the number of buckets returned is less than or equal to the number requested. The possible intervals returned are:

seconds

In multiples of 1, 5, 10 and 30

minutes

In multiples of 1, 5, 10 and 30

hours

In multiples of 1, 3 and 12

days

In multiples of 1, and 7

months

In multiples of 1, and 3

years

In multiples of 1, 5, 10, 20, 50 and 100

In the worst case, where the number of daily buckets are too many for the requested number of buckets, the number of buckets returned will be 1/7th of the number of buckets requested.

Time Zone

Date-times are stored in Elasticsearch in UTC. By default, all bucketing and rounding is also done in UTC. The time_zone parameter can be used to indicate that bucketing should use a different time zone.

Time zones may either be specified as an ISO 8601 UTC offset (e.g. +01:00 or -08:00) or as a timezone id, an identifier used in the TZ database like America/Los_Angeles.

Consider the following example:

  1. resp = client.index(
  2. index="my-index-000001",
  3. id="1",
  4. refresh=True,
  5. document={
  6. "date": "2015-10-01T00:30:00Z"
  7. },
  8. )
  9. print(resp)
  10. resp1 = client.index(
  11. index="my-index-000001",
  12. id="2",
  13. refresh=True,
  14. document={
  15. "date": "2015-10-01T01:30:00Z"
  16. },
  17. )
  18. print(resp1)
  19. resp2 = client.index(
  20. index="my-index-000001",
  21. id="3",
  22. refresh=True,
  23. document={
  24. "date": "2015-10-01T02:30:00Z"
  25. },
  26. )
  27. print(resp2)
  28. resp3 = client.search(
  29. index="my-index-000001",
  30. size="0",
  31. aggs={
  32. "by_day": {
  33. "auto_date_histogram": {
  34. "field": "date",
  35. "buckets": 3
  36. }
  37. }
  38. },
  39. )
  40. print(resp3)
  1. response = client.index(
  2. index: 'my-index-000001',
  3. id: 1,
  4. refresh: true,
  5. body: {
  6. date: '2015-10-01T00:30:00Z'
  7. }
  8. )
  9. puts response
  10. response = client.index(
  11. index: 'my-index-000001',
  12. id: 2,
  13. refresh: true,
  14. body: {
  15. date: '2015-10-01T01:30:00Z'
  16. }
  17. )
  18. puts response
  19. response = client.index(
  20. index: 'my-index-000001',
  21. id: 3,
  22. refresh: true,
  23. body: {
  24. date: '2015-10-01T02:30:00Z'
  25. }
  26. )
  27. puts response
  28. response = client.search(
  29. index: 'my-index-000001',
  30. size: 0,
  31. body: {
  32. aggregations: {
  33. by_day: {
  34. auto_date_histogram: {
  35. field: 'date',
  36. buckets: 3
  37. }
  38. }
  39. }
  40. }
  41. )
  42. puts response
  1. const response = await client.index({
  2. index: "my-index-000001",
  3. id: 1,
  4. refresh: "true",
  5. document: {
  6. date: "2015-10-01T00:30:00Z",
  7. },
  8. });
  9. console.log(response);
  10. const response1 = await client.index({
  11. index: "my-index-000001",
  12. id: 2,
  13. refresh: "true",
  14. document: {
  15. date: "2015-10-01T01:30:00Z",
  16. },
  17. });
  18. console.log(response1);
  19. const response2 = await client.index({
  20. index: "my-index-000001",
  21. id: 3,
  22. refresh: "true",
  23. document: {
  24. date: "2015-10-01T02:30:00Z",
  25. },
  26. });
  27. console.log(response2);
  28. const response3 = await client.search({
  29. index: "my-index-000001",
  30. size: 0,
  31. aggs: {
  32. by_day: {
  33. auto_date_histogram: {
  34. field: "date",
  35. buckets: 3,
  36. },
  37. },
  38. },
  39. });
  40. console.log(response3);
  1. PUT my-index-000001/_doc/1?refresh
  2. {
  3. "date": "2015-10-01T00:30:00Z"
  4. }
  5. PUT my-index-000001/_doc/2?refresh
  6. {
  7. "date": "2015-10-01T01:30:00Z"
  8. }
  9. PUT my-index-000001/_doc/3?refresh
  10. {
  11. "date": "2015-10-01T02:30:00Z"
  12. }
  13. GET my-index-000001/_search?size=0
  14. {
  15. "aggs": {
  16. "by_day": {
  17. "auto_date_histogram": {
  18. "field": "date",
  19. "buckets" : 3
  20. }
  21. }
  22. }
  23. }

UTC is used if no time zone is specified, three 1-hour buckets are returned starting at midnight UTC on 1 October 2015:

  1. {
  2. ...
  3. "aggregations": {
  4. "by_day": {
  5. "buckets": [
  6. {
  7. "key_as_string": "2015-10-01T00:00:00.000Z",
  8. "key": 1443657600000,
  9. "doc_count": 1
  10. },
  11. {
  12. "key_as_string": "2015-10-01T01:00:00.000Z",
  13. "key": 1443661200000,
  14. "doc_count": 1
  15. },
  16. {
  17. "key_as_string": "2015-10-01T02:00:00.000Z",
  18. "key": 1443664800000,
  19. "doc_count": 1
  20. }
  21. ],
  22. "interval": "1h"
  23. }
  24. }
  25. }

If a time_zone of -01:00 is specified, then midnight starts at one hour before midnight UTC:

  1. resp = client.search(
  2. index="my-index-000001",
  3. size="0",
  4. aggs={
  5. "by_day": {
  6. "auto_date_histogram": {
  7. "field": "date",
  8. "buckets": 3,
  9. "time_zone": "-01:00"
  10. }
  11. }
  12. },
  13. )
  14. print(resp)
  1. response = client.search(
  2. index: 'my-index-000001',
  3. size: 0,
  4. body: {
  5. aggregations: {
  6. by_day: {
  7. auto_date_histogram: {
  8. field: 'date',
  9. buckets: 3,
  10. time_zone: '-01:00'
  11. }
  12. }
  13. }
  14. }
  15. )
  16. puts response
  1. const response = await client.search({
  2. index: "my-index-000001",
  3. size: 0,
  4. aggs: {
  5. by_day: {
  6. auto_date_histogram: {
  7. field: "date",
  8. buckets: 3,
  9. time_zone: "-01:00",
  10. },
  11. },
  12. },
  13. });
  14. console.log(response);
  1. GET my-index-000001/_search?size=0
  2. {
  3. "aggs": {
  4. "by_day": {
  5. "auto_date_histogram": {
  6. "field": "date",
  7. "buckets" : 3,
  8. "time_zone": "-01:00"
  9. }
  10. }
  11. }
  12. }

Now three 1-hour buckets are still returned but the first bucket starts at 11:00pm on 30 September 2015 since that is the local time for the bucket in the specified time zone.

  1. {
  2. ...
  3. "aggregations": {
  4. "by_day": {
  5. "buckets": [
  6. {
  7. "key_as_string": "2015-09-30T23:00:00.000-01:00",
  8. "key": 1443657600000,
  9. "doc_count": 1
  10. },
  11. {
  12. "key_as_string": "2015-10-01T00:00:00.000-01:00",
  13. "key": 1443661200000,
  14. "doc_count": 1
  15. },
  16. {
  17. "key_as_string": "2015-10-01T01:00:00.000-01:00",
  18. "key": 1443664800000,
  19. "doc_count": 1
  20. }
  21. ],
  22. "interval": "1h"
  23. }
  24. }
  25. }

The key_as_string value represents midnight on each day in the specified time zone.

When using time zones that follow DST (daylight savings time) changes, buckets close to the moment when those changes happen can have slightly different sizes than neighbouring buckets. For example, consider a DST start in the CET time zone: on 27 March 2016 at 2am, clocks were turned forward 1 hour to 3am local time. If the result of the aggregation was daily buckets, the bucket covering that day will only hold data for 23 hours instead of the usual 24 hours for other buckets. The same is true for shorter intervals like e.g. 12h. Here, we will have only a 11h bucket on the morning of 27 March when the DST shift happens.

Minimum Interval parameter

The minimum_interval allows the caller to specify the minimum rounding interval that should be used. This can make the collection process more efficient, as the aggregation will not attempt to round at any interval lower than minimum_interval.

The accepted units for minimum_interval are:

  • year
  • month
  • day
  • hour
  • minute
  • second
  1. resp = client.search(
  2. index="sales",
  3. size="0",
  4. aggs={
  5. "sale_date": {
  6. "auto_date_histogram": {
  7. "field": "date",
  8. "buckets": 10,
  9. "minimum_interval": "minute"
  10. }
  11. }
  12. },
  13. )
  14. print(resp)
  1. response = client.search(
  2. index: 'sales',
  3. size: 0,
  4. body: {
  5. aggregations: {
  6. sale_date: {
  7. auto_date_histogram: {
  8. field: 'date',
  9. buckets: 10,
  10. minimum_interval: 'minute'
  11. }
  12. }
  13. }
  14. }
  15. )
  16. puts response
  1. const response = await client.search({
  2. index: "sales",
  3. size: 0,
  4. aggs: {
  5. sale_date: {
  6. auto_date_histogram: {
  7. field: "date",
  8. buckets: 10,
  9. minimum_interval: "minute",
  10. },
  11. },
  12. },
  13. });
  14. console.log(response);
  1. POST /sales/_search?size=0
  2. {
  3. "aggs": {
  4. "sale_date": {
  5. "auto_date_histogram": {
  6. "field": "date",
  7. "buckets": 10,
  8. "minimum_interval": "minute"
  9. }
  10. }
  11. }
  12. }

Missing value

The missing parameter defines how documents that are missing a value should be treated. By default they will be ignored but it is also possible to treat them as if they had a value.

  1. resp = client.search(
  2. index="sales",
  3. size="0",
  4. aggs={
  5. "sale_date": {
  6. "auto_date_histogram": {
  7. "field": "date",
  8. "buckets": 10,
  9. "missing": "2000/01/01"
  10. }
  11. }
  12. },
  13. )
  14. print(resp)
  1. response = client.search(
  2. index: 'sales',
  3. size: 0,
  4. body: {
  5. aggregations: {
  6. sale_date: {
  7. auto_date_histogram: {
  8. field: 'date',
  9. buckets: 10,
  10. missing: '2000/01/01'
  11. }
  12. }
  13. }
  14. }
  15. )
  16. puts response
  1. const response = await client.search({
  2. index: "sales",
  3. size: 0,
  4. aggs: {
  5. sale_date: {
  6. auto_date_histogram: {
  7. field: "date",
  8. buckets: 10,
  9. missing: "2000/01/01",
  10. },
  11. },
  12. },
  13. });
  14. console.log(response);
  1. POST /sales/_search?size=0
  2. {
  3. "aggs": {
  4. "sale_date": {
  5. "auto_date_histogram": {
  6. "field": "date",
  7. "buckets": 10,
  8. "missing": "2000/01/01"
  9. }
  10. }
  11. }
  12. }

Documents without a value in the publish_date field will fall into the same bucket as documents that have the value 2000-01-01.