daterange-aggregation.asciidoc 9.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320
  1. [[search-aggregations-bucket-daterange-aggregation]]
  2. === Date Range Aggregation
  3. A range aggregation that is dedicated for date values. The main difference
  4. between this aggregation and the normal
  5. <<search-aggregations-bucket-range-aggregation,range>>
  6. aggregation is that the `from` and `to` values can be expressed in
  7. <<date-math,Date Math>> expressions, and it is also possible to specify a date
  8. format by which the `from` and `to` response fields will be returned.
  9. Note that this aggregation includes the `from` value and excludes the `to` value
  10. for each range.
  11. Example:
  12. [source,js]
  13. --------------------------------------------------
  14. POST /sales/_search?size=0
  15. {
  16. "aggs": {
  17. "range": {
  18. "date_range": {
  19. "field": "date",
  20. "format": "MM-yyy",
  21. "ranges": [
  22. { "to": "now-10M/M" }, <1>
  23. { "from": "now-10M/M" } <2>
  24. ]
  25. }
  26. }
  27. }
  28. }
  29. --------------------------------------------------
  30. // CONSOLE
  31. // TEST[setup:sales s/now-10M\/M/10-2015/]
  32. <1> < now minus 10 months, rounded down to the start of the month.
  33. <2> >= now minus 10 months, rounded down to the start of the month.
  34. In the example above, we created two range buckets, the first will "bucket" all
  35. documents dated prior to 10 months ago and the second will "bucket" all
  36. documents dated since 10 months ago
  37. Response:
  38. [source,js]
  39. --------------------------------------------------
  40. {
  41. ...
  42. "aggregations": {
  43. "range": {
  44. "buckets": [
  45. {
  46. "to": 1.4436576E12,
  47. "to_as_string": "10-2015",
  48. "doc_count": 7,
  49. "key": "*-10-2015"
  50. },
  51. {
  52. "from": 1.4436576E12,
  53. "from_as_string": "10-2015",
  54. "doc_count": 0,
  55. "key": "10-2015-*"
  56. }
  57. ]
  58. }
  59. }
  60. }
  61. --------------------------------------------------
  62. // TESTRESPONSE[s/\.\.\./"took": $body.took,"timed_out": false,"_shards": $body._shards,"hits": $body.hits,/]
  63. ==== Missing Values
  64. The `missing` parameter defines how documents that are missing a value should
  65. be treated. By default they will be ignored but it is also possible to treat
  66. them as if they had a value. This is done by adding a set of fieldname :
  67. value mappings to specify default values per field.
  68. [source,js]
  69. --------------------------------------------------
  70. POST /sales/_search?size=0
  71. {
  72. "aggs": {
  73. "range": {
  74. "date_range": {
  75. "field": "date",
  76. "missing": "1976/11/30",
  77. "ranges": [
  78. {
  79. "key": "Older",
  80. "to": "2016/02/01"
  81. }, <1>
  82. {
  83. "key": "Newer",
  84. "from": "2016/02/01",
  85. "to" : "now/d"
  86. }
  87. ]
  88. }
  89. }
  90. }
  91. }
  92. --------------------------------------------------
  93. // CONSOLE
  94. // TEST[setup:sales]
  95. <1> Documents without a value in the `date` field will be added to the "Older"
  96. bucket, as if they had a date value of "1899-12-31".
  97. [[date-format-pattern]]
  98. ==== Date Format/Pattern
  99. NOTE: this information was copied from
  100. http://www.joda.org/joda-time/apidocs/org/joda/time/format/DateTimeFormat.html[JodaDate]
  101. All ASCII letters are reserved as format pattern letters, which are defined
  102. as follows:
  103. [options="header"]
  104. |=======
  105. |Symbol |Meaning |Presentation |Examples
  106. |G |era |text |AD
  107. |C |century of era (>=0) |number |20
  108. |Y |year of era (>=0) |year |1996
  109. |x |weekyear |year |1996
  110. |w |week of weekyear |number |27
  111. |e |day of week |number |2
  112. |E |day of week |text |Tuesday; Tue
  113. |y |year |year |1996
  114. |D |day of year |number |189
  115. |M |month of year |month |July; Jul; 07
  116. |d |day of month |number |10
  117. |a |halfday of day |text |PM
  118. |K |hour of halfday (0~11) |number |0
  119. |h |clockhour of halfday (1~12) |number |12
  120. |H |hour of day (0~23) |number |0
  121. |k |clockhour of day (1~24) |number |24
  122. |m |minute of hour |number |30
  123. |s |second of minute |number |55
  124. |S |fraction of second |number |978
  125. |z |time zone |text |Pacific Standard Time; PST
  126. |Z |time zone offset/id |zone |-0800; -08:00; America/Los_Angeles
  127. |' |escape for text |delimiter
  128. |'' |single quote |literal |'
  129. |=======
  130. The count of pattern letters determine the format.
  131. Text:: If the number of pattern letters is 4 or more, the full form is used;
  132. otherwise a short or abbreviated form is used if available.
  133. Number:: The minimum number of digits. Shorter numbers are zero-padded to
  134. this amount.
  135. Year:: Numeric presentation for year and weekyear fields are handled
  136. specially. For example, if the count of 'y' is 2, the year will be displayed
  137. as the zero-based year of the century, which is two digits.
  138. Month:: 3 or over, use text, otherwise use number.
  139. Zone:: 'Z' outputs offset without a colon, 'ZZ' outputs the offset with a
  140. colon, 'ZZZ' or more outputs the zone id.
  141. Zone names:: Time zone names ('z') cannot be parsed.
  142. Any characters in the pattern that are not in the ranges of ['a'..'z'] and
  143. ['A'..'Z'] will be treated as quoted text. For instance, characters like ':',
  144. '.', ' ', '#' and '?' will appear in the resulting time text even they are
  145. not embraced within single quotes.
  146. [[time-zones]]
  147. ==== Time zone in date range aggregations
  148. Dates can be converted from another time zone to UTC by specifying the
  149. `time_zone` parameter.
  150. Time zones may either be specified as an ISO 8601 UTC offset (e.g. +01:00 or
  151. -08:00) or as one of the http://www.joda.org/joda-time/timezones.html [time
  152. zone ids] from the TZ database.
  153. The `time_zone` parameter is also applied to rounding in date math expressions.
  154. As an example, to round to the beginning of the day in the CET time zone, you
  155. can do the following:
  156. [source,js]
  157. --------------------------------------------------
  158. POST /sales/_search?size=0
  159. {
  160. "aggs": {
  161. "range": {
  162. "date_range": {
  163. "field": "date",
  164. "time_zone": "CET",
  165. "ranges": [
  166. { "to": "2016/02/01" }, <1>
  167. { "from": "2016/02/01", "to" : "now/d" <2>},
  168. { "from": "now/d" }
  169. ]
  170. }
  171. }
  172. }
  173. }
  174. --------------------------------------------------
  175. // CONSOLE
  176. // TEST[setup:sales]
  177. <1> This date will be converted to `2016-02-15T00:00:00.000+01:00`.
  178. <2> `now/d` will be rounded to the beginning of the day in the CET time zone.
  179. ==== Keyed Response
  180. Setting the `keyed` flag to `true` will associate a unique string key with each
  181. bucket and return the ranges as a hash rather than an array:
  182. [source,js]
  183. --------------------------------------------------
  184. POST /sales/_search?size=0
  185. {
  186. "aggs": {
  187. "range": {
  188. "date_range": {
  189. "field": "date",
  190. "format": "MM-yyy",
  191. "ranges": [
  192. { "to": "now-10M/M" },
  193. { "from": "now-10M/M" }
  194. ],
  195. "keyed": true
  196. }
  197. }
  198. }
  199. }
  200. --------------------------------------------------
  201. // CONSOLE
  202. // TEST[setup:sales s/now-10M\/M/10-2015/]
  203. Response:
  204. [source,js]
  205. --------------------------------------------------
  206. {
  207. ...
  208. "aggregations": {
  209. "range": {
  210. "buckets": {
  211. "*-10-2015": {
  212. "to": 1.4436576E12,
  213. "to_as_string": "10-2015",
  214. "doc_count": 7
  215. },
  216. "10-2015-*": {
  217. "from": 1.4436576E12,
  218. "from_as_string": "10-2015",
  219. "doc_count": 0
  220. }
  221. }
  222. }
  223. }
  224. }
  225. --------------------------------------------------
  226. // TESTRESPONSE[s/\.\.\./"took": $body.took,"timed_out": false,"_shards": $body._shards,"hits": $body.hits,/]
  227. It is also possible to customize the key for each range:
  228. [source,js]
  229. --------------------------------------------------
  230. POST /sales/_search?size=0
  231. {
  232. "aggs": {
  233. "range": {
  234. "date_range": {
  235. "field": "date",
  236. "format": "MM-yyy",
  237. "ranges": [
  238. { "from": "01-2015", "to": "03-2015", "key": "quarter_01" },
  239. { "from": "03-2015", "to": "06-2015", "key": "quarter_02" }
  240. ],
  241. "keyed": true
  242. }
  243. }
  244. }
  245. }
  246. --------------------------------------------------
  247. // CONSOLE
  248. // TEST[setup:sales]
  249. Response:
  250. [source,js]
  251. --------------------------------------------------
  252. {
  253. ...
  254. "aggregations": {
  255. "range": {
  256. "buckets": {
  257. "quarter_01": {
  258. "from": 1.4200704E12,
  259. "from_as_string": "01-2015",
  260. "to": 1.425168E12,
  261. "to_as_string": "03-2015",
  262. "doc_count": 5
  263. },
  264. "quarter_02": {
  265. "from": 1.425168E12,
  266. "from_as_string": "03-2015",
  267. "to": 1.4331168E12,
  268. "to_as_string": "06-2015",
  269. "doc_count": 2
  270. }
  271. }
  272. }
  273. }
  274. }
  275. --------------------------------------------------
  276. // TESTRESPONSE[s/\.\.\./"took": $body.took,"timed_out": false,"_shards": $body._shards,"hits": $body.hits,/]