Define runtime fields in a search request

edit

Define runtime fields in a search request

edit

You can specify a runtime_mappings section in a search request to create runtime fields that exist only as part of the query. You specify a script as part of the runtime_mappings section, just as you would if adding a runtime field to the mappings.

Defining a runtime field in a search request uses the same format as defining a runtime field in the index mapping. Just copy the field definition from the runtime in the index mapping to the runtime_mappings section of the search request.

The following search request adds a day_of_week field to the runtime_mappings section. The field values will be calculated dynamically, and only within the context of this search request:

resp = client.search(
    index="my-index-000001",
    runtime_mappings={
        "day_of_week": {
            "type": "keyword",
            "script": {
                "source": "emit(doc['@timestamp'].value.dayOfWeekEnum.getDisplayName(TextStyle.FULL, Locale.ENGLISH))"
            }
        }
    },
    aggs={
        "day_of_week": {
            "terms": {
                "field": "day_of_week"
            }
        }
    },
)
print(resp)
GET my-index-000001/_search
{
  "runtime_mappings": {
    "day_of_week": {
      "type": "keyword",
      "script": {
        "source": "emit(doc['@timestamp'].value.dayOfWeekEnum.getDisplayName(TextStyle.FULL, Locale.ENGLISH))"
      }
    }
  },
  "aggs": {
    "day_of_week": {
      "terms": {
        "field": "day_of_week"
      }
    }
  }
}

Create runtime fields that use other runtime fields

edit

You can even define runtime fields in a search request that return values from other runtime fields. For example, let’s say you bulk index some sensor data:

resp = client.bulk(
    index="my-index-000001",
    refresh=True,
    operations=[
        {
            "index": {}
        },
        {
            "@timestamp": 1516729294000,
            "model_number": "QVKC92Q",
            "measures": {
                "voltage": "5.2",
                "start": "300",
                "end": "8675309"
            }
        },
        {
            "index": {}
        },
        {
            "@timestamp": 1516642894000,
            "model_number": "QVKC92Q",
            "measures": {
                "voltage": "5.8",
                "start": "300",
                "end": "8675309"
            }
        },
        {
            "index": {}
        },
        {
            "@timestamp": 1516556494000,
            "model_number": "QVKC92Q",
            "measures": {
                "voltage": "5.1",
                "start": "300",
                "end": "8675309"
            }
        },
        {
            "index": {}
        },
        {
            "@timestamp": 1516470094000,
            "model_number": "QVKC92Q",
            "measures": {
                "voltage": "5.6",
                "start": "300",
                "end": "8675309"
            }
        },
        {
            "index": {}
        },
        {
            "@timestamp": 1516383694000,
            "model_number": "HG537PU",
            "measures": {
                "voltage": "4.2",
                "start": "400",
                "end": "8625309"
            }
        },
        {
            "index": {}
        },
        {
            "@timestamp": 1516297294000,
            "model_number": "HG537PU",
            "measures": {
                "voltage": "4.0",
                "start": "400",
                "end": "8625309"
            }
        }
    ],
)
print(resp)
response = client.bulk(
  index: 'my-index-000001',
  refresh: true,
  body: [
    {
      index: {}
    },
    {
      "@timestamp": 1_516_729_294_000,
      model_number: 'QVKC92Q',
      measures: {
        voltage: '5.2',
        start: '300',
        end: '8675309'
      }
    },
    {
      index: {}
    },
    {
      "@timestamp": 1_516_642_894_000,
      model_number: 'QVKC92Q',
      measures: {
        voltage: '5.8',
        start: '300',
        end: '8675309'
      }
    },
    {
      index: {}
    },
    {
      "@timestamp": 1_516_556_494_000,
      model_number: 'QVKC92Q',
      measures: {
        voltage: '5.1',
        start: '300',
        end: '8675309'
      }
    },
    {
      index: {}
    },
    {
      "@timestamp": 1_516_470_094_000,
      model_number: 'QVKC92Q',
      measures: {
        voltage: '5.6',
        start: '300',
        end: '8675309'
      }
    },
    {
      index: {}
    },
    {
      "@timestamp": 1_516_383_694_000,
      model_number: 'HG537PU',
      measures: {
        voltage: '4.2',
        start: '400',
        end: '8625309'
      }
    },
    {
      index: {}
    },
    {
      "@timestamp": 1_516_297_294_000,
      model_number: 'HG537PU',
      measures: {
        voltage: '4.0',
        start: '400',
        end: '8625309'
      }
    }
  ]
)
puts response
const response = await client.bulk({
  index: "my-index-000001",
  refresh: "true",
  operations: [
    {
      index: {},
    },
    {
      "@timestamp": 1516729294000,
      model_number: "QVKC92Q",
      measures: {
        voltage: "5.2",
        start: "300",
        end: "8675309",
      },
    },
    {
      index: {},
    },
    {
      "@timestamp": 1516642894000,
      model_number: "QVKC92Q",
      measures: {
        voltage: "5.8",
        start: "300",
        end: "8675309",
      },
    },
    {
      index: {},
    },
    {
      "@timestamp": 1516556494000,
      model_number: "QVKC92Q",
      measures: {
        voltage: "5.1",
        start: "300",
        end: "8675309",
      },
    },
    {
      index: {},
    },
    {
      "@timestamp": 1516470094000,
      model_number: "QVKC92Q",
      measures: {
        voltage: "5.6",
        start: "300",
        end: "8675309",
      },
    },
    {
      index: {},
    },
    {
      "@timestamp": 1516383694000,
      model_number: "HG537PU",
      measures: {
        voltage: "4.2",
        start: "400",
        end: "8625309",
      },
    },
    {
      index: {},
    },
    {
      "@timestamp": 1516297294000,
      model_number: "HG537PU",
      measures: {
        voltage: "4.0",
        start: "400",
        end: "8625309",
      },
    },
  ],
});
console.log(response);
POST my-index-000001/_bulk?refresh=true
{"index":{}}
{"@timestamp":1516729294000,"model_number":"QVKC92Q","measures":{"voltage":"5.2","start": "300","end":"8675309"}}
{"index":{}}
{"@timestamp":1516642894000,"model_number":"QVKC92Q","measures":{"voltage":"5.8","start": "300","end":"8675309"}}
{"index":{}}
{"@timestamp":1516556494000,"model_number":"QVKC92Q","measures":{"voltage":"5.1","start": "300","end":"8675309"}}
{"index":{}}
{"@timestamp":1516470094000,"model_number":"QVKC92Q","measures":{"voltage":"5.6","start": "300","end":"8675309"}}
{"index":{}}
{"@timestamp":1516383694000,"model_number":"HG537PU","measures":{"voltage":"4.2","start": "400","end":"8625309"}}
{"index":{}}
{"@timestamp":1516297294000,"model_number":"HG537PU","measures":{"voltage":"4.0","start": "400","end":"8625309"}}

You realize after indexing that your numeric data was mapped as type text. You want to aggregate on the measures.start and measures.end fields, but the aggregation fails because you can’t aggregate on fields of type text. Runtime fields to the rescue! You can add runtime fields with the same name as your indexed fields and modify the data type:

resp = client.indices.put_mapping(
    index="my-index-000001",
    runtime={
        "measures.start": {
            "type": "long"
        },
        "measures.end": {
            "type": "long"
        }
    },
)
print(resp)
response = client.indices.put_mapping(
  index: 'my-index-000001',
  body: {
    runtime: {
      'measures.start' => {
        type: 'long'
      },
      'measures.end' => {
        type: 'long'
      }
    }
  }
)
puts response
const response = await client.indices.putMapping({
  index: "my-index-000001",
  runtime: {
    "measures.start": {
      type: "long",
    },
    "measures.end": {
      type: "long",
    },
  },
});
console.log(response);
PUT my-index-000001/_mapping
{
  "runtime": {
    "measures.start": {
      "type": "long"
    },
    "measures.end": {
      "type": "long"
    }
  }
}

Runtime fields take precedence over fields defined with the same name in the index mappings. This flexibility allows you to shadow existing fields and calculate a different value, without modifying the field itself. If you made a mistake in your index mapping, you can use runtime fields to calculate values that override values in the mapping during the search request.

Now, you can easily run an average aggregation on the measures.start and measures.end fields:

resp = client.search(
    index="my-index-000001",
    aggs={
        "avg_start": {
            "avg": {
                "field": "measures.start"
            }
        },
        "avg_end": {
            "avg": {
                "field": "measures.end"
            }
        }
    },
)
print(resp)
response = client.search(
  index: 'my-index-000001',
  body: {
    aggregations: {
      avg_start: {
        avg: {
          field: 'measures.start'
        }
      },
      avg_end: {
        avg: {
          field: 'measures.end'
        }
      }
    }
  }
)
puts response
const response = await client.search({
  index: "my-index-000001",
  aggs: {
    avg_start: {
      avg: {
        field: "measures.start",
      },
    },
    avg_end: {
      avg: {
        field: "measures.end",
      },
    },
  },
});
console.log(response);
GET my-index-000001/_search
{
  "aggs": {
    "avg_start": {
      "avg": {
        "field": "measures.start"
      }
    },
    "avg_end": {
      "avg": {
        "field": "measures.end"
      }
    }
  }
}

The response includes the aggregation results without changing the values for the underlying data:

{
  "aggregations" : {
    "avg_start" : {
      "value" : 333.3333333333333
    },
    "avg_end" : {
      "value" : 8658642.333333334
    }
  }
}

Further, you can define a runtime field as part of a search query that calculates a value, and then run a stats aggregation on that field in the same query.

The duration runtime field doesn’t exist in the index mapping, but we can still search and aggregate on that field. The following query returns the calculated value for the duration field and runs a stats aggregation to compute statistics over numeric values extracted from the aggregated documents.

resp = client.search(
    index="my-index-000001",
    runtime_mappings={
        "duration": {
            "type": "long",
            "script": {
                "source": "\n          emit(doc['measures.end'].value - doc['measures.start'].value);\n          "
            }
        }
    },
    aggs={
        "duration_stats": {
            "stats": {
                "field": "duration"
            }
        }
    },
)
print(resp)
response = client.search(
  index: 'my-index-000001',
  body: {
    runtime_mappings: {
      duration: {
        type: 'long',
        script: {
          source: "\n          emit(doc['measures.end'].value - doc['measures.start'].value);\n          "
        }
      }
    },
    aggregations: {
      duration_stats: {
        stats: {
          field: 'duration'
        }
      }
    }
  }
)
puts response
const response = await client.search({
  index: "my-index-000001",
  runtime_mappings: {
    duration: {
      type: "long",
      script: {
        source:
          "\n          emit(doc['measures.end'].value - doc['measures.start'].value);\n          ",
      },
    },
  },
  aggs: {
    duration_stats: {
      stats: {
        field: "duration",
      },
    },
  },
});
console.log(response);
GET my-index-000001/_search
{
  "runtime_mappings": {
    "duration": {
      "type": "long",
      "script": {
        "source": """
          emit(doc['measures.end'].value - doc['measures.start'].value);
          """
      }
    }
  },
  "aggs": {
    "duration_stats": {
      "stats": {
        "field": "duration"
      }
    }
  }
}

Even though the duration runtime field only exists in the context of a search query, you can search and aggregate on that field. This flexibility is incredibly powerful, enabling you to rectify mistakes in your index mappings and dynamically complete calculations all within a single search request.

{
  "aggregations" : {
    "duration_stats" : {
      "count" : 6,
      "min" : 8624909.0,
      "max" : 8675009.0,
      "avg" : 8658309.0,
      "sum" : 5.1949854E7
    }
  }
}