https://github.com/deniz195/unitdoc
A framework that enables classes that describe physical objects with units and easy serialization.
https://github.com/deniz195/unitdoc
attributes attrs boilerplate classes deserialisation python science serialization units yaml
Last synced: 4 months ago
JSON representation
A framework that enables classes that describe physical objects with units and easy serialization.
- Host: GitHub
- URL: https://github.com/deniz195/unitdoc
- Owner: deniz195
- License: mit
- Created: 2019-06-19T18:16:25.000Z (almost 7 years ago)
- Default Branch: master
- Last Pushed: 2021-02-16T09:05:07.000Z (about 5 years ago)
- Last Synced: 2025-10-28T16:32:08.540Z (7 months ago)
- Topics: attributes, attrs, boilerplate, classes, deserialisation, python, science, serialization, units, yaml
- Language: Python
- Size: 36.1 KB
- Stars: 0
- Watchers: 0
- Forks: 0
- Open Issues: 0
-
Metadata Files:
- Readme: README.md
- License: LICENSE
Awesome Lists containing this project
README
# Unitdoc


[](https://pypi.python.org/pypi/unitdoc/)
Unitdoc deals with data objects which describe physical objects, by providing properties with physical units and easy serialization and deserialization.
Let's look at an example. First, import unitdoc and create the registry that you will use in your application:
```python
from unitdoc import UnitDocRegistry
udr = UnitDocRegistry()
```
Let's create a class that represents a battery
```python
import attr
@udr.serialize()
@attr.s()
class Battery(object):
name = attr.ib()
weight = udr.attrib(default='45g')
volume = udr.attrib(default='16ml', default_unit='ml')
capacity = udr.attrib(default='3.0Ah')
voltage = udr.attrib(default='3.6V', description ='Average voltage')
```
Let's make a `Battery`
```python
a_battery = Battery(name = 'battery', weight='43g')
print(a_battery)
# outputs: Battery(name='battery', weight=, volume=, capacity=, voltage=)
```
Let's do interesting calculations:
```python
energy = (a_battery.capacity * a_battery.voltage).to('Wh')
print(f'{energy}')
# outputs: 10.8 Wh
```
... and more
```python
energy_density = (energy / a_battery.weight).to('Wh/kg')
print(f'{energy} @ {energy_density}')
# outputs: 10.8 Wh @ 251.2 Wh / kg
```
Let's save the battery to a file and reloaded again:
```python
fn = 'a_battery.yaml'
# save to yaml file
with open(fn, 'w') as f:
f.write(a_battery.serialize())
# load from yaml file
with open(fn, 'r') as f:
a_loaded_battery = Battery.deserialize(f.read())
assert a_battery == a_loaded_battery
```
If we look at the `a_battery.yaml` file, we will find:
```yaml
name: battery
weight: !unit 43 g
volume: !unit 16 ml
capacity: !unit 3 Ah
voltage: !unit 3.6 V
```
This serialization, we can directly get by
```python
# look at serialized form
print(a_battery.serialize())
# outputs:
name: battery
weight: !unit 43 g
volume: !unit 16 ml
capacity: !unit 3 Ah
voltage: !unit 3.6 V
```
Have fun!
## More features
Unitdoc facilitates certain operations, which can improve your code.
If you specify `default_unit` in an attribute, quantities are automatically normalized to that unit:
```python
a_battery = Battery(name = 'battery', volume='15903 mm^3')
print(a_battery.volume)
# outputs: 15.9 ml
```
If a `default_unit` is specified, any incompatible unit will raise an exception:
```python
from unitdoc import DimensionalityError
try:
a_battery = Battery(name = 'battery', volume='42 g')
except DimensionalityError as e:
print(e)
# outputs: Cannot convert from 'gram' ([mass]) to 'milliliter' ([length] ** 3)
```
You can retrieve description of parameters, for e.g. data representation code
```python
from unitdoc import get_attr_description
print(get_attr_description(a_battery.__class__, 'voltage'))
# outputs: Average voltage
```
Unitdoc uses the [attrs library](https://github.com/python-attrs/attrs)), check it out!
## Installation
Use the package manager [pip](https://pip.pypa.io/en/stable/) to install unitdoc:
```bash
pip install unitdoc
```
Alternatively, install the latest version from git:
```bash
git clone https://github.com/deniz195/unitdoc
python unitdoc/setup.py install --user
```
## Related packages
Unitdoc is based on the following amazing packages:
- [pint](https://pint.readthedocs.io/) deals with the units
- [ruamel.yamls](https://yaml.readthedocs.io/en/latest/) deals with (de)serializing from semi-structured data (nested dictionaries)
- [attrs](https://github.com/python-attrs/attrs) deals with the boilerplate of data classes
- [cattr](https://github.com/Tinche/cattrs) deals with the unstructuring and restructuring of classes for (de)serialization
The UnitDocRegistry creates registries/converters/parsers for each package and aggregates them. You can leverage the features of each package:
Use unit registry from pint:
```python
q = udr.ureg('1000gram').to('kg')
print(q)
# outputs: 1 kg
```
Use yaml parser from ruaml.yaml:
```python
q_yaml = udr.yaml.dump(dict(weight=q))
print(q_yaml)
# outputs: weight: !unit 1 kg
```
Use cattr converter:
```python
@udr.serialize()
@attr.s()
class Thing(object):
weight = udr.attrib(default='45g', description ='Total weight')
a_thing = Thing()
a_thing_dict = udr.cattr.unstructure(a_thing)
assert type(a_thing_dict) == dict
print(a_thing_dict['weight'])
# output: 45 g
```
## Restrictions
Given the restrictions of the attrs package, updating attributes safely requires certain precautions. E.g. given the `Battery` class from above the following is possible but not desirable
```python
a_battery = Battery(name = 'battery')
a_battery.volume = 99
type(a_battery.volume)
# outputs: int
```
This is not desirable, because unit check and normalizatin is not performed.
An good way to avoid this (and other problems) is to use keyword only (`kw_only=True`) and frozen (`frozen=True`) `attr` objects.
```python
@udr.serialize()
@attr.s(kw_only=True, frozen=True)
class BetterBattery(object):
name = attr.ib()
weight = udr.attrib(default='45g')
volume = udr.attrib(default='16ml', default_unit='ml')
capacity = udr.attrib(default='3.0Ah')
voltage = udr.attrib(default='3.6V', description ='Average voltage')
```
The keyword only restriction, will not allow the creation of objects from positional parameters, so that the following line fails with a Type error:
```python
a_battery = BetterBattery('battery', '42g', '16ml')
```
This is good, because positional arguments can be dangerous when data model changes over time. The following line creates a new object and is stable if the class changes
```python
a_battery = BetterBattery(name='battery', weight='42g', volume='16ml')
```
The frozen instance restriction does not allow to mutate an object, so that this line will fail with a `FrozenInstanceError`
```python
a_battery.volume = 99
```
To update values, you can use the attr.evolve function, which creates a new object with the updated value
```python
a_battery = attr.evolve(a_battery, volume='12cm^3')
```
In this case unit conversion and checks are performed as expected.
While unitdoc works with regular `attr` classes (`@attr.s()`), we strongly recommend using `@attr.s(kw_only=True, frozen=True)`.
## Contributing
Pull requests are welcome. For major changes, please open an issue first to discuss what you would like to change.
Please make sure to update tests as appropriate.
## License
[MIT](https://choosealicense.com/licenses/mit/)