| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261 | [[modules-scripting-fields]]== Accessing document fields and special variablesDepending on where a script is used, it will have access to certain specialvariables and document fields.[discrete]== Update scriptsA script used in the <<docs-update,update>>,<<docs-update-by-query,update-by-query>>, or <<docs-reindex,reindex>>API will have access to the `ctx` variable which exposes:[horizontal]`ctx._source`::     Access to the document <<mapping-source-field,`_source` field>>.`ctx.op`::          The operation that should be applied to the document: `index` or `delete`.`ctx._index` etc::  Access to <<mapping-fields,document metadata fields>>, some of which may be read-only.[discrete]== Search and aggregation scriptsWith the exception of <<script-fields,script fields>> which areexecuted once per search hit, scripts used in search and aggregations will beexecuted once for every document which might match a query or an aggregation.Depending on how many documents you have, this could mean millions or billionsof executions: these scripts need to be fast!Field values can be accessed from a script using<<modules-scripting-doc-vals,doc-values>>,<<modules-scripting-source, the `_source` field>>, or<<modules-scripting-stored, stored fields>>,each of which is explained below.[[scripting-score]][discrete]=== Accessing the score of a document within a scriptScripts used in the <<query-dsl-function-score-query,`function_score` query>>,in <<sort-search-results,script-based sorting>>, or in<<search-aggregations,aggregations>> have access to the `_score` variable whichrepresents the current relevance score of a document.Here's an example of using a script in a<<query-dsl-function-score-query,`function_score` query>> to alter therelevance `_score` of each document:[source,console]-------------------------------------PUT my-index-000001/_doc/1?refresh{  "text": "quick brown fox",  "popularity": 1}PUT my-index-000001/_doc/2?refresh{  "text": "quick fox",  "popularity": 5}GET my-index-000001/_search{  "query": {    "function_score": {      "query": {        "match": {          "text": "quick brown fox"        }      },      "script_score": {        "script": {          "lang": "expression",          "source": "_score * doc['popularity']"        }      }    }  }}-------------------------------------[discrete][[modules-scripting-doc-vals]]=== Doc valuesBy far the fastest most efficient way to access a field value from ascript is to use the `doc['field_name']` syntax, which retrieves the fieldvalue from <<doc-values,doc values>>. Doc values are a columnar field valuestore, enabled by default on all fields except for <<text,analyzed `text` fields>>.[source,console]-------------------------------PUT my-index-000001/_doc/1?refresh{  "cost_price": 100}GET my-index-000001/_search{  "script_fields": {    "sales_price": {      "script": {        "lang":   "expression",        "source": "doc['cost_price'] * markup",        "params": {          "markup": 0.2        }      }    }  }}-------------------------------Doc-values can only return "simple" field values like numbers, dates, geo-points, terms, etc, or arrays of these values if the field is multi-valued.It cannot return JSON objects.[NOTE].Missing fields===================================================The `doc['field']` will throw an error if `field` is missing from the mappings.In `painless`, a check can first be done with `doc.containsKey('field')` to guardaccessing the `doc` map. Unfortunately, there is no way to check for theexistence of the field in mappings in an `expression` script.===================================================[NOTE].Doc values and `text` fields===================================================The `doc['field']` syntax can also be used for <<text,analyzed `text` fields>>if <<fielddata,`fielddata`>> is enabled, but *BEWARE*: enabling fielddata on a`text` field requires loading all of the terms into the JVM heap, which can bevery expensive both in terms of memory and CPU. It seldom makes sense toaccess `text` fields from scripts.===================================================[discrete][[modules-scripting-source]]=== The document `_source`The document <<mapping-source-field,`_source`>> can be accessed using the`_source.field_name` syntax. The `_source` is loaded as a map-of-maps, soproperties within object fields can be accessed as, for example,`_source.name.first`.[IMPORTANT].Prefer doc-values to _source=========================================================Accessing the `_source` field is much slower than using doc-values. The_source field is optimised for returning several fields per result, while docvalues are optimised for accessing the value of a specific field in manydocuments.It makes sense to use `_source` when generating a<<script-fields,script field>> for the top ten hits from asearch result but, for other search and aggregation use cases, always preferusing doc values.=========================================================For instance:[source,console]-------------------------------PUT my-index-000001{  "mappings": {    "properties": {      "first_name": {        "type": "text"      },      "last_name": {        "type": "text"      }    }  }}PUT my-index-000001/_doc/1?refresh{  "first_name": "Barry",  "last_name": "White"}GET my-index-000001/_search{  "script_fields": {    "full_name": {      "script": {        "lang": "painless",        "source": "params._source.first_name + ' ' + params._source.last_name"      }    }  }}-------------------------------[discrete][[modules-scripting-stored]]=== Stored fields_Stored fields_ -- fields explicitly marked as<<mapping-store,`"store": true`>> in the mapping -- can be accessed using the`_fields['field_name'].value` or `_fields['field_name']` syntax:[source,console]-------------------------------PUT my-index-000001{  "mappings": {    "properties": {      "full_name": {        "type": "text",        "store": true      },      "title": {        "type": "text",        "store": true      }    }  }}PUT my-index-000001/_doc/1?refresh{  "full_name": "Alice Ball",  "title": "Professor"}GET my-index-000001/_search{  "script_fields": {    "name_with_title": {      "script": {        "lang": "painless",        "source": "params._fields['title'].value + ' ' + params._fields['full_name'].value"      }    }  }}-------------------------------[TIP].Stored vs `_source`=======================================================The `_source` field is just a special stored field, so the performance issimilar to that of other stored fields. The `_source` provides access to theoriginal document body that was indexed (including the ability to distinguish`null` values from empty fields, single-value arrays from plain scalars, etc).The only time it really makes sense to use stored fields instead of the`_source` field is when the `_source` is very large and it is less costly toaccess a few small stored fields instead of the entire `_source`.=======================================================
 |