Skip to content

Latest commit

 

History

History
57 lines (40 loc) · 2.63 KB

README.md

File metadata and controls

57 lines (40 loc) · 2.63 KB

Concurrent updates

This program simulates 2 concurrent transactions that try to increment the same counter (initialized with 1) in a postgres table under a given isolation level.

Setup

  1. Install packages: pip install -r requirements.txt
  2. Start a postgres database with the following command: docker run --rm -e POSTGRES_PASSWORD=postgres -p 5432:5432 postgres

Usage

usage: concurrent_updates.py [-h] [--serial] {ru,rc,rr,s}

positional arguments:
  {ru,rc,rr,s}  isolation level: ru (read uncomitted) rc (read committed) rr
                (repeatable reads) s (serializable)

options:
  -h, --help    show this help message and exit
  --serial, -s  whether to serialize transaction execution or interleave them.
                Serialized execution will never cause a serialization error

Explanation

When the command is run with the --serial flag, the operations from the 2 transactions (T1 amd T2) will be performed on the counter in the following order:

  1. T1 reads counter
  2. T1 increments counters and commits
  3. T2 reads counter
  4. T2 increments counters and commits

This causes no serialization errors, and the final value of the counter is the correct one (3) under any isolation level.

When the command is run without the --serial flag, the operations from T1 and T2 are interleaved and executed in the following order:

  1. T1 reads counter
  2. T2 reads counter
  3. T1 increments counters and commits
  4. T2 increments counters and commits

And this will result in the following behavior depending on the isolation level:

Isolation level Final counter value DB exception
Read uncommitted 2 (wrong ❌) None
Read committed 2 (wrong ❌) None
Repeatable read 2 (correct ✅) T2 aborted with SerializationFailure
Serializable 2 (correct ✅) T2 aborted with SerializationFailure

As you can see, under the Read uncommitted and Read committed isolation levels, the invalid sequence of operation resulting in the wrong result (final counter value 2 instead of 3) is let go through. On the other hand, under the Repeatable read and Serializable isolation levels, the invalid sequence of operation triggers a SerializationFailure error when T2 tries to update the counter - T2 is aborted and the final counter value (although still 2) is correct, because only T1 has committed and incremented the counter.