Enhanced dataclasses with automatic dict/JSON serialization for nested structures
Project description
expli
Explicit dataclasses with automatic serialization
expli (short for "explicit") enhances Python dataclasses with automatic dictionary and JSON serialization methods, making it effortless to work with nested dataclasses, optional types, and lists.
Installation
pip install expli
Quick Start
from expli import edataclass
@edataclass
class Person:
name: str
age: int
email: str | None = None
# Automatic methods are added!
person = Person("Alice", 30, "alice@example.com")
# Convert to dict
data = person.to_dict()
# Convert to JSON
json_str = person.to_json(indent=2)
# Create from dict
person2 = Person.from_dict(data)
# Create from JSON
person3 = Person.from_json(json_str)
Features
🎯 Enhanced Dataclass Decorator
The @edataclass decorator automatically adds four methods to your dataclass:
to_dict()- Convert instance to dictionaryfrom_dict(data)- Create instance from dictionary (class method)to_json(indent=None)- Convert instance to JSON stringfrom_json(json_str)- Create instance from JSON string (class method)
🔄 Full Recursive Support
expli handles complex nested structures automatically:
from expli import edataclass
@edataclass
class Address:
street: str
city: str
country: str
@edataclass
class Company:
name: str
address: Address
@edataclass
class Person:
name: str
age: int
company: Company | None
hobbies: list[str]
person = Person(
name="Alice",
age=30,
company=Company(
name="Tech Corp",
address=Address("123 Main St", "Boston", "USA")
),
hobbies=["reading", "coding"]
)
# Everything serializes recursively
data = person.to_dict()
# {
# "name": "Alice",
# "age": 30,
# "company": {
# "name": "Tech Corp",
# "address": {
# "street": "123 Main St",
# "city": "Boston",
# "country": "USA"
# }
# },
# "hobbies": ["reading", "coding"]
# }
# And deserializes back perfectly
person2 = Person.from_dict(data)
✨ Supported Types
- Primitive types:
str,int,float,bool, etc. - Optional types:
Type | None - Lists:
list[Type] - Nested dataclasses: Any dataclass as a field
- Lists of dataclasses:
list[DataclassType] - Optional lists:
list[Type] | None
🛠️ Dataclass Parameters
@edataclass supports all standard dataclass parameters:
from expli import edataclass
@edataclass(frozen=True, order=True)
class Config:
api_key: str
timeout: int = 30
API Reference
@edataclass
Enhanced dataclass decorator that adds serialization methods.
Parameters: Same as @dataclass (frozen, order, etc.)
Added Methods:
to_dict(self) -> dict- Convert instance to dictionaryfrom_dict(cls, data: dict) -> Self- Create instance from dictionaryto_json(self, indent=None) -> str- Convert instance to JSON stringfrom_json(cls, json_str: str) -> Self- Create instance from JSON string
to_dict(obj) / easdict(obj)
Standalone function to convert a dataclass instance to a dictionary.
from expli import to_dict, easdict
from dataclasses import dataclass
@dataclass
class Point:
x: int
y: int
point = Point(10, 20)
data = to_dict(point) # or easdict(point)
# {"x": 10, "y": 20}
from_dict(cls, data) / efromdict(cls, data)
Standalone function to create a dataclass instance from a dictionary.
from expli import from_dict, efromdict
from dataclasses import dataclass
@dataclass
class Point:
x: int
y: int
data = {"x": 10, "y": 20}
point = from_dict(Point, data) # or efromdict(Point, data)
Why expli?
Standard dataclasses don't provide built-in serialization for nested structures. While dataclasses.asdict() exists, it doesn't handle deserialization, and neither function is added to your class for convenient access.
expli solves this by:
- ✅ Adding methods directly to your dataclass
- ✅ Handling nested dataclasses recursively
- ✅ Supporting optional types and lists
- ✅ Providing both dict and JSON serialization
- ✅ Working seamlessly with type hints
Comparison
Without expli:
from dataclasses import dataclass, asdict
import json
@dataclass
class Person:
name: str
age: int
person = Person("Alice", 30)
# Manual serialization
data = asdict(person)
json_str = json.dumps(data)
# Manual deserialization (you have to write this!)
def from_dict(data):
return Person(**data)
person2 = from_dict(json.loads(json_str))
With expli:
from expli import edataclass
@edataclass
class Person:
name: str
age: int
person = Person("Alice", 30)
# Built-in methods
data = person.to_dict()
json_str = person.to_json()
person2 = Person.from_dict(data)
person3 = Person.from_json(json_str)
License
MIT
Contributing
Contributions are welcome! Please feel free to submit a Pull Request.
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
Filter files by name, interpreter, ABI, and platform.
If you're not sure about the file name format, learn more about wheel file names.
Copy a direct link to the current filters
File details
Details for the file expli-0.0.2a4.tar.gz.
File metadata
- Download URL: expli-0.0.2a4.tar.gz
- Upload date:
- Size: 6.2 kB
- Tags: Source
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/6.2.0 CPython/3.12.5
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
ae4dae3b76ce5c75dadd14560d67b88b2d6ecf85674daef3e629be15a0c9155e
|
|
| MD5 |
a124041a48e82ad1d8bc496edf8b98b2
|
|
| BLAKE2b-256 |
a5b9c7430a5b3c0537bc8d4151034edacb8f8fdd03e9e42bf9a70b1197dc0f9d
|
File details
Details for the file expli-0.0.2a4-py3-none-any.whl.
File metadata
- Download URL: expli-0.0.2a4-py3-none-any.whl
- Upload date:
- Size: 7.4 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/6.2.0 CPython/3.12.5
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
c9bf39ea097f271d8d34547e5f56a658db725f6d908c1881330993240edec2c0
|
|
| MD5 |
bb115ef18fdbe06b726e77219e67db3c
|
|
| BLAKE2b-256 |
53646d5e4d56d4943be0a0e791b0d50d76f51ee11d9644dd3d993b81dd544dbc
|