Geo-grid processor

Geo-grid processor

Converts geo-grid definitions of grid tiles or cells to regular bounding boxes or polygons which describe their shape. This is useful if there is a need to interact with the tile shapes as spatially indexable fields. For example the geotile field value "4/8/3" could be indexed as a string field, but that would not enable any spatial operations on it. Instead, convert it to the value "POLYGON ((0.0 40.979898069620134, 22.5 40.979898069620134, 22.5 55.77657301866769, 0.0 55.77657301866769, 0.0 40.979898069620134))", which can be indexed as a geo_shape field.

Table 21. geo_grid processor options

NameRequiredDefaultDescription

field

yes

-

The field to interpret as a geo-tile. The field format is determined by the tile_type.

tile_type

yes

-

Three tile formats are understood: geohash, geotile and geohex.

target_field

no

field

The field to assign the polygon shape to, by default field is updated in-place.

parent_field

no

-

If specified and a parent tile exists, save that tile address to this field.

children_field

no

-

If specified and children tiles exist, save those tile addresses to this field as an array of strings.

non_children_field

no

-

If specified and intersecting non-child tiles exist, save their addresses to this field as an array of strings.

precision_field

no

-

If specified, save the tile precision (zoom) as an integer to this field.

ignore_missing

no

-

If true and field does not exist, the processor quietly exits without modifying the document.

target_format

no

“GeoJSON”

Which format to save the generated polygon in. Either WKT or GeoJSON.

description

no

-

Description of the processor. Useful for describing the purpose of the processor or its configuration.

if

no

-

Conditionally execute the processor. See Conditionally run a processor.

ignore_failure

no

false

Ignore failures for the processor. See Handling pipeline failures.

on_failure

no

-

Handle failures for the processor. See Handling pipeline failures.

tag

no

-

Identifier for the processor. Useful for debugging and metrics.

To demonstrate the usage of this ingest processor, consider an index called geocells with a mapping for a field geocell of type geo_shape. In order to populate that index using geotile and geohex fields, define two ingest processors:

  1. resp = client.indices.create(
  2. index="geocells",
  3. mappings={
  4. "properties": {
  5. "geocell": {
  6. "type": "geo_shape"
  7. }
  8. }
  9. },
  10. )
  11. print(resp)
  12. resp1 = client.ingest.put_pipeline(
  13. id="geotile2shape",
  14. description="translate rectangular z/x/y geotile to bounding box",
  15. processors=[
  16. {
  17. "geo_grid": {
  18. "field": "geocell",
  19. "tile_type": "geotile"
  20. }
  21. }
  22. ],
  23. )
  24. print(resp1)
  25. resp2 = client.ingest.put_pipeline(
  26. id="geohex2shape",
  27. description="translate H3 cell to polygon",
  28. processors=[
  29. {
  30. "geo_grid": {
  31. "field": "geocell",
  32. "tile_type": "geohex",
  33. "target_format": "wkt"
  34. }
  35. }
  36. ],
  37. )
  38. print(resp2)
  1. response = client.indices.create(
  2. index: 'geocells',
  3. body: {
  4. mappings: {
  5. properties: {
  6. geocell: {
  7. type: 'geo_shape'
  8. }
  9. }
  10. }
  11. }
  12. )
  13. puts response
  14. response = client.ingest.put_pipeline(
  15. id: 'geotile2shape',
  16. body: {
  17. description: 'translate rectangular z/x/y geotile to bounding box',
  18. processors: [
  19. {
  20. geo_grid: {
  21. field: 'geocell',
  22. tile_type: 'geotile'
  23. }
  24. }
  25. ]
  26. }
  27. )
  28. puts response
  29. response = client.ingest.put_pipeline(
  30. id: 'geohex2shape',
  31. body: {
  32. description: 'translate H3 cell to polygon',
  33. processors: [
  34. {
  35. geo_grid: {
  36. field: 'geocell',
  37. tile_type: 'geohex',
  38. target_format: 'wkt'
  39. }
  40. }
  41. ]
  42. }
  43. )
  44. puts response
  1. const response = await client.indices.create({
  2. index: "geocells",
  3. mappings: {
  4. properties: {
  5. geocell: {
  6. type: "geo_shape",
  7. },
  8. },
  9. },
  10. });
  11. console.log(response);
  12. const response1 = await client.ingest.putPipeline({
  13. id: "geotile2shape",
  14. description: "translate rectangular z/x/y geotile to bounding box",
  15. processors: [
  16. {
  17. geo_grid: {
  18. field: "geocell",
  19. tile_type: "geotile",
  20. },
  21. },
  22. ],
  23. });
  24. console.log(response1);
  25. const response2 = await client.ingest.putPipeline({
  26. id: "geohex2shape",
  27. description: "translate H3 cell to polygon",
  28. processors: [
  29. {
  30. geo_grid: {
  31. field: "geocell",
  32. tile_type: "geohex",
  33. target_format: "wkt",
  34. },
  35. },
  36. ],
  37. });
  38. console.log(response2);
  1. PUT geocells
  2. {
  3. "mappings": {
  4. "properties": {
  5. "geocell": {
  6. "type": "geo_shape"
  7. }
  8. }
  9. }
  10. }
  11. PUT _ingest/pipeline/geotile2shape
  12. {
  13. "description": "translate rectangular z/x/y geotile to bounding box",
  14. "processors": [
  15. {
  16. "geo_grid": {
  17. "field": "geocell",
  18. "tile_type": "geotile"
  19. }
  20. }
  21. ]
  22. }
  23. PUT _ingest/pipeline/geohex2shape
  24. {
  25. "description": "translate H3 cell to polygon",
  26. "processors": [
  27. {
  28. "geo_grid": {
  29. "field": "geocell",
  30. "tile_type": "geohex",
  31. "target_format": "wkt"
  32. }
  33. }
  34. ]
  35. }

These two pipelines can be used to index documents into the geocells index. The geocell field will be the string version of either a rectangular tile with format z/x/y or an H3 cell address, depending on which ingest processor we use when indexing the document. The resulting geometry will be represented and indexed as a geo_shape field in either GeoJSON or the Well-Known Text format.

Example: Rectangular geotile with envelope in GeoJSON

In this example a geocell field with a value defined in z/x/y format is indexed as a GeoJSON Envelope since the ingest-processor above was defined with default target_format.

  1. resp = client.index(
  2. index="geocells",
  3. id="1",
  4. pipeline="geotile2shape",
  5. document={
  6. "geocell": "4/8/5"
  7. },
  8. )
  9. print(resp)
  10. resp1 = client.get(
  11. index="geocells",
  12. id="1",
  13. )
  14. print(resp1)
  1. response = client.index(
  2. index: 'geocells',
  3. id: 1,
  4. pipeline: 'geotile2shape',
  5. body: {
  6. geocell: '4/8/5'
  7. }
  8. )
  9. puts response
  10. response = client.get(
  11. index: 'geocells',
  12. id: 1
  13. )
  14. puts response
  1. const response = await client.index({
  2. index: "geocells",
  3. id: 1,
  4. pipeline: "geotile2shape",
  5. document: {
  6. geocell: "4/8/5",
  7. },
  8. });
  9. console.log(response);
  10. const response1 = await client.get({
  11. index: "geocells",
  12. id: 1,
  13. });
  14. console.log(response1);
  1. PUT geocells/_doc/1?pipeline=geotile2shape
  2. {
  3. "geocell": "4/8/5"
  4. }
  5. GET geocells/_doc/1

The response shows how the ingest-processor has replaced the geocell field with an indexable geo_shape:

  1. {
  2. "_index": "geocells",
  3. "_id": "1",
  4. "_version": 1,
  5. "_seq_no": 0,
  6. "_primary_term": 1,
  7. "found": true,
  8. "_source": {
  9. "geocell": {
  10. "type": "Envelope",
  11. "coordinates": [
  12. [ 0.0, 55.77657301866769 ],
  13. [ 22.5, 40.979898069620134 ]
  14. ]
  15. }
  16. }
  17. }

Kibana map with showing the geotile at 4/8/5 and its four child cells

Example: Hexagonal geohex with polygon in WKT format

In this example a geocell field with an H3 string address is indexed as a WKT Polygon, since this ingest processor explicitly defined the target_format.

  1. resp = client.index(
  2. index="geocells",
  3. id="1",
  4. pipeline="geohex2shape",
  5. document={
  6. "geocell": "811fbffffffffff"
  7. },
  8. )
  9. print(resp)
  10. resp1 = client.get(
  11. index="geocells",
  12. id="1",
  13. )
  14. print(resp1)
  1. response = client.index(
  2. index: 'geocells',
  3. id: 1,
  4. pipeline: 'geohex2shape',
  5. body: {
  6. geocell: '811fbffffffffff'
  7. }
  8. )
  9. puts response
  10. response = client.get(
  11. index: 'geocells',
  12. id: 1
  13. )
  14. puts response
  1. const response = await client.index({
  2. index: "geocells",
  3. id: 1,
  4. pipeline: "geohex2shape",
  5. document: {
  6. geocell: "811fbffffffffff",
  7. },
  8. });
  9. console.log(response);
  10. const response1 = await client.get({
  11. index: "geocells",
  12. id: 1,
  13. });
  14. console.log(response1);
  1. PUT geocells/_doc/1?pipeline=geohex2shape
  2. {
  3. "geocell": "811fbffffffffff"
  4. }
  5. GET geocells/_doc/1

The response shows how the ingest-processor has replaced the geocell field with an indexable geo_shape:

  1. {
  2. "_index": "geocells",
  3. "_id": "1",
  4. "_version": 1,
  5. "_seq_no": 0,
  6. "_primary_term": 1,
  7. "found": true,
  8. "_source": {
  9. "geocell": "POLYGON ((1.1885095294564962 49.470279179513454, 2.0265689212828875 45.18424864858389, 7.509948452934623 43.786609335802495, 12.6773177459836 46.40695743262768, 12.345747342333198 50.55427505169064, 6.259687012061477 51.964770150370896, 3.6300085578113794 50.610463307239115, 1.1885095294564962 49.470279179513454))"
  10. }
  11. }

Kibana map with showing an H3 cell

Example: Enriched tile details

As described in geo_grid processor options, there are many other fields that can be set, which will enrich the information available. For example, with H3 tiles there are 7 child tiles, but only the first is fully contained by the parent. The remaining six are only partially overlapping the parent, and there exist a further six non-child tiles that overlap the parent. This can be investigated by adding parent and child additional fields to the ingest-processor:

  1. resp = client.ingest.put_pipeline(
  2. id="geohex2shape",
  3. description="translate H3 cell to polygon with enriched fields",
  4. processors=[
  5. {
  6. "geo_grid": {
  7. "description": "Ingest H3 cells like '811fbffffffffff' and create polygons",
  8. "field": "geocell",
  9. "tile_type": "geohex",
  10. "target_format": "wkt",
  11. "target_field": "shape",
  12. "parent_field": "parent",
  13. "children_field": "children",
  14. "non_children_field": "nonChildren",
  15. "precision_field": "precision"
  16. }
  17. }
  18. ],
  19. )
  20. print(resp)
  1. response = client.ingest.put_pipeline(
  2. id: 'geohex2shape',
  3. body: {
  4. description: 'translate H3 cell to polygon with enriched fields',
  5. processors: [
  6. {
  7. geo_grid: {
  8. description: "Ingest H3 cells like '811fbffffffffff' and create polygons",
  9. field: 'geocell',
  10. tile_type: 'geohex',
  11. target_format: 'wkt',
  12. target_field: 'shape',
  13. parent_field: 'parent',
  14. children_field: 'children',
  15. non_children_field: 'nonChildren',
  16. precision_field: 'precision'
  17. }
  18. }
  19. ]
  20. }
  21. )
  22. puts response
  1. const response = await client.ingest.putPipeline({
  2. id: "geohex2shape",
  3. description: "translate H3 cell to polygon with enriched fields",
  4. processors: [
  5. {
  6. geo_grid: {
  7. description:
  8. "Ingest H3 cells like '811fbffffffffff' and create polygons",
  9. field: "geocell",
  10. tile_type: "geohex",
  11. target_format: "wkt",
  12. target_field: "shape",
  13. parent_field: "parent",
  14. children_field: "children",
  15. non_children_field: "nonChildren",
  16. precision_field: "precision",
  17. },
  18. },
  19. ],
  20. });
  21. console.log(response);
  1. PUT _ingest/pipeline/geohex2shape
  2. {
  3. "description": "translate H3 cell to polygon with enriched fields",
  4. "processors": [
  5. {
  6. "geo_grid": {
  7. "description": "Ingest H3 cells like '811fbffffffffff' and create polygons",
  8. "field": "geocell",
  9. "tile_type": "geohex",
  10. "target_format": "wkt",
  11. "target_field": "shape",
  12. "parent_field": "parent",
  13. "children_field": "children",
  14. "non_children_field": "nonChildren",
  15. "precision_field": "precision"
  16. }
  17. }
  18. ]
  19. }

Index the document to see a different result:

  1. resp = client.index(
  2. index="geocells",
  3. id="1",
  4. pipeline="geohex2shape",
  5. document={
  6. "geocell": "811fbffffffffff"
  7. },
  8. )
  9. print(resp)
  10. resp1 = client.get(
  11. index="geocells",
  12. id="1",
  13. )
  14. print(resp1)
  1. response = client.index(
  2. index: 'geocells',
  3. id: 1,
  4. pipeline: 'geohex2shape',
  5. body: {
  6. geocell: '811fbffffffffff'
  7. }
  8. )
  9. puts response
  10. response = client.get(
  11. index: 'geocells',
  12. id: 1
  13. )
  14. puts response
  1. const response = await client.index({
  2. index: "geocells",
  3. id: 1,
  4. pipeline: "geohex2shape",
  5. document: {
  6. geocell: "811fbffffffffff",
  7. },
  8. });
  9. console.log(response);
  10. const response1 = await client.get({
  11. index: "geocells",
  12. id: 1,
  13. });
  14. console.log(response1);
  1. PUT geocells/_doc/1?pipeline=geohex2shape
  2. {
  3. "geocell": "811fbffffffffff"
  4. }
  5. GET geocells/_doc/1

The response from this index request:

  1. {
  2. "_index": "geocells",
  3. "_id": "1",
  4. "_version": 1,
  5. "_seq_no": 0,
  6. "_primary_term": 1,
  7. "found": true,
  8. "_source": {
  9. "parent": "801ffffffffffff",
  10. "geocell": "811fbffffffffff",
  11. "precision": 1,
  12. "shape": "POLYGON ((1.1885095294564962 49.470279179513454, 2.0265689212828875 45.18424864858389, 7.509948452934623 43.786609335802495, 12.6773177459836 46.40695743262768, 12.345747342333198 50.55427505169064, 6.259687012061477 51.964770150370896, 3.6300085578113794 50.610463307239115, 1.1885095294564962 49.470279179513454))",
  13. "children": [
  14. "821f87fffffffff",
  15. "821f8ffffffffff",
  16. "821f97fffffffff",
  17. "821f9ffffffffff",
  18. "821fa7fffffffff",
  19. "821faffffffffff",
  20. "821fb7fffffffff"
  21. ],
  22. "nonChildren": [
  23. "821ea7fffffffff",
  24. "82186ffffffffff",
  25. "82396ffffffffff",
  26. "821f17fffffffff",
  27. "821e37fffffffff",
  28. "82194ffffffffff"
  29. ]
  30. }
  31. }

This additional information will then enable, for example, creating a visualization of the H3 cell, its children and its intersecting non-children cells.

Kibana map with three H3 layers: cell