1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74
|
from dataclasses import dataclass, field
from datetime import datetime
from typing import Optional
import typedload.datadumper # noqa
import typedload.dataloader # noqa
from common import Benchmark, Methods, Payment
@dataclass(frozen=True)
class Message:
title: str
body: str
addresses: Optional[list[str]] = None # no Python 3.10 support in 2.14
persistence: Optional[int] = None
@dataclass(frozen=True)
class Client:
id: int
first_name: str = field(metadata={"name": "firstName"})
last_name: str = field(metadata={"name": "lastName"})
def __post_init__(self): # The only way I've found to add constraints
if self.id < 0:
raise ValueError
@dataclass(frozen=True)
class Item:
name: str
price: float
quantity: int = 1
def __post_init__(self):
if self.price < 0 or self.quantity < 1:
raise ValueError
@dataclass(frozen=True)
class Receipt:
store: str
address: str
date: datetime
items: list[Item]
payment: Payment
client: Optional[Client] = None
special_offers: Optional[float] = field(
default=None, metadata={"name": "specialOffers"}
)
def __post_init__(self):
if self.special_offers is not None and self.special_offers < 0:
raise ValueError
loader = typedload.dataloader.Loader()
loader.handlers.insert(
loader.index(datetime),
(lambda tp: tp is datetime, lambda _, value, tp: datetime.fromisoformat(value)),
)
dumper = typedload.datadumper.Dumper()
dumper.handlers.insert(
dumper.index(datetime.now()),
(lambda tp: tp is datetime, lambda _, value, tp: value.isoformat()),
)
def methods(cls: type) -> Methods:
load = loader.load
return Methods(lambda data: load(data, cls), dumper.dump)
benchmarks = Benchmark(methods(Message), methods(Receipt))
|