Skip to content

Commit

Permalink
Add unicode datetime attribute (#33)
Browse files Browse the repository at this point in the history
UnicodeDatetimeAttribute stores datetimes as 8601 ISO strings with offset.

The storage representation of this format will look something like:
{"key": {"S": "2020-11-22T03:22:33.444444-08:00"}}

The attribute by default will add an offset to UTC if not present and make it timezone aware. It also as options for normalizing the date to UTC (for caching purposes) and adds support for custom formatting.
  • Loading branch information
NazarioJL authored Jul 14, 2021
1 parent 1fd641d commit 7a8ea99
Show file tree
Hide file tree
Showing 6 changed files with 293 additions and 1 deletion.
14 changes: 14 additions & 0 deletions README.md
Original file line number Diff line number Diff line change
Expand Up @@ -3,6 +3,7 @@
**Table of Contents** *generated with [DocToc](/~https://github.com/thlorenz/doctoc)*

- [pynamodb-attributes](#pynamodb-attributes)
- [Testing](#testing)

<!-- END doctoc generated TOC please keep comment here to allow auto update -->

Expand All @@ -17,3 +18,16 @@ This Python 3 library contains compound and high-level PynamoDB attributes:
- `TimestampAttribute`, `TimestampMsAttribute`, `TimestampUsAttribute` – serializes `datetime`s as Unix epoch seconds, milliseconds (ms) or microseconds (µs)
- `IntegerDateAttribute` - serializes `date` as an integer representing the Gregorian date (_e.g._ `20181231`)
- `UUIDAttribute` - serializes a `UUID` Python object as a `S` type attribute (_e.g._ `'a8098c1a-f86e-11da-bd1a-00112444be1e'`)
- `UnicodeDatetimeAttribute` - ISO8601 datetime strings with offset information

## Testing

The tests in this repository use an in-memory implementation of [`dynamodb`](https://aws.amazon.com/dynamodb). To run the tests locally, make sure [DynamoDB Local](https://docs.aws.amazon.com/amazondynamodb/latest/developerguide/DynamoDBLocal.html) is running. It is available as a standalone binary, through package managers (e.g. [Homebrew](https://formulae.brew.sh/cask/dynamodb-local)) or as a Docker container:
```shell
docker run -d -p 8000:8000 amazon/dynamodb-local
```

Afterwards, run tests as usual:
```shell
pytest tests
```
2 changes: 2 additions & 0 deletions pynamodb_attributes/__init__.py
Original file line number Diff line number Diff line change
Expand Up @@ -8,6 +8,7 @@
from .timestamp import TimestampAttribute
from .timestamp import TimestampMsAttribute
from .timestamp import TimestampUsAttribute
from .unicode_datetime import UnicodeDatetimeAttribute
from .unicode_delimited_tuple import UnicodeDelimitedTupleAttribute
from .unicode_enum import UnicodeEnumAttribute
from .uuid import UUIDAttribute
Expand All @@ -26,4 +27,5 @@
"TimestampMsAttribute",
"TimestampUsAttribute",
"UUIDAttribute",
"UnicodeDatetimeAttribute",
]
55 changes: 55 additions & 0 deletions pynamodb_attributes/unicode_datetime.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,55 @@
from datetime import datetime
from datetime import timezone
from typing import Any
from typing import Optional

import pynamodb.attributes
from pynamodb.attributes import Attribute


class UnicodeDatetimeAttribute(Attribute[datetime]):
"""
Stores a 'datetime.datetime' object as an ISO8601 formatted string
This is useful for wanting database readable datetime objects that also sort.
>>> class MyModel(Model):
>>> created_at = UnicodeDatetimeAttribute()
"""

attr_type = pynamodb.attributes.STRING

def __init__(
self,
*,
force_tz: bool = True,
force_utc: bool = False,
fmt: Optional[str] = None,
**kwargs: Any,
) -> None:
"""
:param force_tz: If set it will add timezone info to the `datetime` value if no `tzinfo` is currently
set before serializing, defaults to `True`
:param force_utc: If set it will normalize the `datetime` to UTC before serializing the value
:param fmt: If set this value will be used to format the `datetime` object for serialization
and deserialization
"""

super().__init__(**kwargs)
self._force_tz = force_tz
self._force_utc = force_utc
self._fmt = fmt

def deserialize(self, value: str) -> datetime:
return (
datetime.fromisoformat(value)
if self._fmt is None
else datetime.strptime(value, self._fmt)
)

def serialize(self, value: datetime) -> str:
if self._force_tz and value.tzinfo is None:
value = value.replace(tzinfo=timezone.utc)
if self._force_utc:
value = value.astimezone(tz=timezone.utc)
return value.isoformat() if self._fmt is None else value.strftime(self._fmt)
2 changes: 1 addition & 1 deletion setup.py
Original file line number Diff line number Diff line change
Expand Up @@ -3,7 +3,7 @@

setup(
name="pynamodb-attributes",
version="0.3.0",
version="0.3.1",
description="Common attributes for PynamoDB",
url="https://www.github.com/lyft/pynamodb-attributes",
maintainer="Lyft",
Expand Down
15 changes: 15 additions & 0 deletions tests/mypy_test.py
Original file line number Diff line number Diff line change
Expand Up @@ -141,3 +141,18 @@ class MyModel(Model):
reveal_type(MyModel().my_attr) # E: Revealed type is "uuid.UUID*"
""",
)


def test_unicode_datetime():
assert_mypy_output(
"""
from pynamodb.models import Model
from pynamodb_attributes import UnicodeDatetimeAttribute
class MyModel(Model):
my_attr = UnicodeDatetimeAttribute()
reveal_type(MyModel.my_attr) # E: Revealed type is "pynamodb_attributes.unicode_datetime.UnicodeDatetimeAttribute"
reveal_type(MyModel().my_attr) # E: Revealed type is "datetime.datetime*"
""",
)
206 changes: 206 additions & 0 deletions tests/unicode_datetime_attribute_test.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,206 @@
from datetime import datetime
from unittest.mock import ANY

import pytest
from pynamodb.attributes import UnicodeAttribute
from pynamodb.models import Model

from pynamodb_attributes import UnicodeDatetimeAttribute
from tests.connection import _connection
from tests.meta import dynamodb_table_meta


CUSTOM_FORMAT = "%m/%d/%Y, %H:%M:%S"
CUSTOM_FORMAT_DATE = "11/22/2020, 11:22:33"
TEST_ISO_DATE_NO_OFFSET = "2020-11-22T11:22:33.444444"
TEST_ISO_DATE_UTC = "2020-11-22T11:22:33.444444+00:00"
TEST_ISO_DATE_PST = "2020-11-22T03:22:33.444444-08:00"


class MyModel(Model):
Meta = dynamodb_table_meta(__name__)

key = UnicodeAttribute(hash_key=True)
default = UnicodeDatetimeAttribute(null=True)
no_force_tz = UnicodeDatetimeAttribute(force_tz=False, null=True)
force_utc = UnicodeDatetimeAttribute(force_utc=True, null=True)
force_utc_no_force_tz = UnicodeDatetimeAttribute(
force_utc=True,
force_tz=False,
null=True,
)
custom_format = UnicodeDatetimeAttribute(fmt=CUSTOM_FORMAT, null=True)


@pytest.fixture(scope="module", autouse=True)
def create_table():
MyModel.create_table()


@pytest.mark.parametrize(
["value", "expected_str", "expected_value"],
[
(
datetime.fromisoformat(TEST_ISO_DATE_NO_OFFSET),
TEST_ISO_DATE_UTC,
datetime.fromisoformat(TEST_ISO_DATE_UTC),
),
(
datetime.fromisoformat(TEST_ISO_DATE_UTC),
TEST_ISO_DATE_UTC,
datetime.fromisoformat(TEST_ISO_DATE_UTC),
),
(
datetime.fromisoformat(TEST_ISO_DATE_PST),
TEST_ISO_DATE_PST,
datetime.fromisoformat(TEST_ISO_DATE_PST),
),
],
)
def test_default_serialization(value, expected_str, expected_value, uuid_key):
model = MyModel()
model.key = uuid_key
model.default = value

model.save()

actual = MyModel.get(hash_key=uuid_key)
assert actual.default == expected_value

item = _connection(MyModel).get_item(uuid_key)
assert item["Item"] == {"key": ANY, "default": {"S": expected_str}}


@pytest.mark.parametrize(
["value", "expected_str", "expected_value"],
[
(
datetime.fromisoformat(TEST_ISO_DATE_NO_OFFSET),
TEST_ISO_DATE_NO_OFFSET,
datetime.fromisoformat(TEST_ISO_DATE_NO_OFFSET),
),
(
datetime.fromisoformat(TEST_ISO_DATE_UTC),
TEST_ISO_DATE_UTC,
datetime.fromisoformat(TEST_ISO_DATE_UTC),
),
(
datetime.fromisoformat(TEST_ISO_DATE_PST),
TEST_ISO_DATE_PST,
datetime.fromisoformat(TEST_ISO_DATE_PST),
),
],
)
def test_no_force_tz_serialization(value, expected_str, expected_value, uuid_key):
model = MyModel()
model.key = uuid_key
model.no_force_tz = value

model.save()

actual = MyModel.get(hash_key=uuid_key)
item = _connection(MyModel).get_item(uuid_key)

assert item["Item"] == {"key": ANY, "no_force_tz": {"S": expected_str}}

assert actual.no_force_tz == expected_value


@pytest.mark.parametrize(
["value", "expected_str", "expected_value"],
[
(
datetime.fromisoformat(TEST_ISO_DATE_NO_OFFSET),
TEST_ISO_DATE_UTC,
datetime.fromisoformat(TEST_ISO_DATE_UTC),
),
(
datetime.fromisoformat(TEST_ISO_DATE_UTC),
TEST_ISO_DATE_UTC,
datetime.fromisoformat(TEST_ISO_DATE_UTC),
),
(
datetime.fromisoformat(TEST_ISO_DATE_PST),
TEST_ISO_DATE_UTC,
datetime.fromisoformat(TEST_ISO_DATE_UTC),
),
],
)
def test_force_utc_serialization(value, expected_str, expected_value, uuid_key):
model = MyModel()
model.key = uuid_key
model.force_utc = value

model.save()

actual = MyModel.get(hash_key=uuid_key)
item = _connection(MyModel).get_item(uuid_key)

assert item["Item"] == {"key": ANY, "force_utc": {"S": expected_str}}

assert actual.force_utc == expected_value


@pytest.mark.parametrize(
["value", "expected_str", "expected_value"],
[
(
datetime.fromisoformat(TEST_ISO_DATE_UTC),
TEST_ISO_DATE_UTC,
datetime.fromisoformat(TEST_ISO_DATE_UTC),
),
(
datetime.fromisoformat(TEST_ISO_DATE_PST),
TEST_ISO_DATE_UTC,
datetime.fromisoformat(TEST_ISO_DATE_UTC),
),
],
)
def test_force_utc_no_force_tz_serialization(
value,
expected_str,
expected_value,
uuid_key,
):
model = MyModel()
model.key = uuid_key
model.force_utc_no_force_tz = value

model.save()

actual = MyModel.get(hash_key=uuid_key)
item = _connection(MyModel).get_item(uuid_key)

assert item["Item"] == {"key": ANY, "force_utc_no_force_tz": {"S": expected_str}}

assert actual.force_utc_no_force_tz == expected_value


@pytest.mark.parametrize(
["value", "expected_str", "expected_value"],
[
(
datetime.fromisoformat(TEST_ISO_DATE_UTC),
CUSTOM_FORMAT_DATE,
datetime(2020, 11, 22, 11, 22, 33),
),
],
)
def test_custom_format_force_tz_serialization(
value,
expected_str,
expected_value,
uuid_key,
):
model = MyModel()
model.key = uuid_key
model.custom_format = value

model.save()

actual = MyModel.get(hash_key=uuid_key)
item = _connection(MyModel).get_item(uuid_key)

assert item["Item"] == {"key": ANY, "custom_format": {"S": expected_str}}

assert actual.custom_format == expected_value

0 comments on commit 7a8ea99

Please sign in to comment.