Skip to main content

A robust way of dealing with datetimes in python by ensuring all datetimes are timezone aware at runtime.

Project description

Heliclockter

PyPI License Python Versions

Timezone-aware datetimes for Python that just work.

heliclockter is a timezone-aware datetime library that ensures your timestamps are always timezone-aware. It's statically type checkable and runtime enforceable.

Installation

pip install heliclockter

Quick Start

from heliclockter import datetime_utc, datetime_local, datetime_tz

# UTC datetime
utc_now = datetime_utc.now()
# datetime_utc(2022, 11, 4, 15, 28, 10, 478176, tzinfo=zoneinfo.ZoneInfo(key='UTC'))

# Local timezone datetime  
local_now = datetime_local.now()

# Any timezone datetime
from zoneinfo import ZoneInfo
paris_tz = datetime_tz.now(tz=ZoneInfo("Europe/Paris"))

# Create a timestamp 2 hours in the future
future = datetime_utc.future(hours=2)

# Parse strings (naive timestamps assumed UTC)
parsed = datetime_utc.strptime('2022-11-04T15:49:29', '%Y-%m-%dT%H:%M:%S')

Why heliclockter?

Python's standard datetime allows "naive" datetimes without timezone info, leading to bugs when:

  • Mixing naive and aware datetimes (causes runtime TypeErrors)
  • Deploying across different timezones
  • Forgetting to add tzinfo when creating datetimes

heliclockter enforces timezone-aware datetimes at the type level, catching these issues before production.

Key Features

  • Always timezone-aware - No more naive datetime accidents
  • Type safe - Full typing support for better IDE experience
  • Zero dependencies - Lightweight, uses only standard library
  • Pydantic support - Automatic integration when Pydantic is installed
  • Python 3.10+ - Modern Python for modern applications

Examples

Timezone conversions

from heliclockter import datetime_utc, datetime_tz
from zoneinfo import ZoneInfo

# Start with UTC
utc_time = datetime_utc.now()

# To convert to different timezones, create custom classes
class datetime_tokyo(datetime_tz):
    assumed_timezone_for_timezone_naive_input = ZoneInfo('Asia/Tokyo')

class datetime_ny(datetime_tz):
    assumed_timezone_for_timezone_naive_input = ZoneInfo('America/New_York')

# Convert using from_datetime
tokyo_time = datetime_tokyo.from_datetime(utc_time)
ny_time = datetime_ny.from_datetime(utc_time)

Handling naive datetimes

from heliclockter import datetime_utc, datetime_tz
from datetime import datetime

# datetime_utc assumes UTC for naive inputs
naive_dt = datetime(2022, 11, 4, 15, 30, 0)
utc_dt = datetime_utc.from_datetime(naive_dt)  # OK - assumes UTC

# datetime_tz requires explicit timezone
try:
    tz_dt = datetime_tz.from_datetime(naive_dt)  # Raises error
except Exception as e:
    print(e)  # "Cannot create aware datetime from naive if no tz is assumed"

Custom timezone classes

from zoneinfo import ZoneInfo
from heliclockter import datetime_tz

class datetime_cet(datetime_tz):
    """Datetime guaranteed to be in CET timezone."""
    assumed_timezone_for_timezone_naive_input = ZoneInfo('CET')

# Parse naive timestamps as CET
aware_dt = datetime_cet.strptime('2022-11-04T15:49:29', '%Y-%m-%dT%H:%M:%S')

Type safety with mypy

from heliclockter import datetime_utc, datetime_local

def schedule_task(when: datetime_utc) -> None:
    """Schedule a task at a specific UTC time."""
    print(f"Task scheduled for {when.isoformat()}")

# Type checker ensures only UTC datetimes are passed
utc_time = datetime_utc.now()
schedule_task(utc_time)  # ✓ OK

local_time = datetime_local.now()  
schedule_task(local_time)  # ✗ Type error

API Overview

Core Classes

  • datetime_tz - Base class for timezone-aware datetimes
  • datetime_utc - Always UTC (naive inputs assumed UTC)
  • datetime_local - Always local timezone (naive inputs assumed local)

Key Methods

  • now() - Current time
  • from_datetime() - Convert from standard datetime
  • strptime() - Parse string to datetime
  • future()/past() - Create relative timestamps
  • astimezone() - Convert to other timezone

About the Name

heliclockter is a portmanteau of "clock" and "helicopter". Like a helicopter parent, it strictly supervises your datetime handling, ensuring you never make timezone mistakes.

Contributing

We welcome contributions! See CONTRIBUTING.md.

Python and Pydantic compatibility

The table below shows which Pydantic and Python versions are supported for which heliclockter version. Note that the latest version of heliclockter dropped support for Pydantic v1 code completely, meaning that you also can't use heliclockter in combination with the pydantic.v1 module in Pydantic v2.

heliclockter version Pydantic support Python support
1.0 v1 3.9, 3.10, 3.11, 3.12, 3.13
1.1, 1.2, 1.3 v1, v2 3.9, 3.10, 3.11, 3.12, 3.13
2.x v1, v2 3.10, 3.11, 3.12, 3.13
3.x v2 3.10, 3.11, 3.12, 3.13, 3.14

License

BSD 3-Clause License. See LICENSE.

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

heliclockter-3.0.1.tar.gz (19.4 kB view details)

Uploaded Source

Built Distribution

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

heliclockter-3.0.1-py3-none-any.whl (8.2 kB view details)

Uploaded Python 3

File details

Details for the file heliclockter-3.0.1.tar.gz.

File metadata

  • Download URL: heliclockter-3.0.1.tar.gz
  • Upload date:
  • Size: 19.4 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? No
  • Uploaded via: twine/6.2.0 CPython/3.10.12

File hashes

Hashes for heliclockter-3.0.1.tar.gz
Algorithm Hash digest
SHA256 bedc66fe18548abc9088d0bb7c2bcd764f980607cb1954ac83f0e17b1d366c10
MD5 9935975419623fa4482964d2f6791299
BLAKE2b-256 1c3ff66932893ff052d7e3416b3af9084f9523dad3941ff166b3958f89477101

See more details on using hashes here.

File details

Details for the file heliclockter-3.0.1-py3-none-any.whl.

File metadata

  • Download URL: heliclockter-3.0.1-py3-none-any.whl
  • Upload date:
  • Size: 8.2 kB
  • Tags: Python 3
  • Uploaded using Trusted Publishing? No
  • Uploaded via: twine/6.2.0 CPython/3.10.12

File hashes

Hashes for heliclockter-3.0.1-py3-none-any.whl
Algorithm Hash digest
SHA256 b453481854a8d10c9f08b0a272bc757133054741f4ef0fcbb675135b95207ce8
MD5 1e694705dc26786828c2b138fcf93c5f
BLAKE2b-256 381e6fee94fcc46955b4e4447b72ab257fae66bcb2fbdbe54c7c67432ff95318

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