Optimistic concurrency control

Optimistic concurrency control

Elasticsearch is distributed. When documents are created, updated, or deleted, the new version of the document has to be replicated to other nodes in the cluster. Elasticsearch is also asynchronous and concurrent, meaning that these replication requests are sent in parallel, and may arrive at their destination out of sequence. Elasticsearch needs a way of ensuring that an older version of a document never overwrites a newer version.

To ensure an older version of a document doesn’t overwrite a newer version, every operation performed to a document is assigned a sequence number by the primary shard that coordinates that change. The sequence number is increased with each operation and thus newer operations are guaranteed to have a higher sequence number than older operations. Elasticsearch can then use the sequence number of operations to make sure a newer document version is never overridden by a change that has a smaller sequence number assigned to it.

For example, the following indexing command will create a document and assign it an initial sequence number and primary term:

  1. resp = client.index(
  2. index="products",
  3. id="1567",
  4. document={
  5. "product": "r2d2",
  6. "details": "A resourceful astromech droid"
  7. },
  8. )
  9. print(resp)
  1. response = client.index(
  2. index: 'products',
  3. id: 1567,
  4. body: {
  5. product: 'r2d2',
  6. details: 'A resourceful astromech droid'
  7. }
  8. )
  9. puts response
  1. const response = await client.index({
  2. index: "products",
  3. id: 1567,
  4. document: {
  5. product: "r2d2",
  6. details: "A resourceful astromech droid",
  7. },
  8. });
  9. console.log(response);
  1. PUT products/_doc/1567
  2. {
  3. "product" : "r2d2",
  4. "details" : "A resourceful astromech droid"
  5. }

You can see the assigned sequence number and primary term in the _seq_no and _primary_term fields of the response:

  1. {
  2. "_shards": {
  3. "total": 2,
  4. "failed": 0,
  5. "successful": 1
  6. },
  7. "_index": "products",
  8. "_id": "1567",
  9. "_version": 1,
  10. "_seq_no": 362,
  11. "_primary_term": 2,
  12. "result": "created"
  13. }

Elasticsearch keeps tracks of the sequence number and primary term of the last operation to have changed each of the documents it stores. The sequence number and primary term are returned in the _seq_no and _primary_term fields in the response of the GET API:

  1. resp = client.get(
  2. index="products",
  3. id="1567",
  4. )
  5. print(resp)
  1. response = client.get(
  2. index: 'products',
  3. id: 1567
  4. )
  5. puts response
  1. const response = await client.get({
  2. index: "products",
  3. id: 1567,
  4. });
  5. console.log(response);
  1. GET products/_doc/1567

returns:

  1. {
  2. "_index": "products",
  3. "_id": "1567",
  4. "_version": 1,
  5. "_seq_no": 362,
  6. "_primary_term": 2,
  7. "found": true,
  8. "_source": {
  9. "product": "r2d2",
  10. "details": "A resourceful astromech droid"
  11. }
  12. }

Note: The Search API can return the _seq_no and _primary_term for each search hit by setting seq_no_primary_term parameter.

The sequence number and the primary term uniquely identify a change. By noting down the sequence number and primary term returned, you can make sure to only change the document if no other change was made to it since you retrieved it. This is done by setting the if_seq_no and if_primary_term parameters of the index API, update API, or delete API.

For example, the following indexing call will make sure to add a tag to the document without losing any potential change to the description or an addition of another tag by another API:

  1. resp = client.index(
  2. index="products",
  3. id="1567",
  4. if_seq_no="362",
  5. if_primary_term="2",
  6. document={
  7. "product": "r2d2",
  8. "details": "A resourceful astromech droid",
  9. "tags": [
  10. "droid"
  11. ]
  12. },
  13. )
  14. print(resp)
  1. const response = await client.index({
  2. index: "products",
  3. id: 1567,
  4. if_seq_no: 362,
  5. if_primary_term: 2,
  6. document: {
  7. product: "r2d2",
  8. details: "A resourceful astromech droid",
  9. tags: ["droid"],
  10. },
  11. });
  12. console.log(response);
  1. PUT products/_doc/1567?if_seq_no=362&if_primary_term=2
  2. {
  3. "product": "r2d2",
  4. "details": "A resourceful astromech droid",
  5. "tags": [ "droid" ]
  6. }