Skip to main content

Data parsing and validation library for Python

Project description

PyPI - Version PyPI - Downloads PyPI - License

modelity

Data parsing and validation library for Python.

About

Modelity is a data parsing and validation library, written purely in Python, and based on the idea that data parsing and validation should be separated from each other, but being a part of single toolkit for ease of use.

In Modelity, data parsing is executed automatically once data model is instantiated or modified, while model validation needs to be explicitly called by the user. Thanks to this approach, models can be feed with data progressively (f.e. in response to user’s input), while still being able to validate at any time (f.e. in reaction to button click).

Features

  • Declare models using type annotations
  • Uses slots, not descriptors, making reading from a model as fast as possible
  • Clean separation between data parsing stage (executed when model is created or modified) and model validation stage (executed on demand)
  • Clean differentiation between unset fields (via special Unset sentinel) and optional fields set to None
  • Easily customizable via pre- and postprocessors (executed during data parsing), model-level validators, and field-level validators (both executed during model validation)
  • Ability do access any field via root model (the one for each validation is executed) from any custom validator, allowing to implement complex cross-field validation logic
  • Ability to add custom validation context for even more complex validation strategies (like having different validators when model is created, when model is updated or when model is fetched over the API).
  • Use of predefined error codes instead of error messages for easier customization of error reporting if needed
  • Ease of providing custom types simply by defining __modelity_type_descriptor__ static method in user-defined type, or by using type_descriptor_factory hook for registering 3rd party types.

Rationale

Why I have created this library?

First reason is that I didn’t find such clean separation in known data parsing tools, and found myself needing such freedom in several projects - both private, and commercial ones. Separation between parsing and validation steps simplifies validators, as validators in models can assume that they are called when model is successfully instantiated, with all fields parsed to their allowed types, therefore they can access all model’s fields without any extra checks.

Second reason is that I often found myself writing validation logic from the scratch for various reasons, especially for large models with lots of dependencies. Each time I had to validate some complex logic manually I was asking myself, why don’t merge all these ideas and make a library that already has these kind of helpers? For example, I sometimes needed to access parent model when validating field that itself is another, nested model. With Modelity, it is easy, as root model (the one that is validated) is populated to all nested models' validators recursively.

Third reason is that I wanted to finish my over 10 years old, abandoned project Formify (the name is already in use, so I have chosen new name for new project) which I was developing in free time at the beginning of my professional work as a Python developer. That project was originally made to handle form parsing and validation to be used along with web framework. Although the project was never finished, I’ve resurrected some ideas from it, especially parsing and validation separation. You can still find source code on my GitHub profile: https://github.com/mwiatrzyk.

And last but not least, I don’t intend to compete with any of the existing alternatives — and there are plenty of them. I simply created this project for fun and decided to release it once it became reasonably usable, hoping that maybe someone else will find it helpful.😊

Example

Here's a condensed example of how to use Modelity:

import json
import datetime
import typing

from modelity.api import Model, ValidationError, ModelError, validate, dump, load

# 1. Define models

class Address(Model):
    address_line1: str
    address_line2: typing.Optional[str]
    city: str
    state_province: typing.Optional[str]
    postal_code: str
    country_code: str

class Person(Model):
    name: str
    second_name: typing.Optional[str]
    surname: str
    dob: datetime.date
    address: Address


# 2. Create instances (parsing runs automatically)

addr = Address(
    address_line1="221B Baker Street",
    address_line2=None,
    city="London",
    state_province=None,
    postal_code="NW1 6XE",
    country_code="GB"
)

person = Person(
    name="Sherlock",
    second_name=None,
    surname="Holmes",
    dob=datetime.date(1854, 1, 6),
    address=addr
)

#: 3. Validate instances (on demand)

try:
    validate(person)
except ValidationError as e:
    print("Model is not valid: ", e)
    raise

# 4. Dump to JSON-serializable dict

person_dict = dump(person)
person_json = json.dumps(person_dict)  # Dump to JSON; use any lib you like to do that

# 5. Parse from dict

person_dict = json.loads(person_json)
try:
    same_person = load(Person, person_dict)  # Parsing + validation made by helper
except ModelError as e:  # Base for: ValidationError, ParsingError
    print("Model parsing or validation failed: ", e)
    raise

# 6. Accessing fields (just like using normal dataclasses).

print(same_person.address.country_code)

Documentation

Please visit project's ReadTheDocs site: https://modelity.readthedocs.io/en/latest/.

Disclaimer

Modelity is an independent open-source project for the Python ecosystem. It is not affiliated with, sponsored by, or endorsed by any company, organization, or product of the same or similar name. Any similarity in names is purely coincidental and does not imply any association.

License

This project is released under the terms of the MIT license.

Author

Maciej Wiatrzyk maciej.wiatrzyk@gmail.com

Project details


Download files

Download the file for your platform. If you're not sure which to choose, learn more about installing packages.

Source Distribution

modelity-0.33.0.tar.gz (38.6 kB view details)

Uploaded Source

Built Distribution

If you're not sure about the file name format, learn more about wheel file names.

modelity-0.33.0-py3-none-any.whl (46.4 kB view details)

Uploaded Python 3

File details

Details for the file modelity-0.33.0.tar.gz.

File metadata

  • Download URL: modelity-0.33.0.tar.gz
  • Upload date:
  • Size: 38.6 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? No
  • Uploaded via: poetry/2.3.2 CPython/3.13.12 Linux/6.8.0-1040-aws

File hashes

Hashes for modelity-0.33.0.tar.gz
Algorithm Hash digest
SHA256 35dba42b7f209bfebc56f4e5dbf8b8da2f76773ebf865893c56b25c161dab3a6
MD5 679abeb9bb7cd19a76bfc5b439fc2b0a
BLAKE2b-256 9e74bfc7702059e8fe4143a4103a2c41a91c336bf6ae6cada0c2dc6307369f9d

See more details on using hashes here.

File details

Details for the file modelity-0.33.0-py3-none-any.whl.

File metadata

  • Download URL: modelity-0.33.0-py3-none-any.whl
  • Upload date:
  • Size: 46.4 kB
  • Tags: Python 3
  • Uploaded using Trusted Publishing? No
  • Uploaded via: poetry/2.3.2 CPython/3.13.12 Linux/6.8.0-1040-aws

File hashes

Hashes for modelity-0.33.0-py3-none-any.whl
Algorithm Hash digest
SHA256 9f45ce589f629e15b70e778ebf4351dce3299df7903a4a58ff19c32ef73fe9a6
MD5 0260ff44ea86fdd985022555e54c7688
BLAKE2b-256 c3e2663fefb97704506bb3f2d873ea9356ec1cce4deb7086e7fd06cc83e9ffcf

See more details on using hashes here.

Supported by

AWS Cloud computing and Security Sponsor Datadog Monitoring Depot Continuous Integration Fastly CDN Google Download Analytics Pingdom Monitoring Sentry Error logging StatusPage Status page