Skip to main content

Lock mechanism implemented with PostgreSQL advisory locks.

Project description

postgresql-lock

Lock mechanism implemented with PostgreSQL advisory locks.

Easily implement distributed database locking.

Install

pip install postgresql-lock

Supported database interfaces

  • asyncpg
    • asynchronous
  • psycopg2
    • synchronous
  • psycopg3
    • asynchronous
    • synchronous
  • sqlalchemy (supports version 1 & 2; can use any underlying database interface)
    • asynchronous
    • synchronous

Why would I use this?

  • PostgreSQL table locks aren't sufficient for your use-case
  • PostgreSQL row locks don't work on INSERT
  • You want to prevent race conditions between INSERT and UPDATE on the same primary key
  • None of the aforementioned details fit your use-case, but you have PostgreSQL installed and need to prevent race conditions in a distributed system

Default operation

By default postgresql-lock will use session lock scope in blocking mode with rollback_on_error enabled. The session lock scope means only a single database connection can acquire the lock at a time.

Usage

All work revolves around the Lock class.

The easiest way to use Lock is with with or async with statements. The lock will be released automatically. If rollback_on_error is enabled (default), rollbacks are automatically handled prior to release.

Using with and async with implies blocking mode.

from postgresql_lock import Lock

# setup connection
conn = ...

# create and use lock
with Lock(conn, "shared-identifier"):
    print("Acquired lock!")

    # do something here

Now compare the above example to the equivalent try/finally example below:

from postgresql_lock import Lock

# setup connection
conn = ...

# create lock
lock = Lock(conn, "shared-identifier")

try:
    # acquire lock
    lock.acquire()

    print("Acquired lock!")

    try:
        # do something here
        pass

    except Exception as exc:
        # handle_error() will rollback the transaction by default
        lock.handle_error(exc)

        raise exc
finally:
    # release lock (this is safe to run even if the lock has not been acquired)
    lock.release()

Asynchronous usage (without async with)

from postgresql_lock import Lock

# setup connection
conn = ...

# create lock
lock = Lock(conn, "shared-identifier")

try:
    # acquire lock
    await lock.acquire_async()

    print("Acquired lock!")

    try:
        # do something here
        pass

    except Exception as exc:
        # handle_error_async() will rollback the transaction by default
        await lock.handle_error_async(exc)

        raise exc
finally:
    # release lock (this is safe to run even if the lock has not been acquired)
    await lock.release_async()

Non-blocking mode (supports async as well)

from postgresql_lock import Lock

# setup connection
conn = ...

# create lock
lock = Lock(conn, "shared-identifier")

# acquire lock
if lock.acquire(block=False):
    # do something here
    pass

else:
    # could not acquire lock
    pass

# release lock (this is safe to run even if the lock has not been acquired)
lock.release()

Specify the database interface manually

from postgresql_lock import Lock

# setup connection
conn = ...

# create and use lock
lock = Lock(conn, "shared-identifier", interface="asyncpg")

# do things with the lock

Handle rollbacks manually

from postgresql_lock import Lock

# setup connection
conn = ...

# create and use lock
lock = Lock(conn, "shared-identifier", rollback_on_error=False)

# do things with the lock

Changelog

  • 0.2.2
    • Fix: error raised when releasing transaction level locks
  • 0.2.1
    • Moved public Lock fields to properties
  • 0.1.9
    • Fix: release_async() bug for sqlalchemy connections
  • 0.1.8
    • Add logger() function
    • Use "postgresql_lock" logger name
  • 0.1.7
    • Add logging statements
  • 0.1.6
    • Use int.from_bytes() to convert lock key into integer
    • Fix: psycopg3 close() not being awaited bug
  • 0.1.5
    • Rename package from postgres-lock to postgresql-lock
  • 0.1.4
    • Add py.typed for mypy
  • 0.1.3
    • Key can be any object
  • 0.1.2
    • Add Lock.rollback_on_error (default true)
    • Add Lock.handle_error() & Lock.handle_error_async()
  • 0.1.1
    • Key can be str or int

Project details


Download files

Download the file for your platform. If you're not sure which to choose, learn more about installing packages.

Source Distribution

postgresql_lock-0.2.2.tar.gz (7.6 kB view details)

Uploaded Source

Built Distribution

If you're not sure about the file name format, learn more about wheel file names.

postgresql_lock-0.2.2-py3-none-any.whl (10.4 kB view details)

Uploaded Python 3

File details

Details for the file postgresql_lock-0.2.2.tar.gz.

File metadata

  • Download URL: postgresql_lock-0.2.2.tar.gz
  • Upload date:
  • Size: 7.6 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? No
  • Uploaded via: poetry/2.1.3 CPython/3.12.3 Linux/6.11.0-29-generic

File hashes

Hashes for postgresql_lock-0.2.2.tar.gz
Algorithm Hash digest
SHA256 48135101d3daafbc4b81c6adb8f8e448ae30d84535e0f2cfa9beb0f6f4cf911f
MD5 ef153ce3711ed2ba5b7f7ae5640e2864
BLAKE2b-256 b7637e05c9c85933b2a390bbb3e00b74a6fa9968f28fee8d67796d7f348d714e

See more details on using hashes here.

File details

Details for the file postgresql_lock-0.2.2-py3-none-any.whl.

File metadata

  • Download URL: postgresql_lock-0.2.2-py3-none-any.whl
  • Upload date:
  • Size: 10.4 kB
  • Tags: Python 3
  • Uploaded using Trusted Publishing? No
  • Uploaded via: poetry/2.1.3 CPython/3.12.3 Linux/6.11.0-29-generic

File hashes

Hashes for postgresql_lock-0.2.2-py3-none-any.whl
Algorithm Hash digest
SHA256 12b3850ac76b7891d27fc5fce26cab33294b25c198a46f09cfd2588941ea433d
MD5 eb671d60907503637f4d28b094b43440
BLAKE2b-256 69f88a8a389b328d540f06a6990c4b30fef1009337661f73aecd8e4d7d333c01

See more details on using hashes here.

Supported by

AWS Cloud computing and Security Sponsor Datadog Monitoring Depot Continuous Integration Fastly CDN Google Download Analytics Pingdom Monitoring Sentry Error logging StatusPage Status page