Skip to main content

Django json field empowered by pydantic

Project description

Django Structured JSON Field PyPI Codecov GitHub Workflow Status GitHub Docs



Django Structured JSON Field logo





This is a Django field that allows you to declare the structure of a JSON field and validate it.

Features

  • Define the structure of a JSON field using Pydantic models
  • Validate the JSON field against the defined structure
  • Use relationships between models inside the JSON field 🤯
  • Easily integrate with Django Rest Framework serializers
  • Admin editor for the JSON field with autocomplete search for related models 👀

Documentation

Check out our documentation for detailed guides and examples on:

  • Installation and basic usage
  • Working with relationships
  • Admin integration
  • REST Framework integration
  • Caching
  • Settings configuration

Installation

pip install django-structured-json-field

Usage

from django.db import models
from structured.fields import StructuredJSONField
from structured.pydantic.models import BaseModel

# Define this schema as you would do with a Pydantic model
class MySchema(BaseModel):
    name: str
    age: int = None

def init_data():
    return MySchema(name='')

# Create a model with a StructuredJSONField with the schema you defined
class MyModel(models.Model):
    structured_data = StructuredJSONField(schema=MySchema, default=init_data)

Relationships

This field supports relationships between models, you can define them in your schema and they will be treated as normal django relationships. It also supports recursive schemas.

Recursion

You can define recursive schemas by declaring the attribute type as a string:

from typing import Optional, List

class MySchema(BaseModel):
    name: str
    age: int = None
    parent: Optional['MySchema'] = None
    relateds: List['MySchema'] = []

Foreign Keys

You can also define model relationships in your schema:

from structured.pydantic.fields import ForeignKey

class MySchema(BaseModel):
    name: str
    age: int = None
    fk_field: ForeignKey['MyModel'] = None

This will treat the parent field as a normal django ForeignKey.

Tip:

You can omit the ForeignKey field and just use the model class as the type annotation:

class MySchema(BaseModel):
    name: str
    age: int = None
    fk_field: MyModel = None

the field will still be treated as a ForeignKey if the type annotation is a subclass of django models.Model.

ManyToMany

If you need a ManyToMany relationship, you can use the QuerySet field:

from structured.pydantic.fields import QuerySet

class MySchema(BaseModel):
    name: str
    age: int = None
    parents: QuerySet['MyModel']

QuerySet fields will generate a django object manager that will allow you to query the related objects as you would do with a normal django QuerySet.

instance = MySchema(name='test', age=10, parents=MyModel.objects.all())
# You can filter the queryset
instance.parents.filter(name='test')
# You can count the queryset
instance.parents.count()
# You can get the first element of the queryset, etc...
instance.parents.first()

Admin integration

The field is integrated with the Django admin, you can use the autocomplete search to select the related models. To allow the autocomplete search you need to include structured.urls in your urls.py file:

from django.urls import path, include

urlpatterns = [
    path('admin/', admin.site.urls),
    path('', include('structured.urls')),
]

Rest Framework integration

You can easily integrate structured fields with Django Rest Framework serializers, just use the StructuredModelSerializer as the base class for your serializer:

from rest_framework import serializers
from structured.contrib.rest_framework import StructuredModelSerializer

class MyModelSerializer(StructuredModelSerializer):
    class Meta:
        model = MyModel
        fields = '__all__'

Errors generated by pydantic validation will be automatically translated to DRF errors.

Cache

To prevent the field from making multiple identical queries a caching technique is used. The cache is still a work in progress, please open an issue if you find any problem. Actually the cache covers all the relations inside a StructuredJSONField, optimizing the queries during the serialization process.

Cache engine progress:

  • Shared cache between ForeignKey fields and QuerySet fields
  • Shared cache through nested schemas
  • Shared cache through nested lists of schemas
  • Shared cache between all StructuredJSONFields in the same instance
  • Shared cache between multiple instances of the same model
  • Cache invalidation mechanism

Settings

You can manage structured field behaviour modifying the STRUCTURED_FIELD setting in your settings.py file. Here a list of the available settings and their default values:

STRUCTURED_FIELD = {
    'CACHE':{
        'ENABLED': True,
        'SHARED': False # ⚠️ EXPERIMENTAL: this enables a thread-shared cache, it's not recommended to use it in production.
    },
}

Contributing

The project is open to contributions, just open an issue or a PR.

Running tests

pip install -r requirements-dev.txt
make test

Running test app

pip install -r requirements-dev.txt
python manage.py migrate
python manage.py runserver

License

This project is licensed under the MIT License - see the LICENSE file for details

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

django_structured_json_field-1.0.0.tar.gz (1.9 MB view details)

Uploaded Source

Built Distribution

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

django_structured_json_field-1.0.0-py2.py3-none-any.whl (2.0 MB view details)

Uploaded Python 2Python 3

File details

Details for the file django_structured_json_field-1.0.0.tar.gz.

File metadata

File hashes

Hashes for django_structured_json_field-1.0.0.tar.gz
Algorithm Hash digest
SHA256 585bc6f7c1355ccf6f655c42c310adb889c609f4eb930fc0f53c867a4e2d2175
MD5 231911ef0d214585601b6faba29e6d35
BLAKE2b-256 5f5d63cac0ee82188d3bcc97ca5eb8667bb862c501e911ba3d4f3c2b8fc32196

See more details on using hashes here.

Provenance

The following attestation bundles were made for django_structured_json_field-1.0.0.tar.gz:

Publisher: cd.yml on bnznamco/django-structured-field

Attestations: Values shown here reflect the state when the release was signed and may no longer be current.

File details

Details for the file django_structured_json_field-1.0.0-py2.py3-none-any.whl.

File metadata

File hashes

Hashes for django_structured_json_field-1.0.0-py2.py3-none-any.whl
Algorithm Hash digest
SHA256 18845cde44e442d14820599e6a12aef9e7f7f33021b47d79fe8904fd1efd9717
MD5 32a04989e1b6c29bd2ade9eb42fa4106
BLAKE2b-256 2d9fcffa64ec5a3071e25b7c08755e1645627e87f31e41a1f268614ec913b45e

See more details on using hashes here.

Provenance

The following attestation bundles were made for django_structured_json_field-1.0.0-py2.py3-none-any.whl:

Publisher: cd.yml on bnznamco/django-structured-field

Attestations: Values shown here reflect the state when the release was signed and may no longer be current.

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