Name | dataclassy JSON |
Version |
1.0.1
JSON |
| download |
home_page | https://github.com/biqqles/dataclassy |
Summary | A fast and flexible reimplementation of data classes |
upload_time | 2023-01-14 12:47:59 |
maintainer | |
docs_url | None |
author | biqqles |
requires_python | >=3.6 |
license | |
keywords |
|
VCS |
|
bugtrack_url |
|
requirements |
No requirements were recorded.
|
Travis-CI |
No Travis.
|
coveralls test coverage |
No coveralls.
|
# dataclassy
**dataclassy** is a reimplementation of data classes in Python - an alternative to the built-in [dataclasses module](https://docs.python.org/3/library/dataclasses.html) that avoids many of [its](https://stackoverflow.com/questions/54678337) [common](https://stackoverflow.com/q/51575931) [pitfalls](https://stackoverflow.com/q/50180735). dataclassy is designed to be more flexible, less verbose, and more powerful than dataclasses, while retaining a familiar interface.
In short, dataclassy is a library for moving data around your Python programs that's optimised for speed, simplicity and developer happiness.
```python
from dataclassy import dataclass
from typing import Dict
@dataclass
class Pet:
name: str
species: str
fluffy: bool
foods: Dict[str, int] = {}
```
## Why use dataclassy?
This section describes various motivations for using **dataclassy** over **dataclasses**.
#### Backwards compatibility
dataclassy implements the decorator options of the latest version of dataclasses, plus its own, such that they are compatible back to Python 3.6. It is [tested against](.github/workflows/ci.yml) **CPython 3.6 through 3.11** and **PyPy 3.6 through 3.9**.
#### Upgrade your data classes
- new decorator options
- [`slots`](#slots) generates `__slots__` to reduce memory footprint and improve attribute lookup performance
- [`kwargs`](#kwargs) appends `**kwargs` to `__init__`'s parameter list to consume unexpected arguments
- [`iter`](#iter) allows class instances to be destructured, like named tuples
- [`hide_internals`](#hide_internals) automatically hides private fields from `__repr__` and excludes them from comparison and iteration
- `@dataclass` usage and options are inherited (subclasses do not have to reuse the decorator)
- fields can be in any order - fields with defaults are [reordered](#parameter-reordering) - making inheritance feasible
- mutable default values (`list`, `set`, `dict` and more) are [automatically copied](#default-values) upon initialisation
- new functions: [`is_dataclass_instance`](#is_dataclass_instanceobj) and [`values`](#valuesdataclass-internalsfalse)
#### Additionally, dataclassy
- implements all the decorator options and functions of dataclasses
- is tiny (~160 LOC; less than 25% the size of dataclasses)
- has no external dependencies, and no stdlib imports other than `types`, `typing` and `reprlib`
- is [fast](benchmarks.py), matching dataclasses' performance when `slots=False` and significantly exceeding it when `slots=True`
- supports multiple inheritance and custom metaclasses
- comes with [support for mypy](#mypy-support)
#### Other differences
dataclassy's API is strongly influenced by dataclasses', but with a focus on minimalism and elegance.
- there's no `field` or `Field`. Use [`Hashed`](#hashed), [`Internal`](#internal) or [`factory`](#factorycreator) to replicate its functions
- there's no `InitVar`. Simply add arguments to `__post_init__`
- there's no need for `ClassVar`. Simply omit the field's type hint to ignore it
#### Also consider
- [`attrs`](https://github.com/python-attrs/attrs) if you need complex validation and type conversions
- [`pydantic`](https://github.com/samuelcolvin/pydantic) if you need strict type checking
## Usage
### Installation
Install the latest release from [PyPI](https://pypi.org/project/dataclassy/) with pip:
```console
pip install dataclassy
```
Or install the latest development version straight from this repository:
```console
pip install https://github.com/biqqles/dataclassy/archive/master.zip -U
```
### Migration
dataclassy's API is broadly similar to dataclasses. If you simply use the decorator and other functions, it is possible to instantly migrate from dataclasses to dataclassy by simply changing
```Python
from dataclasses import *
```
to
```Python
from dataclassy import *
```
Otherwise, you will have to make a couple of easy refactorings (that should leave you with cleaner code!). Consult the table under [Differences](#differences) or skip ahead to [Examples](#examples) to see dataclassy in action.
#### Similarities
dataclassy's `dataclass` decorator takes all of the same arguments as dataclasses', plus its own, and should therefore be a drop-in replacement.
dataclassy also implements all dataclasses' [functions](#functions): `is_dataclass`, `fields`, `replace`, `make_dataclass`, `asdict` and `astuple` (the last two are aliased from `as_dict` and `as_tuple` respectively), and they should work as you expect.
#### Differences
Although dataclassy's API is similar to dataclasses', [compatibility with it is not a goal](https://straight-shoota.github.io/crystal-book/feature/tutorials-initial/) (this is similar to the relationship between Crystal and Ruby).
dataclassy has several important differences from dataclasses, mainly reflective of its minimalistic style and implementation. These differences are enumerated below and fully expanded on in the next section.
| |dataclasses |dataclassy |
|---------------------------------|:-------------------------------------------|:---------------------------------------|
|*init-only variables* |fields with type `InitVar` |arguments to `__post_init__` |
|*class variables* |fields with type `ClassVar` |fields without type annotation |
|*mutable defaults* |`a: Dict = field(default_factory=dict)` |`a: Dict = {}` |
|*dynamic defaults* |`b: MyClass = field(default_factory=MyClass)`|`b: MyClass = factory(MyClass)` |
|*field excluded from `repr`* |`c: int = field(repr=False)` |`Internal` type wrapper or `_name` |
|*"late init" field* |`d: int = field(init=False)` |`d: int = None` |
|*abstract data class* |`class Foo(ABC):` |`class Foo(metaclass=ABCMeta):` |
There are a couple of minor differences, too:
- `fields` returns `Dict[str, Type]` instead of `Dict[Field, Type]` and has an additional parameter which filters internal fields
- Attempting to modify a frozen instance raises `AttributeError` with an explanation rather than `FrozenInstanceError`
Finally, there are some quality of life improvements that, while not being directly implicated in migration, will allow you to make your code cleaner:
- `@dataclass` does not need to be applied to every subclass - its behaviour and options are inherited
- Unlike dataclasses, fields with defaults do not need to follow those without them. This is particularly useful when working with subclasses, which is almost impossible with dataclasses
- dataclassy adds a `DataClass` type annotation to represent variables that should be generic data class instances
- dataclassy has the `is_dataclass_instance` suggested as a [recipe](https://docs.python.org/3/library/dataclasses.html#dataclasses.is_dataclass) for dataclasses built-in
- The generated comparison methods (when `order=True`) are compatible with supertypes and subtypes of the class. This means that heterogeneous collections of instances with the same superclass can be sorted
It is also worth noting that internally, dataclasses and dataclassy work in different ways. You can think of dataclassy as _turning your class into a different type of thing_ (indeed, it uses a metaclass) and dataclasses as merely _adding things to your class_ (it does not).
### Examples
#### The basics
To define a data class, simply apply the `@dataclass` decorator to a class definition ([see above](#dataclassy)).
Without arguments to the decorator, the resulting class will behave very similarly to its equivalent from the built-in module. However, dataclassy's decorator has some additional options over dataclasses', and it is also inherited so that subclasses of data classes are automatically data classes too.
The decorator generates various methods for the class. Which ones exactly depend on the options to the decorator. For example, `@dataclass(repr=False)` will prevent a `__repr__` method from being generated. `@dataclass` is equivalent to using the decorator with default parameters (i.e. `@dataclass` and `@dataclass()` are equivalent). Options to the decorator are detailed fully in the [next section](#decorator-options).
#### Class variables
You can exclude a class attribute from dataclassy's mechanisms entirely by simply defining it without a type annotation. This can be used for class variables and constants.
#### Parameter reordering
dataclassy modifies the order of fields when converting them into parameters for the generated `__init__`. Specifically, fields with default values always follow those without them. This stems from Python's requirement that _parameters_ with default arguments follow those without them. Conceptually, you can think of the process to generate the parameter list like this:
1. dataclassy takes the fields in definition order
2. it splits them into two lists, the first being fields without default values and the second being fields with them
3. it appends the second list to the first
This simple design decision prevents the dreaded `TypeError: non-default argument '...' follows default argument` error that anyone who has tried to do serious inheritance using dataclasses will know well.
You can verify the signature of the generated initialiser for any class using `signature` from the `inspect` module. For example, using the definition linked to above, `inspect.signature(Pet)` will return `(name: str, species: str, fluffy: bool, foods: Dict[str, int] = {}))`.
If we then decided to subclass `Pet` to add a new field, `hungry`:
```python
@dataclass
class HungryPet(Pet):
hungry: bool
```
You will see that `inspect.signature(HungryPet)` returns `(name: str, species: str, fluffy: bool, hungry: bool, foods: Dict[str, int] = {})`.
#### Inheritance
Unlike dataclasses, dataclassy's decorator only needs to be applied once, and all subclasses will become data classes with the same options as the parent class. The decorator can still be reapplied to subclasses in order to apply new parameters.
To change the type, or to add or change the default value of a field in a subclass, simply redeclare it in the subclass.
#### Post-initialisation processing
If an initialiser is requested (`init=True`), dataclassy automatically sets the attributes of the class upon initialisation. You can define code that should run after this happens - this is called _post-init processing_.
The method that contains this logic should be called `__post_init__`. Like with dataclasses, if `init=False` or the class has no fields, `__post_init__` will not be called.
```Python
@dataclass
class CustomInit:
a: int
b: int
def __post_init__(self):
self.c = self.a / self.b
```
In this example, when the class is instantiated with `CustomInit(1, 2)`, the field `c` is calculated as `0.5`.
Like with any function, your `__post_init__` can also take parameters which exist only in the context of `__post_init__`. These can be used for arguments to the class that you do not want to store as fields. A parameter cannot have the name of a class field; this is to prevent ambiguity.
#### Default values
Default values for fields work exactly as default arguments to functions (and in fact this is how they are implemented), with one difference: for copyable defaults, a copy is automatically created for each class instance. This means that a new copy of the `list` field `foods` in `Pet` above will be created each time it is instantiated, so that appending to that attribute in one instance will not affect other instances. A "copyable default" is defined as any object implementing a `copy` method, which includes all the built-in mutable collections (including `defaultdict`).
If you want to create new instances of objects which do not have a `copy` method, use the [`factory`](#factorycreator) function. This function takes any zero-argument callable. When the class is instantiated, this callable is executed to produce a default value for the field:
```Python
class MyClass:
pass
@dataclass
class CustomDefault:
m: MyClass = factory(MyClass)
CustomDefault() # CustomDefault(m=<__main__.MyClass object at 0x7f8b156feb50>)
CustomDefault() # CustomDefault(m=<__main__.MyClass object at 0x7f8b156fc7d0>)
```
## API
### Decorator
#### `@dataclass(init=True, repr=True, eq=True, order=False, unsafe_hash=False, frozen=False, hide_internals=True, iter=False, kwargs=False, slots=False, meta=DataClassMeta)`
The decorator used to signify that a class definition should become a data class. The decorator returns a new data class with generated methods as detailed below. If the class already defines a particular method, it will not be replaced with a generated one.
Without arguments, its behaviour is, superficially, almost identical to its equivalent in the built-in module. However, dataclassy's decorator only needs to be applied once, and all subclasses will become data classes with the same parameters. The decorator can still be reapplied to subclasses in order to change parameters.
A data class' fields are defined using Python's type annotations syntax. To change the type or default value of a field in a subclass, simply redeclare it.
This decorator takes advantage of two equally important features added in Python 3.6: [variable annotations](https://docs.python.org/3/glossary.html#term-variable-annotation) and [dictionaries being ordered](https://docs.python.org/3.7/tutorial/datastructures.html#dictionaries). (The latter is technically an [implementation detail](https://docs.python.org/3.6/whatsnew/3.6.html#whatsnew36-compactdict) of Python 3.6, only becoming standardised in Python 3.7, but is the case for all current implementations of Python 3.6, i.e. CPython and PyPy.)
#### Decorator options
> The term "field", as used in this section, refers to a class-level variable with a type annotation. For more information, see the documentation for [`fields()`](#fieldsdataclass-internalsfalse) below.
##### `init`
If true (the default), generate an [`__init__`](https://docs.python.org/3/reference/datamodel.html#object.__init__) method that has as parameters all fields up its inheritance chain. These are ordered in definition order, with all fields with default values placed towards the end, following all fields without them. The method initialises the class by applying these parameters to the class as attributes. If defined, it will also call `__post_init__` with any remaining arguments.
This ordering is an important distinction from dataclasses, where all fields are simply ordered in definition order, and is what allows dataclassy's data classes to be far more flexible in terms of inheritance.
A shallow copy will be created for mutable arguments (defined as those defining a `copy` method). This means that default field values that are mutable (e.g. a list) will not be mutated between instances.
##### `repr`
If true (the default), generate a [`__repr__`](https://docs.python.org/3/reference/datamodel.html#object.__repr__) method that displays all fields (or if [`hide_internals`](#hide_internals) is true, all fields excluding [internal](#internal) ones) of the data class instance and their values.
##### `eq`
If true (the default), generate an [`__eq__`](https://docs.python.org/3/reference/datamodel.html#object.__eq__) method that compares this data class to another of the same type as if they were tuples created by [`as_tuple`](#as_tupledataclass), excluding [internal fields](#internal) if [`hide_internals`](#hide_internals) is true.
##### `order`
If true, a [`__lt__`](https://docs.python.org/3/reference/datamodel.html#object.__lt__) method is generated, making the class *orderable*. If `eq` is also true, all other comparison methods are also generated. These methods compare this data class to another of the same type (or a subclass) as if they were tuples created by [`as_tuple`](#as_tupledataclass), excluding [internal fields](#internal) if [`hide_internals`](#hide_internals) is true. The normal rules of [lexicographical comparison](https://docs.python.org/3/reference/expressions.html#value-comparisons) apply.
##### `unsafe_hash`
If true, force the generation of a [`__hash__`](https://docs.python.org/3/reference/datamodel.html#object.__hash__) method that attempts to hash the class as if it were a tuple of its hashable fields. If `unsafe_hash` is false, `__hash__` will only be generated if `eq` and `frozen` are both true.
##### `match_args`
If true (the default), generate a `__match_args__` attribute that enables structural pattern matching on Python 3.10+.
##### `kw_only`
If true, all parameters to the generated `__init__` are marked as [keyword-only](https://peps.python.org/pep-3102). This includes arguments passed through to `__post_init__`.
##### `frozen`
If true, instances are nominally immutable: fields cannot be overwritten or deleted after initialisation in `__init__`. Attempting to do so will raise an `AttributeError`. **Warning: incurs a significant initialisation performance penalty.**
##### `hide_internals`
If true (the default), [internal fields](#internal) are not included in the generated [`__repr__`](#repr), comparison functions ([`__eq__`](#eq), [ `__lt__`](#order), etc.), or [`__iter__`](#iter).
##### `iter`
If true, generate an [`__iter__`](https://docs.python.org/3/reference/datamodel.html#object.__iter__) method that returns the values of the class's fields, in order of definition, noting that [internal fields](#internal) are excluded when [`hide_internals`](#hide_internals) is true. This can be used to destructure a data class instance, as with a Scala `case class` or a Python `namedtuple`.
##### `kwargs`
If true, add [`**kwargs`](https://docs.python.org/3.3/glossary.html#term-parameter) to the end of the parameter list for `__init__`. This simplifies data class instantiation from dictionaries that may have keys in addition to the fields of the data class (i.e. `SomeDataClass(**some_dict)`).
##### `slots`
If true, generate a [`__slots__`](https://docs.python.org/3/reference/datamodel.html#slots) attribute for the class. This reduces the memory footprint of instances and attribute lookup overhead. However, `__slots__` come with a few [restrictions](https://docs.python.org/3/reference/datamodel.html#notes-on-using-slots) (for example, multiple inheritance becomes tricky) that you should be aware of.
##### `meta`
Set this parameter to use a metaclass other than dataclassy's own. This metaclass must subclass [`dataclassy.dataclass.DataClassMeta`](dataclassy/dataclass.py).
`DataClassMeta` is best considered less stable than the parts of the library available in the root namespace. Only use a custom metaclass if absolutely necessary.
### Functions
#### `factory(producer)`
Takes a zero-argument callable and creates a _factory_ that executes this callable to generate a default value for the field at class initialisation time.
#### `is_dataclass(obj)`
Returns True if `obj` is a data class as implemented in this module.
#### `is_dataclass_instance(obj)`
Returns True if `obj` is an instance of a data class as implemented in this module.
#### `fields(dataclass, internals=False)`
Return a dict of `dataclass`'s fields and their types. `internals` selects whether to include internal fields. `dataclass` can be either a data class or an instance of a data class.
A field is defined as a class-level variable with a [type annotation](https://docs.python.org/3/glossary.html#term-variable-annotation). Variables defined in the class without type annotations are completely excluded from dataclassy's consideration. Class variables and constants can therefore be indicated by the absence of type annotations.
#### `values(dataclass, internals=False)`
Return a dict of `dataclass`'s fields and their values. `internals` selects whether to include internal fields. `dataclass` must be an instance of a data class.
#### `as_dict(dataclass dict_factory=dict)`
Recursively create a dict of a data class instance's fields and their values.
This function is recursively called on data classes, named tuples and iterables.
#### `as_tuple(dataclass)`
Recursively create a tuple of the values of a data class instance's fields, in definition order.
This function is recursively called on data classes, named tuples and iterables.
#### `make_dataclass(name, fields, defaults, bases=(), **options)`
Dynamically create a data class with name `name`, fields `fields`, default field values `defaults` and inheriting from `bases`.
#### `replace(dataclass, **changes)`
Return a new copy of `dataclass` with field values replaced as specified in `changes`.
### Type hints
#### `Internal`
The `Internal` type wrapper marks a field as being "internal" to the data class. Fields which begin with the ["internal use"](https://www.python.org/dev/peps/pep-0008/#descriptive-naming-styles) idiomatic indicator `_` or the [private field](https://docs.python.org/3/tutorial/classes.html#private-variables) interpreter indicator `__` are automatically treated as internal fields. The `Internal` type wrapper therefore serves as an alternative method of indicating that a field is internal for situations where you are unable to name your fields in this way.
#### `Hashed`
Use `Hashed` to wrap the type annotations of fields that you want to be included in a data class' `__hash__`. The value hashed by `__hash__` consists of a tuple of the instance's type followed by any fields marked as `Hashed`.
#### `DataClass`
Use this type hint to indicate that a variable, parameter or field should be a generic data class instance. For example, dataclassy uses these in the signatures of `as_dict`, `as_tuple` and `values` to show that these functions should be called on data class instances.
### Mypy support
In order to use dataclassy in projects with mypy, you will need to use the mypy plugin. You can create a `mypy.ini` or `.mypy.ini` for such projects with the following content:
```ini
[mypy]
plugins = dataclassy.mypy
```
Raw data
{
"_id": null,
"home_page": "https://github.com/biqqles/dataclassy",
"name": "dataclassy",
"maintainer": "",
"docs_url": null,
"requires_python": ">=3.6",
"maintainer_email": "",
"keywords": "",
"author": "biqqles",
"author_email": "biqqles@proton.me",
"download_url": "https://files.pythonhosted.org/packages/44/c0/a4a376e62b566708dbd157af4ad160406b82bcd9559b2eb38c66093d4e8b/dataclassy-1.0.1.tar.gz",
"platform": null,
"description": "# dataclassy\n**dataclassy** is a reimplementation of data classes in Python - an alternative to the built-in [dataclasses module](https://docs.python.org/3/library/dataclasses.html) that avoids many of [its](https://stackoverflow.com/questions/54678337) [common](https://stackoverflow.com/q/51575931) [pitfalls](https://stackoverflow.com/q/50180735). dataclassy is designed to be more flexible, less verbose, and more powerful than dataclasses, while retaining a familiar interface.\n\nIn short, dataclassy is a library for moving data around your Python programs that's optimised for speed, simplicity and developer happiness.\n\n```python\nfrom dataclassy import dataclass\nfrom typing import Dict\n\n@dataclass\nclass Pet:\n name: str\n species: str\n fluffy: bool\n foods: Dict[str, int] = {}\n```\n\n\n## Why use dataclassy?\nThis section describes various motivations for using **dataclassy** over **dataclasses**.\n\n#### Backwards compatibility\ndataclassy implements the decorator options of the latest version of dataclasses, plus its own, such that they are compatible back to Python 3.6. It is [tested against](.github/workflows/ci.yml) **CPython 3.6 through 3.11** and **PyPy 3.6 through 3.9**.\n\n#### Upgrade your data classes\n- new decorator options\n - [`slots`](#slots) generates `__slots__` to reduce memory footprint and improve attribute lookup performance\n - [`kwargs`](#kwargs) appends `**kwargs` to `__init__`'s parameter list to consume unexpected arguments\n - [`iter`](#iter) allows class instances to be destructured, like named tuples\n - [`hide_internals`](#hide_internals) automatically hides private fields from `__repr__` and excludes them from comparison and iteration\n- `@dataclass` usage and options are inherited (subclasses do not have to reuse the decorator)\n- fields can be in any order - fields with defaults are [reordered](#parameter-reordering) - making inheritance feasible\n- mutable default values (`list`, `set`, `dict` and more) are [automatically copied](#default-values) upon initialisation\n- new functions: [`is_dataclass_instance`](#is_dataclass_instanceobj) and [`values`](#valuesdataclass-internalsfalse)\n\n#### Additionally, dataclassy\n- implements all the decorator options and functions of dataclasses\n- is tiny (~160 LOC; less than 25% the size of dataclasses)\n- has no external dependencies, and no stdlib imports other than `types`, `typing` and `reprlib`\n- is [fast](benchmarks.py), matching dataclasses' performance when `slots=False` and significantly exceeding it when `slots=True`\n- supports multiple inheritance and custom metaclasses\n- comes with [support for mypy](#mypy-support)\n\n#### Other differences\ndataclassy's API is strongly influenced by dataclasses', but with a focus on minimalism and elegance.\n\n- there's no `field` or `Field`. Use [`Hashed`](#hashed), [`Internal`](#internal) or [`factory`](#factorycreator) to replicate its functions\n- there's no `InitVar`. Simply add arguments to `__post_init__`\n- there's no need for `ClassVar`. Simply omit the field's type hint to ignore it\n\n#### Also consider\n\n- [`attrs`](https://github.com/python-attrs/attrs) if you need complex validation and type conversions\n- [`pydantic`](https://github.com/samuelcolvin/pydantic) if you need strict type checking\n\n\n## Usage\n### Installation\nInstall the latest release from [PyPI](https://pypi.org/project/dataclassy/) with pip:\n\n```console\npip install dataclassy\n```\n\nOr install the latest development version straight from this repository:\n\n```console\npip install https://github.com/biqqles/dataclassy/archive/master.zip -U\n```\n\n\n### Migration\ndataclassy's API is broadly similar to dataclasses. If you simply use the decorator and other functions, it is possible to instantly migrate from dataclasses to dataclassy by simply changing\n\n```Python\nfrom dataclasses import *\n```\n\nto\n\n```Python\nfrom dataclassy import *\n```\n\nOtherwise, you will have to make a couple of easy refactorings (that should leave you with cleaner code!). Consult the table under [Differences](#differences) or skip ahead to [Examples](#examples) to see dataclassy in action.\n\n#### Similarities\ndataclassy's `dataclass` decorator takes all of the same arguments as dataclasses', plus its own, and should therefore be a drop-in replacement.\n\ndataclassy also implements all dataclasses' [functions](#functions): `is_dataclass`, `fields`, `replace`, `make_dataclass`, `asdict` and `astuple` (the last two are aliased from `as_dict` and `as_tuple` respectively), and they should work as you expect.\n\n#### Differences\nAlthough dataclassy's API is similar to dataclasses', [compatibility with it is not a goal](https://straight-shoota.github.io/crystal-book/feature/tutorials-initial/) (this is similar to the relationship between Crystal and Ruby).\n\ndataclassy has several important differences from dataclasses, mainly reflective of its minimalistic style and implementation. These differences are enumerated below and fully expanded on in the next section.\n\n| |dataclasses |dataclassy |\n|---------------------------------|:-------------------------------------------|:---------------------------------------|\n|*init-only variables* |fields with type `InitVar` |arguments to `__post_init__` |\n|*class variables* |fields with type `ClassVar` |fields without type annotation |\n|*mutable defaults* |`a: Dict = field(default_factory=dict)` |`a: Dict = {}` |\n|*dynamic defaults* |`b: MyClass = field(default_factory=MyClass)`|`b: MyClass = factory(MyClass)` |\n|*field excluded from `repr`* |`c: int = field(repr=False)` |`Internal` type wrapper or `_name` |\n|*\"late init\" field* |`d: int = field(init=False)` |`d: int = None` |\n|*abstract data class* |`class Foo(ABC):` |`class Foo(metaclass=ABCMeta):` |\n\nThere are a couple of minor differences, too:\n\n- `fields` returns `Dict[str, Type]` instead of `Dict[Field, Type]` and has an additional parameter which filters internal fields\n- Attempting to modify a frozen instance raises `AttributeError` with an explanation rather than `FrozenInstanceError`\n\nFinally, there are some quality of life improvements that, while not being directly implicated in migration, will allow you to make your code cleaner:\n\n- `@dataclass` does not need to be applied to every subclass - its behaviour and options are inherited\n- Unlike dataclasses, fields with defaults do not need to follow those without them. This is particularly useful when working with subclasses, which is almost impossible with dataclasses\n- dataclassy adds a `DataClass` type annotation to represent variables that should be generic data class instances\n- dataclassy has the `is_dataclass_instance` suggested as a [recipe](https://docs.python.org/3/library/dataclasses.html#dataclasses.is_dataclass) for dataclasses built-in\n- The generated comparison methods (when `order=True`) are compatible with supertypes and subtypes of the class. This means that heterogeneous collections of instances with the same superclass can be sorted\n\nIt is also worth noting that internally, dataclasses and dataclassy work in different ways. You can think of dataclassy as _turning your class into a different type of thing_ (indeed, it uses a metaclass) and dataclasses as merely _adding things to your class_ (it does not).\n\n\n### Examples\n#### The basics\nTo define a data class, simply apply the `@dataclass` decorator to a class definition ([see above](#dataclassy)).\n\nWithout arguments to the decorator, the resulting class will behave very similarly to its equivalent from the built-in module. However, dataclassy's decorator has some additional options over dataclasses', and it is also inherited so that subclasses of data classes are automatically data classes too.\n\nThe decorator generates various methods for the class. Which ones exactly depend on the options to the decorator. For example, `@dataclass(repr=False)` will prevent a `__repr__` method from being generated. `@dataclass` is equivalent to using the decorator with default parameters (i.e. `@dataclass` and `@dataclass()` are equivalent). Options to the decorator are detailed fully in the [next section](#decorator-options).\n\n#### Class variables\nYou can exclude a class attribute from dataclassy's mechanisms entirely by simply defining it without a type annotation. This can be used for class variables and constants.\n\n#### Parameter reordering\ndataclassy modifies the order of fields when converting them into parameters for the generated `__init__`. Specifically, fields with default values always follow those without them. This stems from Python's requirement that _parameters_ with default arguments follow those without them. Conceptually, you can think of the process to generate the parameter list like this:\n\n1. dataclassy takes the fields in definition order\n2. it splits them into two lists, the first being fields without default values and the second being fields with them\n3. it appends the second list to the first\n\nThis simple design decision prevents the dreaded `TypeError: non-default argument '...' follows default argument` error that anyone who has tried to do serious inheritance using dataclasses will know well.\n\nYou can verify the signature of the generated initialiser for any class using `signature` from the `inspect` module. For example, using the definition linked to above, `inspect.signature(Pet)` will return `(name: str, species: str, fluffy: bool, foods: Dict[str, int] = {}))`.\n\nIf we then decided to subclass `Pet` to add a new field, `hungry`:\n\n```python\n@dataclass\nclass HungryPet(Pet):\n hungry: bool\n```\n\nYou will see that `inspect.signature(HungryPet)` returns `(name: str, species: str, fluffy: bool, hungry: bool, foods: Dict[str, int] = {})`.\n\n#### Inheritance\nUnlike dataclasses, dataclassy's decorator only needs to be applied once, and all subclasses will become data classes with the same options as the parent class. The decorator can still be reapplied to subclasses in order to apply new parameters.\n\nTo change the type, or to add or change the default value of a field in a subclass, simply redeclare it in the subclass.\n\n#### Post-initialisation processing\nIf an initialiser is requested (`init=True`), dataclassy automatically sets the attributes of the class upon initialisation. You can define code that should run after this happens - this is called _post-init processing_. \n\nThe method that contains this logic should be called `__post_init__`. Like with dataclasses, if `init=False` or the class has no fields, `__post_init__` will not be called.\n\n```Python\n@dataclass\nclass CustomInit:\n a: int\n b: int\n \n def __post_init__(self):\n self.c = self.a / self.b\n```\n\nIn this example, when the class is instantiated with `CustomInit(1, 2)`, the field `c` is calculated as `0.5`.\n\nLike with any function, your `__post_init__` can also take parameters which exist only in the context of `__post_init__`. These can be used for arguments to the class that you do not want to store as fields. A parameter cannot have the name of a class field; this is to prevent ambiguity.\n\n#### Default values\nDefault values for fields work exactly as default arguments to functions (and in fact this is how they are implemented), with one difference: for copyable defaults, a copy is automatically created for each class instance. This means that a new copy of the `list` field `foods` in `Pet` above will be created each time it is instantiated, so that appending to that attribute in one instance will not affect other instances. A \"copyable default\" is defined as any object implementing a `copy` method, which includes all the built-in mutable collections (including `defaultdict`).\n\nIf you want to create new instances of objects which do not have a `copy` method, use the [`factory`](#factorycreator) function. This function takes any zero-argument callable. When the class is instantiated, this callable is executed to produce a default value for the field:\n\n```Python\nclass MyClass:\n pass\n\n@dataclass\nclass CustomDefault:\n m: MyClass = factory(MyClass)\n\nCustomDefault() # CustomDefault(m=<__main__.MyClass object at 0x7f8b156feb50>)\nCustomDefault() # CustomDefault(m=<__main__.MyClass object at 0x7f8b156fc7d0>)\n```\n\n## API\n### Decorator\n#### `@dataclass(init=True, repr=True, eq=True, order=False, unsafe_hash=False, frozen=False, hide_internals=True, iter=False, kwargs=False, slots=False, meta=DataClassMeta)`\nThe decorator used to signify that a class definition should become a data class. The decorator returns a new data class with generated methods as detailed below. If the class already defines a particular method, it will not be replaced with a generated one.\n\nWithout arguments, its behaviour is, superficially, almost identical to its equivalent in the built-in module. However, dataclassy's decorator only needs to be applied once, and all subclasses will become data classes with the same parameters. The decorator can still be reapplied to subclasses in order to change parameters.\n\nA data class' fields are defined using Python's type annotations syntax. To change the type or default value of a field in a subclass, simply redeclare it.\n\nThis decorator takes advantage of two equally important features added in Python 3.6: [variable annotations](https://docs.python.org/3/glossary.html#term-variable-annotation) and [dictionaries being ordered](https://docs.python.org/3.7/tutorial/datastructures.html#dictionaries). (The latter is technically an [implementation detail](https://docs.python.org/3.6/whatsnew/3.6.html#whatsnew36-compactdict) of Python 3.6, only becoming standardised in Python 3.7, but is the case for all current implementations of Python 3.6, i.e. CPython and PyPy.)\n\n\n#### Decorator options\n> The term \"field\", as used in this section, refers to a class-level variable with a type annotation. For more information, see the documentation for [`fields()`](#fieldsdataclass-internalsfalse) below.\n\n##### `init`\nIf true (the default), generate an [`__init__`](https://docs.python.org/3/reference/datamodel.html#object.__init__) method that has as parameters all fields up its inheritance chain. These are ordered in definition order, with all fields with default values placed towards the end, following all fields without them. The method initialises the class by applying these parameters to the class as attributes. If defined, it will also call `__post_init__` with any remaining arguments.\n\nThis ordering is an important distinction from dataclasses, where all fields are simply ordered in definition order, and is what allows dataclassy's data classes to be far more flexible in terms of inheritance. \n\nA shallow copy will be created for mutable arguments (defined as those defining a `copy` method). This means that default field values that are mutable (e.g. a list) will not be mutated between instances.\n\n##### `repr`\nIf true (the default), generate a [`__repr__`](https://docs.python.org/3/reference/datamodel.html#object.__repr__) method that displays all fields (or if [`hide_internals`](#hide_internals) is true, all fields excluding [internal](#internal) ones) of the data class instance and their values.\n\n##### `eq`\nIf true (the default), generate an [`__eq__`](https://docs.python.org/3/reference/datamodel.html#object.__eq__) method that compares this data class to another of the same type as if they were tuples created by [`as_tuple`](#as_tupledataclass), excluding [internal fields](#internal) if [`hide_internals`](#hide_internals) is true.\n\n##### `order`\nIf true, a [`__lt__`](https://docs.python.org/3/reference/datamodel.html#object.__lt__) method is generated, making the class *orderable*. If `eq` is also true, all other comparison methods are also generated. These methods compare this data class to another of the same type (or a subclass) as if they were tuples created by [`as_tuple`](#as_tupledataclass), excluding [internal fields](#internal) if [`hide_internals`](#hide_internals) is true. The normal rules of [lexicographical comparison](https://docs.python.org/3/reference/expressions.html#value-comparisons) apply.\n\n##### `unsafe_hash`\nIf true, force the generation of a [`__hash__`](https://docs.python.org/3/reference/datamodel.html#object.__hash__) method that attempts to hash the class as if it were a tuple of its hashable fields. If `unsafe_hash` is false, `__hash__` will only be generated if `eq` and `frozen` are both true.\n\n##### `match_args`\nIf true (the default), generate a `__match_args__` attribute that enables structural pattern matching on Python 3.10+.\n\n##### `kw_only`\nIf true, all parameters to the generated `__init__` are marked as [keyword-only](https://peps.python.org/pep-3102). This includes arguments passed through to `__post_init__`.\n\n##### `frozen`\nIf true, instances are nominally immutable: fields cannot be overwritten or deleted after initialisation in `__init__`. Attempting to do so will raise an `AttributeError`. **Warning: incurs a significant initialisation performance penalty.**\n\n##### `hide_internals`\nIf true (the default), [internal fields](#internal) are not included in the generated [`__repr__`](#repr), comparison functions ([`__eq__`](#eq), [ `__lt__`](#order), etc.), or [`__iter__`](#iter).\n\n##### `iter`\nIf true, generate an [`__iter__`](https://docs.python.org/3/reference/datamodel.html#object.__iter__) method that returns the values of the class's fields, in order of definition, noting that [internal fields](#internal) are excluded when [`hide_internals`](#hide_internals) is true. This can be used to destructure a data class instance, as with a Scala `case class` or a Python `namedtuple`.\n\n##### `kwargs`\nIf true, add [`**kwargs`](https://docs.python.org/3.3/glossary.html#term-parameter) to the end of the parameter list for `__init__`. This simplifies data class instantiation from dictionaries that may have keys in addition to the fields of the data class (i.e. `SomeDataClass(**some_dict)`).\n\n##### `slots`\nIf true, generate a [`__slots__`](https://docs.python.org/3/reference/datamodel.html#slots) attribute for the class. This reduces the memory footprint of instances and attribute lookup overhead. However, `__slots__` come with a few [restrictions](https://docs.python.org/3/reference/datamodel.html#notes-on-using-slots) (for example, multiple inheritance becomes tricky) that you should be aware of.\n\n##### `meta`\nSet this parameter to use a metaclass other than dataclassy's own. This metaclass must subclass [`dataclassy.dataclass.DataClassMeta`](dataclassy/dataclass.py).\n\n`DataClassMeta` is best considered less stable than the parts of the library available in the root namespace. Only use a custom metaclass if absolutely necessary.\n\n\n### Functions\n#### `factory(producer)`\nTakes a zero-argument callable and creates a _factory_ that executes this callable to generate a default value for the field at class initialisation time.\n\n#### `is_dataclass(obj)`\nReturns True if `obj` is a data class as implemented in this module.\n\n#### `is_dataclass_instance(obj)`\nReturns True if `obj` is an instance of a data class as implemented in this module.\n\n#### `fields(dataclass, internals=False)`\nReturn a dict of `dataclass`'s fields and their types. `internals` selects whether to include internal fields. `dataclass` can be either a data class or an instance of a data class.\n\nA field is defined as a class-level variable with a [type annotation](https://docs.python.org/3/glossary.html#term-variable-annotation). Variables defined in the class without type annotations are completely excluded from dataclassy's consideration. Class variables and constants can therefore be indicated by the absence of type annotations.\n\n#### `values(dataclass, internals=False)`\nReturn a dict of `dataclass`'s fields and their values. `internals` selects whether to include internal fields. `dataclass` must be an instance of a data class.\n\n#### `as_dict(dataclass dict_factory=dict)`\nRecursively create a dict of a data class instance's fields and their values.\n\nThis function is recursively called on data classes, named tuples and iterables.\n\n#### `as_tuple(dataclass)`\nRecursively create a tuple of the values of a data class instance's fields, in definition order.\n\nThis function is recursively called on data classes, named tuples and iterables.\n\n#### `make_dataclass(name, fields, defaults, bases=(), **options)`\nDynamically create a data class with name `name`, fields `fields`, default field values `defaults` and inheriting from `bases`.\n\n#### `replace(dataclass, **changes)`\nReturn a new copy of `dataclass` with field values replaced as specified in `changes`.\n\n### Type hints\n#### `Internal`\nThe `Internal` type wrapper marks a field as being \"internal\" to the data class. Fields which begin with the [\"internal use\"](https://www.python.org/dev/peps/pep-0008/#descriptive-naming-styles) idiomatic indicator `_` or the [private field](https://docs.python.org/3/tutorial/classes.html#private-variables) interpreter indicator `__` are automatically treated as internal fields. The `Internal` type wrapper therefore serves as an alternative method of indicating that a field is internal for situations where you are unable to name your fields in this way.\n\n#### `Hashed`\nUse `Hashed` to wrap the type annotations of fields that you want to be included in a data class' `__hash__`. The value hashed by `__hash__` consists of a tuple of the instance's type followed by any fields marked as `Hashed`.\n\n#### `DataClass`\nUse this type hint to indicate that a variable, parameter or field should be a generic data class instance. For example, dataclassy uses these in the signatures of `as_dict`, `as_tuple` and `values` to show that these functions should be called on data class instances.\n\n### Mypy support\nIn order to use dataclassy in projects with mypy, you will need to use the mypy plugin. You can create a `mypy.ini` or `.mypy.ini` for such projects with the following content:\n\n```ini\n[mypy]\nplugins = dataclassy.mypy\n```\n",
"bugtrack_url": null,
"license": "",
"summary": "A fast and flexible reimplementation of data classes",
"version": "1.0.1",
"split_keywords": [],
"urls": [
{
"comment_text": "",
"digests": {
"blake2b_256": "a3968378dfb8668c1d26533e310b2b062ebbd6f02a2acbb172f3efaa46d126ad",
"md5": "c89a10d3fa60ad68a0d766163326dce2",
"sha256": "445040d809cb3bdabd420a2d9db1cb1c9bbee185ed82e9ab966977f9721d8af0"
},
"downloads": -1,
"filename": "dataclassy-1.0.1-py3-none-any.whl",
"has_sig": false,
"md5_digest": "c89a10d3fa60ad68a0d766163326dce2",
"packagetype": "bdist_wheel",
"python_version": "py3",
"requires_python": ">=3.6",
"size": 24226,
"upload_time": "2023-01-14T12:47:56",
"upload_time_iso_8601": "2023-01-14T12:47:56.879714Z",
"url": "https://files.pythonhosted.org/packages/a3/96/8378dfb8668c1d26533e310b2b062ebbd6f02a2acbb172f3efaa46d126ad/dataclassy-1.0.1-py3-none-any.whl",
"yanked": false,
"yanked_reason": null
},
{
"comment_text": "",
"digests": {
"blake2b_256": "44c0a4a376e62b566708dbd157af4ad160406b82bcd9559b2eb38c66093d4e8b",
"md5": "cd1b077f5c322b11d162068fc8725f13",
"sha256": "e5a08a304f5f31b35983d3d14e60f00240fbcb38b4aea49db7d19160958b9a3d"
},
"downloads": -1,
"filename": "dataclassy-1.0.1.tar.gz",
"has_sig": false,
"md5_digest": "cd1b077f5c322b11d162068fc8725f13",
"packagetype": "sdist",
"python_version": "source",
"requires_python": ">=3.6",
"size": 28786,
"upload_time": "2023-01-14T12:47:59",
"upload_time_iso_8601": "2023-01-14T12:47:59.367470Z",
"url": "https://files.pythonhosted.org/packages/44/c0/a4a376e62b566708dbd157af4ad160406b82bcd9559b2eb38c66093d4e8b/dataclassy-1.0.1.tar.gz",
"yanked": false,
"yanked_reason": null
}
],
"upload_time": "2023-01-14 12:47:59",
"github": true,
"gitlab": false,
"bitbucket": false,
"github_user": "biqqles",
"github_project": "dataclassy",
"travis_ci": false,
"coveralls": false,
"github_actions": true,
"lcname": "dataclassy"
}