[[partial-updates]] === Partial Updates to Documents
In <update
API, we can make partial updates like incrementing a counter in a
single request.
We also said that documents are immutable: they cannot be changed, only
replaced. The update
API must obey the same rules. Externally, it
appears as though we are partially updating a document in place. Internally,
however, the update
API simply manages the same retrieve-change-reindex
process that we have already described. The difference is that this process
happens within a shard, thus avoiding the network overhead of multiple
requests. By reducing the time between the retrieve and reindex steps, we
also reduce the likelihood of there being conflicting changes from other
processes.
The simplest form of the update
request accepts a partial document as the
doc
parameter, which just gets merged with the existing document. Objects
are merged together, existing scalar fields are overwritten, and new fields are
added. For instance, we could add a tags
field and a views
field to our
blog post as follows:
POST /website/blog/1/_update { "doc" : { "tags" : [ "testing" ], "views": 0 }
// SENSE: 030_Data/45_Partial_update.json
If the request succeeds, we see a response similar to that
of the index
request:
{ "_index" : "website", "_id" : "1", "_type" : "blog", "_version" : 3
Retrieving the document shows the updated _source
field:
{ "_index": "website", "_type": "blog", "_id": "1", "_version": 3, "found": true, "_source": { "title": "My first blog entry", "text": "Starting to get the hang of this...", "tags": [ "testing" ], <1> "views": 0 <1> }
// SENSE: 030_Data/45_Partial_update.json
<1> Our new fields have been added to the _source
.
==== Using Scripts to Make Partial Updates
Scripts can be used in the update
API to change the contents of the _source
field, which ((("_source field", sortas="source field")))is referred to inside an update script as ctx._source
. For
instance, we could use a script to increment the number of views
that our
blog post has had:
POST /website/blog/1/_update { "script" : "ctx._source.views+=1"
// SENSE: 030_Data/45_Partial_update.json
.Scripting with Groovy
For those ((("documents", "partial updates", "using scripts")))((("updating documents", "partial updates", "using scripts")))moments when the API just isn't enough, Elasticsearch allows you to
write your own custom logic in a script.((("scripts", "using to make partial updates"))) Scripting is supported in many APIs
including search, sorting, aggregations, and document updates. Scripts can be
passed in as part of the request, retrieved from the special .scripts
index, or loaded from disk.
The default scripting language ((("Groovy")))is a http://groovy.codehaus.org/[Groovy], a fast and expressive scripting language, similar in syntax to JavaScript. It runs in a sandbox to prevent malicious users from breaking out of Elasticsearch and attacking the server.
You can read more about scripting in the http://www.elasticsearch.org/guide/en/elasticsearch/reference/current/modules-scripting.html[scripting reference documentation].
We can also use a script to add a new tag to the tags
array. In this
example we specify the new tag as a parameter rather than hardcoding it in
the script itself. This allows Elasticsearch to reuse the script in the
future, without having to compile a new script every time we want to add
another tag:
POST /website/blog/1/_update { "script" : "ctx._source.tags+=new_tag", "params" : { "new_tag" : "search" }
// SENSE: 030_Data/45_Partial_update.json
Fetching the document shows the effect of the last two requests:
{ "_index": "website", "_type": "blog", "_id": "1", "_version": 5, "found": true, "_source": { "title": "My first blog entry", "text": "Starting to get the hang of this...", "tags": ["testing", "search"], <1> "views": 1 <2> }
<1> The search
tag has been appended to the tags
array.
<2> The views
field has been incremented.
We can even choose to delete a document based on its contents,
by setting ctx.op
to delete
:
POST /website/blog/1/_update { "script" : "ctx.op = ctx._source.views == count ? 'delete' : 'none'", "params" : { "count": 1 }
// SENSE: 030_Data/45_Partial_update.json
==== Updating a Document That May Not Yet Exist
Imagine that we need to store a((("updating documents", "that don't already exist"))) page view counter in Elasticsearch. Every time that a user views a page, we increment the counter for that page. But if it is a new page, we can't be sure that the counter already exists. If we try to update a nonexistent document, the update will fail.
In cases like these, we can use((("upsert parameter"))) the upsert
parameter to specify the
document that should be created if it doesn't already exist:
POST /website/pageviews/1/_update { "script" : "ctx._source.views+=1", "upsert": { "views": 1 }
// SENSE: 030_Data/45_Upsert.json
The first time we run this request, the upsert
value is indexed as a new
document, which initializes the views
field to 1
. On subsequent runs, the
document already exists, so the script
update is applied instead,
incrementing the views
counter.
==== Updates and Conflicts
In the introduction to this section, we said((("updating documents", "conflicts and")))((("conflicts", "updates and"))) that the smaller the window between
the retrieve and reindex steps, the smaller the opportunity for
conflicting changes. But it doesn't eliminate the possibility completely. It
is still possible that a request from another process could change the
document before update
has managed to reindex it.
To avoid losing data, the update
API retrieves the current _version
of the document in the retrieve step, and passes that to the index
request
during the reindex step.
If another process has changed the document between retrieve and reindex,
then the _version
number won't match and the update request will fail.
For many uses of partial update, it doesn't matter that a document has been changed. For instance, if two processes are both incrementing the page-view counter, it doesn't matter in which order it happens; if a conflict occurs, the only thing we need to do is reattempt the update.
This can be done automatically by((("query strings", "retry_on_conflict parameter")))((("retry_on_conflict parameter"))) setting the retry_on_conflict
parameter to
the number of times that update
should retry before failing; it defaults
to 0
.
POST /website/pageviews/1/_update?retry_on_conflict=5 <1> { "script" : "ctx._source.views+=1", "upsert": { "views": 0 }
// SENSE: 030_Data/45_Upsert.json
<1> Retry this update five times before failing.
This works well for operations such as incrementing a counter, where the order of
increments does not matter, but in other situations the order of
changes is important. Like the <update
API
adopts a last-write-wins approach by default, but it also accepts a
version
parameter that allows you to use
<