Skip to main content

Support postgresql 'infinity'::DATE in psycopg3

Project description

psycopg Infinite dates

Postgresql has support for special dates, 'infinity' and '-infinity', which compare respectively above or below any other dates. This module gives psycopg3 support for those dates.

Background

Suppose you have a set of contracts with suppliers. The end date of the contract would be an SQL DATE, represented as a datetime.date in Python.

Supplier Name Start Date End Date
ConcreteWorks 2023-04-01 2024-03-31
Bricks R Us 2022-03-25 2023-09-24
Lawn Maintenance 2023-06-03 NULL

Some suppliers are on fixed contracts with a known termination date. It is temping to represent the open-ended lawn maintenance contract as having a NULL end date, after all, all contracts eventually reach an end, we just don't know when.

This results in awkward SQL queries, such as:

SELECT supplier_name FROM suppliers WHERE end_date >= CURRENT_DATE OR end_date IS NULL

NULL is not a value and is therefore not indexed. The fact that we had to explicitly include NULL, as if it were a normal value, is a giveaway that this data is not normalized.

While we do know not know what the end date will be, we do know that it is after today; it isn't a completely unknown quantity.

We could use a sentinel value, such as 1999-12-31. That's obviously a bad choice because it is in the past. How about 2099-12-31? That could also represent real data, if the data pertains to future events. There isn't any choice of sentinel value that could never be confused real data in some context.

Postgres has two special values for dates and timestamps that can be used instead https://www.postgresql.org/docs/current/datatype-datetime.html#DATATYPE-DATETIME-SPECIAL-VALUES

  • 'infinity'::DATE compares later than all other dates.
  • '-infinity'::DATE compares before than all other dates.

Now we can use this for our lawn maintenance contract:

Supplier Name Start Date End Date
ConcreteWorks 2023-04-01 2024-03-31
Bricks R Us 2022-03-25 2023-09-24
Lawn Maintenance 2023-06-03 infinity

And our SQL query is simpler too:

SELECT supplier_name FROM suppliers WHERE end_date >= CURRENT_DATE

The problem

So far, so good. Let's try that SQL query in Python:

with dbh.cursor() as cur:
    cur.execute("SELECT supplier_name FROM suppliers WHERE end_date >= CURRENT_DATE")
    suppliers = cur.fetchall()

This yields the error: psycopg.DataError: date too large (after year 10K): 'infinity'. Not what we wanted!

The problem is that the corresponding Python type, datetime.date, cannot represent this value. It feels like we're back to square one.

The solution

psycopg3 has the ability to register new dumpers and loaders that get data from the database and send data to the database. Their job is to convert between Postgresql representation and Python objects.

This module adds support implements a Python type for each of these special values, and creates the corresponding dumpers and loaders to enable the special values to be handled.

The new types implement basic date arithmetic and comparisons in a similar way to Postgresql, such as:

  • 'infinity' == 'infinity'
  • 'infinity' > datetime.date > '-infinity'
  • 'infinity' + datetime.timedelta == 'infinity'
  • 'infinity' - 'infinity' raises a ValueError

Using the module

For all database connections

import psycopg_infdate
import psycopg

psycopg_infdate.register_inf_date_handler(psycopg)
dbh = psycopg.connect('...')
with dbh.cursor() as cur:
    cur.execute("SELECT 'infinity'::DATE")
    answer = cur.fetchall()

For one database connection

import psycopg_infdate
import psycopg

dbh = psycopg.connect('...')
psycopg_infdate.register_inf_date_handler(dbh)
with dbh.cursor() as cur:
    cur.execute("SELECT 'infinity'::DATE")
    answer = cur.fetchall()

For just this cursor

import psycopg_infdate
import psycopg

dbh = psycopg.connect('...')
with dbh.cursor() as cur:
    psycopg_infdate.register_inf_date_handler(cur)
    cur.execute("SELECT 'infinity'::DATE")
    answer = cur.fetchall()

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

psycopg_infdate-1.0.1.tar.gz (7.0 kB view details)

Uploaded Source

Built Distribution

psycopg_infdate-1.0.1-py3-none-any.whl (7.3 kB view details)

Uploaded Python 3

File details

Details for the file psycopg_infdate-1.0.1.tar.gz.

File metadata

  • Download URL: psycopg_infdate-1.0.1.tar.gz
  • Upload date:
  • Size: 7.0 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? No
  • Uploaded via: twine/4.0.2 CPython/3.11.2

File hashes

Hashes for psycopg_infdate-1.0.1.tar.gz
Algorithm Hash digest
SHA256 c8a49848dd8fa967b1f95a457a0205fcd49404238f76336a1406e9de12e4fe3f
MD5 9b38fb551658467a67452f32da4eb90f
BLAKE2b-256 691fdad6ca569246b7d07d234744973deb1278175bb70984acbf4c4d041972d4

See more details on using hashes here.

Provenance

File details

Details for the file psycopg_infdate-1.0.1-py3-none-any.whl.

File metadata

File hashes

Hashes for psycopg_infdate-1.0.1-py3-none-any.whl
Algorithm Hash digest
SHA256 efc571f589c6b4709ec792aebb6eb00bbcf123be1cd5c12e846354008ed92bac
MD5 f267f9720162c2976d432deadbce9666
BLAKE2b-256 be6eeabcbe0d06addeb64750ebaf7460b37187b3b2e6d5c6469a69dc069ef1a3

See more details on using hashes here.

Provenance

Supported by

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