Geohash grid aggregation

Geohash grid aggregation

A multi-bucket aggregation that groups geo_point and geo_shape values into buckets that represent a grid. The resulting grid can be sparse and only contains cells that have matching data. Each cell is labeled using a geohash which is of user-definable precision.

  • High precision geohashes have a long string length and represent cells that cover only a small area.
  • Low precision geohashes have a short string length and represent cells that each cover a large area.

Geohashes used in this aggregation can have a choice of precision between 1 and 12.

The highest-precision geohash of length 12 produces cells that cover less than a square metre of land and so high-precision requests can be very costly in terms of RAM and result sizes. Please see the example below on how to first filter the aggregation to a smaller geographic area before requesting high-levels of detail.

You can only use geohash_grid to aggregate an explicitly mapped geo_point or geo_shape field. If the geo_point field contains an array, geohash_grid aggregates all the array values.

Simple low-precision request

  1. resp = client.indices.create(
  2. index="museums",
  3. mappings={
  4. "properties": {
  5. "location": {
  6. "type": "geo_point"
  7. }
  8. }
  9. },
  10. )
  11. print(resp)
  12. resp1 = client.bulk(
  13. index="museums",
  14. refresh=True,
  15. operations=[
  16. {
  17. "index": {
  18. "_id": 1
  19. }
  20. },
  21. {
  22. "location": "POINT (4.912350 52.374081)",
  23. "name": "NEMO Science Museum"
  24. },
  25. {
  26. "index": {
  27. "_id": 2
  28. }
  29. },
  30. {
  31. "location": "POINT (4.901618 52.369219)",
  32. "name": "Museum Het Rembrandthuis"
  33. },
  34. {
  35. "index": {
  36. "_id": 3
  37. }
  38. },
  39. {
  40. "location": "POINT (4.914722 52.371667)",
  41. "name": "Nederlands Scheepvaartmuseum"
  42. },
  43. {
  44. "index": {
  45. "_id": 4
  46. }
  47. },
  48. {
  49. "location": "POINT (4.405200 51.222900)",
  50. "name": "Letterenhuis"
  51. },
  52. {
  53. "index": {
  54. "_id": 5
  55. }
  56. },
  57. {
  58. "location": "POINT (2.336389 48.861111)",
  59. "name": "Musée du Louvre"
  60. },
  61. {
  62. "index": {
  63. "_id": 6
  64. }
  65. },
  66. {
  67. "location": "POINT (2.327000 48.860000)",
  68. "name": "Musée d'Orsay"
  69. }
  70. ],
  71. )
  72. print(resp1)
  73. resp2 = client.search(
  74. index="museums",
  75. size="0",
  76. aggregations={
  77. "large-grid": {
  78. "geohash_grid": {
  79. "field": "location",
  80. "precision": 3
  81. }
  82. }
  83. },
  84. )
  85. print(resp2)
  1. response = client.indices.create(
  2. index: 'museums',
  3. body: {
  4. mappings: {
  5. properties: {
  6. location: {
  7. type: 'geo_point'
  8. }
  9. }
  10. }
  11. }
  12. )
  13. puts response
  14. response = client.bulk(
  15. index: 'museums',
  16. refresh: true,
  17. body: [
  18. {
  19. index: {
  20. _id: 1
  21. }
  22. },
  23. {
  24. location: 'POINT (4.912350 52.374081)',
  25. name: 'NEMO Science Museum'
  26. },
  27. {
  28. index: {
  29. _id: 2
  30. }
  31. },
  32. {
  33. location: 'POINT (4.901618 52.369219)',
  34. name: 'Museum Het Rembrandthuis'
  35. },
  36. {
  37. index: {
  38. _id: 3
  39. }
  40. },
  41. {
  42. location: 'POINT (4.914722 52.371667)',
  43. name: 'Nederlands Scheepvaartmuseum'
  44. },
  45. {
  46. index: {
  47. _id: 4
  48. }
  49. },
  50. {
  51. location: 'POINT (4.405200 51.222900)',
  52. name: 'Letterenhuis'
  53. },
  54. {
  55. index: {
  56. _id: 5
  57. }
  58. },
  59. {
  60. location: 'POINT (2.336389 48.861111)',
  61. name: 'Musée du Louvre'
  62. },
  63. {
  64. index: {
  65. _id: 6
  66. }
  67. },
  68. {
  69. location: 'POINT (2.327000 48.860000)',
  70. name: "Musée d'Orsay"
  71. }
  72. ]
  73. )
  74. puts response
  75. response = client.search(
  76. index: 'museums',
  77. size: 0,
  78. body: {
  79. aggregations: {
  80. "large-grid": {
  81. geohash_grid: {
  82. field: 'location',
  83. precision: 3
  84. }
  85. }
  86. }
  87. }
  88. )
  89. puts response
  1. const response = await client.indices.create({
  2. index: "museums",
  3. mappings: {
  4. properties: {
  5. location: {
  6. type: "geo_point",
  7. },
  8. },
  9. },
  10. });
  11. console.log(response);
  12. const response1 = await client.bulk({
  13. index: "museums",
  14. refresh: "true",
  15. operations: [
  16. {
  17. index: {
  18. _id: 1,
  19. },
  20. },
  21. {
  22. location: "POINT (4.912350 52.374081)",
  23. name: "NEMO Science Museum",
  24. },
  25. {
  26. index: {
  27. _id: 2,
  28. },
  29. },
  30. {
  31. location: "POINT (4.901618 52.369219)",
  32. name: "Museum Het Rembrandthuis",
  33. },
  34. {
  35. index: {
  36. _id: 3,
  37. },
  38. },
  39. {
  40. location: "POINT (4.914722 52.371667)",
  41. name: "Nederlands Scheepvaartmuseum",
  42. },
  43. {
  44. index: {
  45. _id: 4,
  46. },
  47. },
  48. {
  49. location: "POINT (4.405200 51.222900)",
  50. name: "Letterenhuis",
  51. },
  52. {
  53. index: {
  54. _id: 5,
  55. },
  56. },
  57. {
  58. location: "POINT (2.336389 48.861111)",
  59. name: "Musée du Louvre",
  60. },
  61. {
  62. index: {
  63. _id: 6,
  64. },
  65. },
  66. {
  67. location: "POINT (2.327000 48.860000)",
  68. name: "Musée d'Orsay",
  69. },
  70. ],
  71. });
  72. console.log(response1);
  73. const response2 = await client.search({
  74. index: "museums",
  75. size: 0,
  76. aggregations: {
  77. "large-grid": {
  78. geohash_grid: {
  79. field: "location",
  80. precision: 3,
  81. },
  82. },
  83. },
  84. });
  85. console.log(response2);
  1. PUT /museums
  2. {
  3. "mappings": {
  4. "properties": {
  5. "location": {
  6. "type": "geo_point"
  7. }
  8. }
  9. }
  10. }
  11. POST /museums/_bulk?refresh
  12. {"index":{"_id":1}}
  13. {"location": "POINT (4.912350 52.374081)", "name": "NEMO Science Museum"}
  14. {"index":{"_id":2}}
  15. {"location": "POINT (4.901618 52.369219)", "name": "Museum Het Rembrandthuis"}
  16. {"index":{"_id":3}}
  17. {"location": "POINT (4.914722 52.371667)", "name": "Nederlands Scheepvaartmuseum"}
  18. {"index":{"_id":4}}
  19. {"location": "POINT (4.405200 51.222900)", "name": "Letterenhuis"}
  20. {"index":{"_id":5}}
  21. {"location": "POINT (2.336389 48.861111)", "name": "Musée du Louvre"}
  22. {"index":{"_id":6}}
  23. {"location": "POINT (2.327000 48.860000)", "name": "Musée d'Orsay"}
  24. POST /museums/_search?size=0
  25. {
  26. "aggregations": {
  27. "large-grid": {
  28. "geohash_grid": {
  29. "field": "location",
  30. "precision": 3
  31. }
  32. }
  33. }
  34. }

Response:

  1. {
  2. ...
  3. "aggregations": {
  4. "large-grid": {
  5. "buckets": [
  6. {
  7. "key": "u17",
  8. "doc_count": 3
  9. },
  10. {
  11. "key": "u09",
  12. "doc_count": 2
  13. },
  14. {
  15. "key": "u15",
  16. "doc_count": 1
  17. }
  18. ]
  19. }
  20. }
  21. }

High-precision requests

When requesting detailed buckets (typically for displaying a “zoomed in” map) a filter like geo_bounding_box should be applied to narrow the subject area otherwise potentially millions of buckets will be created and returned.

  1. resp = client.search(
  2. index="museums",
  3. size="0",
  4. aggregations={
  5. "zoomed-in": {
  6. "filter": {
  7. "geo_bounding_box": {
  8. "location": {
  9. "top_left": "POINT (4.9 52.4)",
  10. "bottom_right": "POINT (5.0 52.3)"
  11. }
  12. }
  13. },
  14. "aggregations": {
  15. "zoom1": {
  16. "geohash_grid": {
  17. "field": "location",
  18. "precision": 8
  19. }
  20. }
  21. }
  22. }
  23. },
  24. )
  25. print(resp)
  1. response = client.search(
  2. index: 'museums',
  3. size: 0,
  4. body: {
  5. aggregations: {
  6. "zoomed-in": {
  7. filter: {
  8. geo_bounding_box: {
  9. location: {
  10. top_left: 'POINT (4.9 52.4)',
  11. bottom_right: 'POINT (5.0 52.3)'
  12. }
  13. }
  14. },
  15. aggregations: {
  16. "zoom1": {
  17. geohash_grid: {
  18. field: 'location',
  19. precision: 8
  20. }
  21. }
  22. }
  23. }
  24. }
  25. }
  26. )
  27. puts response
  1. const response = await client.search({
  2. index: "museums",
  3. size: 0,
  4. aggregations: {
  5. "zoomed-in": {
  6. filter: {
  7. geo_bounding_box: {
  8. location: {
  9. top_left: "POINT (4.9 52.4)",
  10. bottom_right: "POINT (5.0 52.3)",
  11. },
  12. },
  13. },
  14. aggregations: {
  15. zoom1: {
  16. geohash_grid: {
  17. field: "location",
  18. precision: 8,
  19. },
  20. },
  21. },
  22. },
  23. },
  24. });
  25. console.log(response);
  1. POST /museums/_search?size=0
  2. {
  3. "aggregations": {
  4. "zoomed-in": {
  5. "filter": {
  6. "geo_bounding_box": {
  7. "location": {
  8. "top_left": "POINT (4.9 52.4)",
  9. "bottom_right": "POINT (5.0 52.3)"
  10. }
  11. }
  12. },
  13. "aggregations": {
  14. "zoom1": {
  15. "geohash_grid": {
  16. "field": "location",
  17. "precision": 8
  18. }
  19. }
  20. }
  21. }
  22. }
  23. }

The geohashes returned by the geohash_grid aggregation can be also used for zooming in. To zoom into the first geohash u17 returned in the previous example, it should be specified as both top_left and bottom_right corner:

  1. resp = client.search(
  2. index="museums",
  3. size="0",
  4. aggregations={
  5. "zoomed-in": {
  6. "filter": {
  7. "geo_bounding_box": {
  8. "location": {
  9. "top_left": "u17",
  10. "bottom_right": "u17"
  11. }
  12. }
  13. },
  14. "aggregations": {
  15. "zoom1": {
  16. "geohash_grid": {
  17. "field": "location",
  18. "precision": 8
  19. }
  20. }
  21. }
  22. }
  23. },
  24. )
  25. print(resp)
  1. response = client.search(
  2. index: 'museums',
  3. size: 0,
  4. body: {
  5. aggregations: {
  6. "zoomed-in": {
  7. filter: {
  8. geo_bounding_box: {
  9. location: {
  10. top_left: 'u17',
  11. bottom_right: 'u17'
  12. }
  13. }
  14. },
  15. aggregations: {
  16. "zoom1": {
  17. geohash_grid: {
  18. field: 'location',
  19. precision: 8
  20. }
  21. }
  22. }
  23. }
  24. }
  25. }
  26. )
  27. puts response
  1. const response = await client.search({
  2. index: "museums",
  3. size: 0,
  4. aggregations: {
  5. "zoomed-in": {
  6. filter: {
  7. geo_bounding_box: {
  8. location: {
  9. top_left: "u17",
  10. bottom_right: "u17",
  11. },
  12. },
  13. },
  14. aggregations: {
  15. zoom1: {
  16. geohash_grid: {
  17. field: "location",
  18. precision: 8,
  19. },
  20. },
  21. },
  22. },
  23. },
  24. });
  25. console.log(response);
  1. POST /museums/_search?size=0
  2. {
  3. "aggregations": {
  4. "zoomed-in": {
  5. "filter": {
  6. "geo_bounding_box": {
  7. "location": {
  8. "top_left": "u17",
  9. "bottom_right": "u17"
  10. }
  11. }
  12. },
  13. "aggregations": {
  14. "zoom1": {
  15. "geohash_grid": {
  16. "field": "location",
  17. "precision": 8
  18. }
  19. }
  20. }
  21. }
  22. }
  23. }
  1. {
  2. ...
  3. "aggregations": {
  4. "zoomed-in": {
  5. "doc_count": 3,
  6. "zoom1": {
  7. "buckets": [
  8. {
  9. "key": "u173zy3j",
  10. "doc_count": 1
  11. },
  12. {
  13. "key": "u173zvfz",
  14. "doc_count": 1
  15. },
  16. {
  17. "key": "u173zt90",
  18. "doc_count": 1
  19. }
  20. ]
  21. }
  22. }
  23. }
  24. }

For “zooming in” on the system that don’t support geohashes, the bucket keys should be translated into bounding boxes using one of available geohash libraries. For example, for javascript the node-geohash library can be used:

  1. var geohash = require('ngeohash');
  2. // bbox will contain [ 52.03125, 4.21875, 53.4375, 5.625 ]
  3. // [ minlat, minlon, maxlat, maxlon]
  4. var bbox = geohash.decode_bbox('u17');

Requests with additional bounding box filtering

The geohash_grid aggregation supports an optional bounds parameter that restricts the cells considered to those that intersects the bounds provided. The bounds parameter accepts the bounding box in all the same accepted formats of the bounds specified in the Geo Bounding Box Query. This bounding box can be used with or without an additional geo_bounding_box query filtering the points prior to aggregating. It is an independent bounding box that can intersect with, be equal to, or be disjoint to any additional geo_bounding_box queries defined in the context of the aggregation.

  1. resp = client.search(
  2. index="museums",
  3. size="0",
  4. aggregations={
  5. "tiles-in-bounds": {
  6. "geohash_grid": {
  7. "field": "location",
  8. "precision": 8,
  9. "bounds": {
  10. "top_left": "POINT (4.21875 53.4375)",
  11. "bottom_right": "POINT (5.625 52.03125)"
  12. }
  13. }
  14. }
  15. },
  16. )
  17. print(resp)
  1. response = client.search(
  2. index: 'museums',
  3. size: 0,
  4. body: {
  5. aggregations: {
  6. "tiles-in-bounds": {
  7. geohash_grid: {
  8. field: 'location',
  9. precision: 8,
  10. bounds: {
  11. top_left: 'POINT (4.21875 53.4375)',
  12. bottom_right: 'POINT (5.625 52.03125)'
  13. }
  14. }
  15. }
  16. }
  17. }
  18. )
  19. puts response
  1. const response = await client.search({
  2. index: "museums",
  3. size: 0,
  4. aggregations: {
  5. "tiles-in-bounds": {
  6. geohash_grid: {
  7. field: "location",
  8. precision: 8,
  9. bounds: {
  10. top_left: "POINT (4.21875 53.4375)",
  11. bottom_right: "POINT (5.625 52.03125)",
  12. },
  13. },
  14. },
  15. },
  16. });
  17. console.log(response);
  1. POST /museums/_search?size=0
  2. {
  3. "aggregations": {
  4. "tiles-in-bounds": {
  5. "geohash_grid": {
  6. "field": "location",
  7. "precision": 8,
  8. "bounds": {
  9. "top_left": "POINT (4.21875 53.4375)",
  10. "bottom_right": "POINT (5.625 52.03125)"
  11. }
  12. }
  13. }
  14. }
  15. }
  1. {
  2. ...
  3. "aggregations": {
  4. "tiles-in-bounds": {
  5. "buckets": [
  6. {
  7. "key": "u173zy3j",
  8. "doc_count": 1
  9. },
  10. {
  11. "key": "u173zvfz",
  12. "doc_count": 1
  13. },
  14. {
  15. "key": "u173zt90",
  16. "doc_count": 1
  17. }
  18. ]
  19. }
  20. }
  21. }

Cell dimensions at the equator

The table below shows the metric dimensions for cells covered by various string lengths of geohash. Cell dimensions vary with latitude and so the table is for the worst-case scenario at the equator.

GeoHash length

Area width x height

1

5,009.4km x 4,992.6km

2

1,252.3km x 624.1km

3

156.5km x 156km

4

39.1km x 19.5km

5

4.9km x 4.9km

6

1.2km x 609.4m

7

152.9m x 152.4m

8

38.2m x 19m

9

4.8m x 4.8m

10

1.2m x 59.5cm

11

14.9cm x 14.9cm

12

3.7cm x 1.9cm

Aggregating geo_shape fields

Aggregating on Geoshape fields works just as it does for points, except that a single shape can be counted for in multiple tiles. A shape will contribute to the count of matching values if any part of its shape intersects with that tile. Below is an image that demonstrates this:

geoshape grid

Options

field

Mandatory. Field containing indexed geo-point or geo-shape values. Must be explicitly mapped as a geo_point or a geo_shape field. If the field contains an array, geohash_grid aggregates all array values.

precision

Optional. The string length of the geohashes used to define cells/buckets in the results. Defaults to 5. The precision can either be defined in terms of the integer precision levels mentioned above. Values outside of [1,12] will be rejected. Alternatively, the precision level can be approximated from a distance measure like “1km”, “10m”. The precision level is calculate such that cells will not exceed the specified size (diagonal) of the required precision. When this would lead to precision levels higher than the supported 12 levels, (e.g. for distances <5.6cm) the value is rejected.

bounds

Optional. The bounding box to filter the points in the bucket.

size

Optional. The maximum number of geohash buckets to return (defaults to 10,000). When results are trimmed, buckets are prioritised based on the volumes of documents they contain.

shard_size

Optional. To allow for more accurate counting of the top cells returned in the final result the aggregation defaults to returning max(10,(size x number-of-shards)) buckets from each shard. If this heuristic is undesirable, the number considered from each shard can be over-ridden using this parameter.