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
Release history Release notifications | RSS feed
Download files
Download the file for your platform. If you're not sure which to choose, learn more about installing packages.
Source Distribution
Built Distribution
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
Algorithm | Hash digest | |
---|---|---|
SHA256 | c8a49848dd8fa967b1f95a457a0205fcd49404238f76336a1406e9de12e4fe3f |
|
MD5 | 9b38fb551658467a67452f32da4eb90f |
|
BLAKE2b-256 | 691fdad6ca569246b7d07d234744973deb1278175bb70984acbf4c4d041972d4 |
Provenance
File details
Details for the file psycopg_infdate-1.0.1-py3-none-any.whl
.
File metadata
- Download URL: psycopg_infdate-1.0.1-py3-none-any.whl
- Upload date:
- Size: 7.3 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/4.0.2 CPython/3.11.2
File hashes
Algorithm | Hash digest | |
---|---|---|
SHA256 | efc571f589c6b4709ec792aebb6eb00bbcf123be1cd5c12e846354008ed92bac |
|
MD5 | f267f9720162c2976d432deadbce9666 |
|
BLAKE2b-256 | be6eeabcbe0d06addeb64750ebaf7460b37187b3b2e6d5c6469a69dc069ef1a3 |