chronomeleon


Namechronomeleon JSON
Version 0.1.3 PyPI version JSON
download
home_pageNone
SummaryA python package to flexibly adapt start and end date(times) to your system background
upload_time2024-10-11 06:21:02
maintainerNone
docs_urlNone
authorNone
requires_python>=3.10
licenseMIT
keywords conversion date dateonly exclusive inclusive migration time
VCS
bugtrack_url
requirements pytz
Travis-CI No Travis.
coveralls test coverage No coveralls.
            # Chronomeleon

[![License: MIT](https://img.shields.io/badge/License-MIT-yellow.svg)](LICENSE)
![Python Versions (officially) supported](https://img.shields.io/pypi/pyversions/chronomeleon.svg)
![Pypi status badge](https://img.shields.io/pypi/v/chronomeleon)
![Unittests status badge](https://github.com/Hochfrequenz/chronomeleon/workflows/Unittests/badge.svg)
![Coverage status badge](https://github.com/Hochfrequenz/chronomeleon/workflows/Coverage/badge.svg)
![Linting status badge](https://github.com/Hochfrequenz/chronomeleon/workflows/Linting/badge.svg)
![Black status badge](https://github.com/Hochfrequenz/chronomeleon/workflows/Formatting/badge.svg)

Chronomeleon is a Python package that converts and maps date and time information from their representation in one system to another.
It's meant to be used in data migration projects.

## Rationale
While converting a datetime alone is possible with either Python builtin tools or libraries like `pendulum` and `arrow`,
things become more complicated when you have to convert time slices or ranges or when the source and target system interpret dates and times differently.

Think your migrating e.g., contracts from system A to system B.
In system A might have an API, a data dump or something else from where you can read,
that a contract starts at `2024-01-01` and ends at `2024-06-30`.

Now assume, the same contract in system B starts at `2023-12-31T23:00:00Z` and ends at `2024-06-30T21:59:59Z`.

For this little conversion, although simple, you have to consider a lot:
* Are date and times implicitly "meant" in any certain time zone? Here, system A seems to implicitly assume, everything as German local time, whereas system B uses explicit UTC offsets.
* What about the resolution? Although using dates only might be sufficient for the modeling the business logic, as soon as the resolution of system B is higher, you have to start interpreting stuff.
  * What if there was a system C, which supported microseconds but only stored the date and time in a single integer?
* What about the end date (times)? It seems that system A uses the end date as inclusive (contract ends at the end of june), whereas system B uses it as exclusive (start of the followup contract == end of the previous contract).

Chronomeleon has two purposes:
1. It forces you to make assumptions explicit.
2. Once the assumptions are explicit, it helps you do the conversion.

The latter is no rocket science (and neither is any line of code in chronomeleon), but making assumptions explicit is crucial and that's why using it is beneficial.

When you're constantly wondering why other coders seem to randomly
* add or subtract a day, a second, a tick here and there
* pass around naive dates and datetimes and try to convert them to UTC or other timezones with no clear reason

then chronomeleon is for you.

Chronomeleon makes your code more readable and makes your assumption clear.
This allows you to spot errors in your or your teammates code more easily and explain why things are done the way they are.

## How to use it?
Install it from pypi:
```bash
pip install chronomeleon
```

Then, in your code: Make assumptions about the source and target system explicit.
To do so, chronomeleon provides you with so-called MappingConfig objects.

Here's an advanced example, that shows the capabilities of Chronomeleon:
```python
from datetime import date, datetime, timedelta

import pytz

from chronomeleon import ChronoAssumption, MappingConfig, adapt_to_target

config = MappingConfig( # make assumptions explicit
    source=ChronoAssumption(
        implicit_timezone=pytz.timezone("Europe/Berlin"),
        resolution=timedelta(days=1),
        is_inclusive_end=True,
        is_gastag_aware=False,
    ),
    target=ChronoAssumption(resolution=timedelta(milliseconds=1), is_inclusive_end=True, is_gastag_aware=True),
    is_end=True,
    is_gas=True,
)
source_value = date(2021, 12, 31)
result = adapt_to_target(source_value, config) # do the mapping
assert result == datetime(2022, 1, 1, 4, 59, 59, microsecond=999000, tzinfo=pytz.utc)
```


## Setup for Local Development
Follow the instructions from our [template repository](https://github.com/Hochfrequenz/python_template_repository?tab=readme-ov-file#how-to-use-this-repository-on-your-machine).
tl;dr: `tox`.

## Contribute
You are very welcome to contribute to this template repository by opening a pull request against the main branch.

            

Raw data

            {
    "_id": null,
    "home_page": null,
    "name": "chronomeleon",
    "maintainer": null,
    "docs_url": null,
    "requires_python": ">=3.10",
    "maintainer_email": null,
    "keywords": "conversion, date, dateonly, exclusive, inclusive, migration, time",
    "author": null,
    "author_email": "Hochfrequenz Unternehmensberatung GmbH <info+github@hochfrequenz.de>",
    "download_url": "https://files.pythonhosted.org/packages/35/a9/78d42871441b860d6f899e31b608ae806ae279fb187f0210143ae9d5ec5f/chronomeleon-0.1.3.tar.gz",
    "platform": null,
    "description": "# Chronomeleon\n\n[![License: MIT](https://img.shields.io/badge/License-MIT-yellow.svg)](LICENSE)\n![Python Versions (officially) supported](https://img.shields.io/pypi/pyversions/chronomeleon.svg)\n![Pypi status badge](https://img.shields.io/pypi/v/chronomeleon)\n![Unittests status badge](https://github.com/Hochfrequenz/chronomeleon/workflows/Unittests/badge.svg)\n![Coverage status badge](https://github.com/Hochfrequenz/chronomeleon/workflows/Coverage/badge.svg)\n![Linting status badge](https://github.com/Hochfrequenz/chronomeleon/workflows/Linting/badge.svg)\n![Black status badge](https://github.com/Hochfrequenz/chronomeleon/workflows/Formatting/badge.svg)\n\nChronomeleon is a Python package that converts and maps date and time information from their representation in one system to another.\nIt's meant to be used in data migration projects.\n\n## Rationale\nWhile converting a datetime alone is possible with either Python builtin tools or libraries like `pendulum` and `arrow`,\nthings become more complicated when you have to convert time slices or ranges or when the source and target system interpret dates and times differently.\n\nThink your migrating e.g., contracts from system A to system B.\nIn system A might have an API, a data dump or something else from where you can read,\nthat a contract starts at `2024-01-01` and ends at `2024-06-30`.\n\nNow assume, the same contract in system B starts at `2023-12-31T23:00:00Z` and ends at `2024-06-30T21:59:59Z`.\n\nFor this little conversion, although simple, you have to consider a lot:\n* Are date and times implicitly \"meant\" in any certain time zone? Here, system A seems to implicitly assume, everything as German local time, whereas system B uses explicit UTC offsets.\n* What about the resolution? Although using dates only might be sufficient for the modeling the business logic, as soon as the resolution of system B is higher, you have to start interpreting stuff.\n  * What if there was a system C, which supported microseconds but only stored the date and time in a single integer?\n* What about the end date (times)? It seems that system A uses the end date as inclusive (contract ends at the end of june), whereas system B uses it as exclusive (start of the followup contract == end of the previous contract).\n\nChronomeleon has two purposes:\n1. It forces you to make assumptions explicit.\n2. Once the assumptions are explicit, it helps you do the conversion.\n\nThe latter is no rocket science (and neither is any line of code in chronomeleon), but making assumptions explicit is crucial and that's why using it is beneficial.\n\nWhen you're constantly wondering why other coders seem to randomly\n* add or subtract a day, a second, a tick here and there\n* pass around naive dates and datetimes and try to convert them to UTC or other timezones with no clear reason\n\nthen chronomeleon is for you.\n\nChronomeleon makes your code more readable and makes your assumption clear.\nThis allows you to spot errors in your or your teammates code more easily and explain why things are done the way they are.\n\n## How to use it?\nInstall it from pypi:\n```bash\npip install chronomeleon\n```\n\nThen, in your code: Make assumptions about the source and target system explicit.\nTo do so, chronomeleon provides you with so-called MappingConfig objects.\n\nHere's an advanced example, that shows the capabilities of Chronomeleon:\n```python\nfrom datetime import date, datetime, timedelta\n\nimport pytz\n\nfrom chronomeleon import ChronoAssumption, MappingConfig, adapt_to_target\n\nconfig = MappingConfig( # make assumptions explicit\n    source=ChronoAssumption(\n        implicit_timezone=pytz.timezone(\"Europe/Berlin\"),\n        resolution=timedelta(days=1),\n        is_inclusive_end=True,\n        is_gastag_aware=False,\n    ),\n    target=ChronoAssumption(resolution=timedelta(milliseconds=1), is_inclusive_end=True, is_gastag_aware=True),\n    is_end=True,\n    is_gas=True,\n)\nsource_value = date(2021, 12, 31)\nresult = adapt_to_target(source_value, config) # do the mapping\nassert result == datetime(2022, 1, 1, 4, 59, 59, microsecond=999000, tzinfo=pytz.utc)\n```\n\n\n## Setup for Local Development\nFollow the instructions from our [template repository](https://github.com/Hochfrequenz/python_template_repository?tab=readme-ov-file#how-to-use-this-repository-on-your-machine).\ntl;dr: `tox`.\n\n## Contribute\nYou are very welcome to contribute to this template repository by opening a pull request against the main branch.\n",
    "bugtrack_url": null,
    "license": "MIT",
    "summary": "A python package to flexibly adapt start and end date(times) to your system background",
    "version": "0.1.3",
    "project_urls": {
        "Changelog": "https://github.com/Hochfrequenz/chronomeleon/releases",
        "Homepage": "https://github.com/Hochfrequenz/chronomeleon"
    },
    "split_keywords": [
        "conversion",
        " date",
        " dateonly",
        " exclusive",
        " inclusive",
        " migration",
        " time"
    ],
    "urls": [
        {
            "comment_text": "",
            "digests": {
                "blake2b_256": "15424e568a8e20f082294193b4ec92cd91b4ab7d377586d750cabfe7146d2d84",
                "md5": "3ed2e7dc5b80bfd50e993c1dde15ef19",
                "sha256": "a0a323894e007781288baf888e98225d747e03eda367795040d03f96140c2828"
            },
            "downloads": -1,
            "filename": "chronomeleon-0.1.3-py3-none-any.whl",
            "has_sig": false,
            "md5_digest": "3ed2e7dc5b80bfd50e993c1dde15ef19",
            "packagetype": "bdist_wheel",
            "python_version": "py3",
            "requires_python": ">=3.10",
            "size": 9297,
            "upload_time": "2024-10-11T06:21:00",
            "upload_time_iso_8601": "2024-10-11T06:21:00.474450Z",
            "url": "https://files.pythonhosted.org/packages/15/42/4e568a8e20f082294193b4ec92cd91b4ab7d377586d750cabfe7146d2d84/chronomeleon-0.1.3-py3-none-any.whl",
            "yanked": false,
            "yanked_reason": null
        },
        {
            "comment_text": "",
            "digests": {
                "blake2b_256": "35a978d42871441b860d6f899e31b608ae806ae279fb187f0210143ae9d5ec5f",
                "md5": "ca0b7782a88d206d07d9fd70a1e5ccaa",
                "sha256": "2708f1ba5c3ea23297ae7d43fe906a22dffb3ca9d7271c045834afc590d68036"
            },
            "downloads": -1,
            "filename": "chronomeleon-0.1.3.tar.gz",
            "has_sig": false,
            "md5_digest": "ca0b7782a88d206d07d9fd70a1e5ccaa",
            "packagetype": "sdist",
            "python_version": "source",
            "requires_python": ">=3.10",
            "size": 13959,
            "upload_time": "2024-10-11T06:21:02",
            "upload_time_iso_8601": "2024-10-11T06:21:02.288714Z",
            "url": "https://files.pythonhosted.org/packages/35/a9/78d42871441b860d6f899e31b608ae806ae279fb187f0210143ae9d5ec5f/chronomeleon-0.1.3.tar.gz",
            "yanked": false,
            "yanked_reason": null
        }
    ],
    "upload_time": "2024-10-11 06:21:02",
    "github": true,
    "gitlab": false,
    "bitbucket": false,
    "codeberg": false,
    "github_user": "Hochfrequenz",
    "github_project": "chronomeleon",
    "travis_ci": false,
    "coveralls": false,
    "github_actions": true,
    "requirements": [
        {
            "name": "pytz",
            "specs": [
                [
                    "==",
                    "2024.2"
                ]
            ]
        }
    ],
    "tox": true,
    "lcname": "chronomeleon"
}
        
Elapsed time: 0.35216s