2025-03-27 PostgreSQL deadlock solution

 

Scenario: Task Queue Processing

Multiple workers need to process tasks from a pending_tasks table without conflicts.

1. Table Setup

2. Worker 1 (Session A)

Must use

ORDER BY task_id

and

FOR UPDATE SKIP LOCKED

3. Worker 2 (Session B)

4. Update and Commit

Add commit after every update

Worker 1:

Worker 2:

Key Features Demonstrated

  1. Non-blocking: Workers don't wait for each other's locks

  2. Fair ordering: ORDER BY task_id ensures oldest tasks are processed first

  3. Atomic claims: Each task is guaranteed to be processed by only one worker

Does not work with GROUP BY - Alternative

To use FOR UPDATE SKIP LOCKED with GROUP BY in PostgreSQL, you cannot directly combine these clauses because GROUP BY requires aggregate functions and FOR UPDATE locks individual rows, not grouped results37. However, there are workarounds:

Workaround Using Advisory Locks

For scenarios requiring exclusive access to grouped data (e.g., all jobs for a specific person_id), use advisory locks to coordinate at the application level:

How it works:

  1. pg_try_advisory_xact_lock(person_id) attempts to lock the person_id without waiting3.

  2. If the lock succeeds (true), the worker processes all jobs for that person_id.

  3. If the lock fails (false), the worker skips this person_id and continues to the next available one.

Limitations of FOR UPDATE SKIP LOCKED

Alternative Approach

If you need to process all jobs for a group (e.g., all pending jobs for a person_id), use a subquery to first identify the group, then lock its rows: