Foreach processor

Foreach processor

Runs an ingest processor on each element of an array or object.

All ingest processors can run on array or object elements. However, if the number of elements is unknown, it can be cumbersome to process each one in the same way.

The foreach processor lets you specify a field containing array or object values and a processor to run on each element in the field.

Table 20. Foreach Options

NameRequiredDefaultDescription

field

yes

-

Field containing array or object values.

processor

yes

-

Ingest processor to run on each element.

ignore_missing

no

false

If true, the processor silently exits without changing the document if the field is null or missing.

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.

Access keys and values

When iterating through an array or object, the foreach processor stores the current element’s value in the _ingest._value ingest metadata field. _ingest._value contains the entire element value, including any child fields. You can access child field values using dot notation on the _ingest._value field.

When iterating through an object, the foreach processor also stores the current element’s key as a string in _ingest._key.

You can access and change _ingest._key and _ingest._value in the processor. For an example, see the object example.

Failure handling

If the foreach processor fails to process an element and no on_failure processor is specified, the foreach processor silently exits. This leaves the entire array or object value unchanged.

Examples

The following examples show how you can use the foreach processor with different data types and options:

Array

Assume the following document:

  1. {
  2. "values" : ["foo", "bar", "baz"]
  3. }

When this foreach processor operates on this sample document:

  1. {
  2. "foreach" : {
  3. "field" : "values",
  4. "processor" : {
  5. "uppercase" : {
  6. "field" : "_ingest._value"
  7. }
  8. }
  9. }
  10. }

Then the document will look like this after processing:

  1. {
  2. "values" : ["FOO", "BAR", "BAZ"]
  3. }
Array of objects

Assume the following document:

  1. {
  2. "persons" : [
  3. {
  4. "id" : "1",
  5. "name" : "John Doe"
  6. },
  7. {
  8. "id" : "2",
  9. "name" : "Jane Doe"
  10. }
  11. ]
  12. }

In this case, the id field needs to be removed, so the following foreach processor is used:

  1. {
  2. "foreach" : {
  3. "field" : "persons",
  4. "processor" : {
  5. "remove" : {
  6. "field" : "_ingest._value.id"
  7. }
  8. }
  9. }
  10. }

After processing the result is:

  1. {
  2. "persons" : [
  3. {
  4. "name" : "John Doe"
  5. },
  6. {
  7. "name" : "Jane Doe"
  8. }
  9. ]
  10. }

For another array of objects example, refer to the attachment processor documentation.

Object

You can also use the foreach processor on object fields. For example, the following document contains a products field with object values.

  1. {
  2. "products" : {
  3. "widgets" : {
  4. "total_sales" : 50,
  5. "unit_price": 1.99,
  6. "display_name": ""
  7. },
  8. "sprockets" : {
  9. "total_sales" : 100,
  10. "unit_price": 9.99,
  11. "display_name": "Super Sprockets"
  12. },
  13. "whizbangs" : {
  14. "total_sales" : 200,
  15. "unit_price": 19.99,
  16. "display_name": "Wonderful Whizbangs"
  17. }
  18. }
  19. }

The following foreach processor changes the value of products.display_name to uppercase.

  1. {
  2. "foreach": {
  3. "field": "products",
  4. "processor": {
  5. "uppercase": {
  6. "field": "_ingest._value.display_name"
  7. }
  8. }
  9. }
  10. }

When run on the document, the foreach processor returns:

  1. {
  2. "products" : {
  3. "widgets" : {
  4. "total_sales" : 50,
  5. "unit_price" : 1.99,
  6. "display_name" : ""
  7. },
  8. "sprockets" : {
  9. "total_sales" : 100,
  10. "unit_price" : 9.99,
  11. "display_name" : "SUPER SPROCKETS"
  12. },
  13. "whizbangs" : {
  14. "total_sales" : 200,
  15. "unit_price" : 19.99,
  16. "display_name" : "WONDERFUL WHIZBANGS"
  17. }
  18. }
  19. }

The following foreach processor sets each element’s key to the value of products.display_name. If products.display_name contains an empty string, the processor deletes the element.

  1. {
  2. "foreach": {
  3. "field": "products",
  4. "processor": {
  5. "set": {
  6. "field": "_ingest._key",
  7. "value": "{{_ingest._value.display_name}}"
  8. }
  9. }
  10. }
  11. }

When run on the previous document, the foreach processor returns:

  1. {
  2. "products" : {
  3. "Wonderful Whizbangs" : {
  4. "total_sales" : 200,
  5. "unit_price" : 19.99,
  6. "display_name" : "Wonderful Whizbangs"
  7. },
  8. "Super Sprockets" : {
  9. "total_sales" : 100,
  10. "unit_price" : 9.99,
  11. "display_name" : "Super Sprockets"
  12. }
  13. }
  14. }
Failure handling

The wrapped processor can have a on_failure definition. For example, the id field may not exist on all person objects. Instead of failing the index request, you can use an on_failure block to send the document to the failure_index index for later inspection:

  1. {
  2. "foreach" : {
  3. "field" : "persons",
  4. "processor" : {
  5. "remove" : {
  6. "field" : "_value.id",
  7. "on_failure" : [
  8. {
  9. "set" : {
  10. "field": "_index",
  11. "value": "failure_index"
  12. }
  13. }
  14. ]
  15. }
  16. }
  17. }
  18. }

In this example, if the remove processor does fail, then the array elements that have been processed thus far will be updated.