djangorestframework-deepserializer


Namedjangorestframework-deepserializer JSON
Version 0.1.17 PyPI version JSON
download
home_pagehttps://github.com/Horou/djangorestframework-deepserializer
SummaryA package to create deep serializer for django rest framework
upload_time2024-04-14 11:30:27
maintainerNone
docs_urlNone
authorHorou and Enzo_frnt
requires_pythonNone
licenseNone
keywords django django rest framework deep depth serializer viewset nested nested serializer
VCS
bugtrack_url
requirements No requirements were recorded.
Travis-CI No Travis.
coveralls test coverage No coveralls.
            # DeepSerializer

Another Django REST framework addon for those who don't have time.

## Introduction

`djangorestframework-deepserializer` is a Django REST framework package that provides deep serialization of nested JSON. It supports various types of relationships including `one_to_one`, `one_to_many`, `many_to_one`, `many_to_many`, and also in reverse through their `related_name`. All the database calls are already optimized to the maximum using prefetch_related, select_related and some powerful algo without losing any DRF functionality.This package is particularly useful if you really don't want to work.
This projects presume that you already have some or all your django models completed.

## Installation

You can install `djangorestframework-deepserializer` using pip:

```bash
pip install djangorestframework-deepserializer
```

## Usage

### For ultra-fast development.
If you just want to have an API ready for your model.

`models.py`
```Python
from django.db import models

class Image(models.Model):
    id = models.IntegerField(primary_key=True, editable=False)
    url = models.CharField(max_length=200)

class Tag(models.Model):
    name = models.CharField(primary_key=True)
    description = models.TextField(max_length=4000)


class Book(models.Model):
    title = models.CharField(primary_key=True)
    description = models.TextField(max_length=4000)
    tags = models.ManyToManyField(Tag)
    cover = models.ForeignKey(Image, on_delete=models.CASCADE, blank=True, null=True, related_name="books")


class Chapter(models.Model):
    class Meta:
        unique_together = (('book', 'number'),)
    id = models.CharField(primary_key=True, editable=False)
    book = models.ForeignKey(Book, on_delete=models.CASCADE, related_name="chapters")
    number = models.IntegerField(default=0)
    content = models.TextField()

    def save(self, *args, **kwargs):
        self.id = "_".join(
            str(getattr(self, field).pk if isinstance(getattr(self, field), models.Model) else getattr(self, field))
            for field in self._meta.unique_together[0]
        )
        super().save(*args, **kwargs)
```

`urls.py`
```Python
from rest_framework import routers
from deepserializer import DeepViewSet, DeepCreateViewSet
from myapp.models import Book, Chapter, Tag

router = routers.DefaultRouter()
DeepViewSet.init_router(router, [
    Chapter,
    Tag
])
# The DeepCreateViewSet class possesses the deep_create action that allows the creation or update of nested models.
DeepCreateViewSet.init_router(router, [Book])
```
If you need the read only ViewSets version, replace DeepViewSet with ReadOnlyDeepViewSet

The `init_router` function will create all the necessary ViewSets and Serializers for The given models and will register them in the router.

### For ultra-fast development, with a bit of control.
If you need one of the serializer to act in a specific way, for example also return the number of chapter, you can write your own:

`serializers.py`
```Python
from rest_framework import serializers
from deepserializer import DeepSerializer
from myapp.models import Book, Chapter, Tag

class BookSerializer(DeepSerializer):
    class Meta:
        model = Book
        depth = 10
        fields = '__all__'
        # use_case = "" # by default the use_case is an empty string

    chapters_count = serializers.SerializerMethodField()

    def get_chapters_count(self, obj):
        return len(obj.chapters)
```
This serializer, because of the absence of `use_case` inside the Meta class, will be considered the main serializer for this model and will automatically be retrieved when this model need a serializer.

`urls.py`
```Python
from rest_framework import routers
from deepserializer import DeepViewSet
from myapp.models import Book, Chapter, Tag
from myapp.serializers import * # is needed to allow DeepViewSet to load the serializer in its dict of serializers

router = routers.DefaultRouter()
router.register("Book", DeepViewSet.get_view(Book), basename="Book")
```
The Serializer used for the Book model will be BookSerializer, and this at any depth.

### For ultra-fast development, with even more control.
If you need one of the viewsets to act in a specific way, for example using one serializer for list and another for the rest, you can write your own:

`serializers.py`
```Python
.
.
.

class NoInfoBookSerializer(DeepSerializer):
    class Meta:
        model = Book
        depth = 0
        fields = ("title", "description")
        use_case = "NoInfo"
```

`views.py`
```Python
from deepserializer import DeepViewSet

from myapp.models import Book
from myapp.serializers import NoInfoBookSerializer

class BookViewSets(DeepViewSet):
    queryset = Book.objects

    def get_serializer_class(self):
        if hasattr(self, 'action') and self.action == 'list':
            # return DeepSerializer.get_serializer_class(self.queryset.model, use_case="NoInfo")  # will do the same thing, importing it this way will protect against circular import
            return NoInfoBookSerializer
        return NoInfoBookSerializer.get_serializer_class(self.queryset.model) # this will retrieve BookSerializer without having to manually import it
```
This viewsets will use two different serializer depending on the current action.
`get_serializer_class` function will get a defined serializer if it exists or create a new one for the given model and use_case if not.

### For deep serialization of nested Model.
If you have a dict of list of ...., and you want to create it in one request, you can:

The posted request:
```JSON
{
  "title": "My Book",
  "description": "My first try to write something",
  "tags": [
    {
      "name": "action",
      "description": "battles!!!!!"
    },
    {
      "name": "adventure",
      "description": "Story driven"
    }
  ],
  "chapters": [
    {
      "number": 1,
      "content": "Chapter 1 ..."
    },
    {
      "number": 2,
      "content": "Chapter 2 ..."
    },
    {
      "number": 3,
      "content": "Chapter 3 ..."
    }
  ]
}
```

`views.py`
```Python
from rest_framework import status
from rest_framework.decorators import action
from rest_framework.response import Response
from deepserializer import DeepViewSet, DeepCreateViewSet

from myapp.models import Book, Chapter

class DeepCreateBookViewSets(DeepViewSet):
    queryset = Book.objects


    @action(detail=False, methods=['post'])
    def deep_create(self, request):
        serializer = self.get_serializer()
        results = serializer.deep_update_or_create(self.queryset.model, request.data, raise_exception=True)
        return Response(results, status=status.HTTP_201_CREATED)

# or just this:

class DeepCreateBookViewSets(DeepCreateViewSet):
    queryset = Book.objects
```
The `deep_update_or_create` function get a list of data and return either a list of primary_keys or a list of representations
based on the optional parameter verbose.

The optional parameter are:
- `verbose` Define the amount of returned ny the function, by default it is `True`:
    - If `verbose=True` it will return a list of representations, in this case it will return almost the same as the `request.data` but with `id` and `book` inside the `chapters` dicts.
    - If `verbose=False` it will return a list of primary_keys, in this case it will return `["My Book"]`.
- `delete_models`: List of all the model to delete the previously linked instances not present in the `request.data`

If a validation error occurred during the creation process it will return the representations, regardless of `verbose`, with only the problematic fields + a new `ERROR` field, even for the models

The `update` or `create` in the `deep_update_or_create` function are based on the primary key of the dict:
- With primary key:  The function will search for this data in the database:
    - If the key exist: it will update this instance with the given data (the update will be with `partial=True`), but only one time. If other dict are found with this primary key for this model, the updating process will be skipped.
    - If the key does not exist: It will create a new instance of this model with the given data, but only one time. If other dict are found with this primary key for this model, the creation process will be skipped.
- Without primary key: The function will not search for this data in the database and will directly create it.

With this you can do things like this:
```JSON
[
  {
    "title": "My Book",
    "description": "My first try to write something",
    "tags": [
      {
        "name": "action",
        "description": "battles!!!!!"
      },
      "adventure"
    ],
    "chapters": [],
    "cover": {
      "id": -1,
      "url": "http://example.com/000000"
    },
  },
  {
    "title": "My Book 2",
    "description": "My second try to write something",
    "tags": [
      {
        "name": "action",
        "description": "battles!!!!!"
      }
    ],
    "chapters": [],
    "cover": {
      "id": -1,
      "url": "http://example.com/000000"
    },
  }
]
```
In the case that `action` and `-1` do not exist in the database:
The `action` Tag will be created only one time and will be linked to both `My Book` and `My Book 2`,
And the `-1` Image will be created one time with the new primary key `42` (for example) and be linked to both `My Book` and `My Book 2`.

### The types of relationships that are supported include:

- `one_to_one`: One instance of a model is related to one instance of another model.
- `one_to_many`: One instance of a model is related to many instances of another model.
- `many_to_one`: Many instances of a model are related to one instance of another model.
- `many_to_many`: Many instances of a model are related to many instances of another model.

And in reverse with:

- `related_name`: The name to use for the relation from the related object back to this one.

# Contributing

Contributions are welcome! Please read the contributing guidelines before getting started.

# License

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

            

Raw data

            {
    "_id": null,
    "home_page": "https://github.com/Horou/djangorestframework-deepserializer",
    "name": "djangorestframework-deepserializer",
    "maintainer": null,
    "docs_url": null,
    "requires_python": null,
    "maintainer_email": null,
    "keywords": "Django, Django REST Framework, Deep, Depth, serializer, viewset, nested, nested serializer",
    "author": "Horou and Enzo_frnt",
    "author_email": null,
    "download_url": "https://files.pythonhosted.org/packages/d4/59/f6f272d75d2e138da59fce25f30a33462fd561259fac0b29169fe15cc099/djangorestframework_deepserializer-0.1.17.tar.gz",
    "platform": null,
    "description": "# DeepSerializer\n\nAnother Django REST framework addon for those who don't have time.\n\n## Introduction\n\n`djangorestframework-deepserializer` is a Django REST framework package that provides deep serialization of nested JSON. It supports various types of relationships including `one_to_one`, `one_to_many`, `many_to_one`, `many_to_many`, and also in reverse through their `related_name`. All the database calls are already optimized to the maximum using prefetch_related, select_related and some powerful algo without losing any DRF functionality.This package is particularly useful if you really don't want to work.\nThis projects presume that you already have some or all your django models completed.\n\n## Installation\n\nYou can install `djangorestframework-deepserializer` using pip:\n\n```bash\npip install djangorestframework-deepserializer\n```\n\n## Usage\n\n### For ultra-fast development.\nIf you just want to have an API ready for your model.\n\n`models.py`\n```Python\nfrom django.db import models\n\nclass Image(models.Model):\n    id = models.IntegerField(primary_key=True, editable=False)\n    url = models.CharField(max_length=200)\n\nclass Tag(models.Model):\n    name = models.CharField(primary_key=True)\n    description = models.TextField(max_length=4000)\n\n\nclass Book(models.Model):\n    title = models.CharField(primary_key=True)\n    description = models.TextField(max_length=4000)\n    tags = models.ManyToManyField(Tag)\n    cover = models.ForeignKey(Image, on_delete=models.CASCADE, blank=True, null=True, related_name=\"books\")\n\n\nclass Chapter(models.Model):\n    class Meta:\n        unique_together = (('book', 'number'),)\n    id = models.CharField(primary_key=True, editable=False)\n    book = models.ForeignKey(Book, on_delete=models.CASCADE, related_name=\"chapters\")\n    number = models.IntegerField(default=0)\n    content = models.TextField()\n\n    def save(self, *args, **kwargs):\n        self.id = \"_\".join(\n            str(getattr(self, field).pk if isinstance(getattr(self, field), models.Model) else getattr(self, field))\n            for field in self._meta.unique_together[0]\n        )\n        super().save(*args, **kwargs)\n```\n\n`urls.py`\n```Python\nfrom rest_framework import routers\nfrom deepserializer import DeepViewSet, DeepCreateViewSet\nfrom myapp.models import Book, Chapter, Tag\n\nrouter = routers.DefaultRouter()\nDeepViewSet.init_router(router, [\n    Chapter,\n    Tag\n])\n# The DeepCreateViewSet class possesses the deep_create action that allows the creation or update of nested models.\nDeepCreateViewSet.init_router(router, [Book])\n```\nIf you need the read only ViewSets version, replace DeepViewSet with ReadOnlyDeepViewSet\n\nThe `init_router` function will create all the necessary ViewSets and Serializers for The given models and will register them in the router.\n\n### For ultra-fast development, with a bit of control.\nIf you need one of the serializer to act in a specific way, for example also return the number of chapter, you can write your own:\n\n`serializers.py`\n```Python\nfrom rest_framework import serializers\nfrom deepserializer import DeepSerializer\nfrom myapp.models import Book, Chapter, Tag\n\nclass BookSerializer(DeepSerializer):\n    class Meta:\n        model = Book\n        depth = 10\n        fields = '__all__'\n        # use_case = \"\" # by default the use_case is an empty string\n\n    chapters_count = serializers.SerializerMethodField()\n\n    def get_chapters_count(self, obj):\n        return len(obj.chapters)\n```\nThis serializer, because of the absence of `use_case` inside the Meta class, will be considered the main serializer for this model and will automatically be retrieved when this model need a serializer.\n\n`urls.py`\n```Python\nfrom rest_framework import routers\nfrom deepserializer import DeepViewSet\nfrom myapp.models import Book, Chapter, Tag\nfrom myapp.serializers import * # is needed to allow DeepViewSet to load the serializer in its dict of serializers\n\nrouter = routers.DefaultRouter()\nrouter.register(\"Book\", DeepViewSet.get_view(Book), basename=\"Book\")\n```\nThe Serializer used for the Book model will be BookSerializer, and this at any depth.\n\n### For ultra-fast development, with even more control.\nIf you need one of the viewsets to act in a specific way, for example using one serializer for list and another for the rest, you can write your own:\n\n`serializers.py`\n```Python\n.\n.\n.\n\nclass NoInfoBookSerializer(DeepSerializer):\n    class Meta:\n        model = Book\n        depth = 0\n        fields = (\"title\", \"description\")\n        use_case = \"NoInfo\"\n```\n\n`views.py`\n```Python\nfrom deepserializer import DeepViewSet\n\nfrom myapp.models import Book\nfrom myapp.serializers import NoInfoBookSerializer\n\nclass BookViewSets(DeepViewSet):\n    queryset = Book.objects\n\n    def get_serializer_class(self):\n        if hasattr(self, 'action') and self.action == 'list':\n            # return DeepSerializer.get_serializer_class(self.queryset.model, use_case=\"NoInfo\")  # will do the same thing, importing it this way will protect against circular import\n            return NoInfoBookSerializer\n        return NoInfoBookSerializer.get_serializer_class(self.queryset.model) # this will retrieve BookSerializer without having to manually import it\n```\nThis viewsets will use two different serializer depending on the current action.\n`get_serializer_class` function will get a defined serializer if it exists or create a new one for the given model and use_case if not.\n\n### For deep serialization of nested Model.\nIf you have a dict of list of ...., and you want to create it in one request, you can:\n\nThe posted request:\n```JSON\n{\n  \"title\": \"My Book\",\n  \"description\": \"My first try to write something\",\n  \"tags\": [\n    {\n      \"name\": \"action\",\n      \"description\": \"battles!!!!!\"\n    },\n    {\n      \"name\": \"adventure\",\n      \"description\": \"Story driven\"\n    }\n  ],\n  \"chapters\": [\n    {\n      \"number\": 1,\n      \"content\": \"Chapter 1 ...\"\n    },\n    {\n      \"number\": 2,\n      \"content\": \"Chapter 2 ...\"\n    },\n    {\n      \"number\": 3,\n      \"content\": \"Chapter 3 ...\"\n    }\n  ]\n}\n```\n\n`views.py`\n```Python\nfrom rest_framework import status\nfrom rest_framework.decorators import action\nfrom rest_framework.response import Response\nfrom deepserializer import DeepViewSet, DeepCreateViewSet\n\nfrom myapp.models import Book, Chapter\n\nclass DeepCreateBookViewSets(DeepViewSet):\n    queryset = Book.objects\n\n\n    @action(detail=False, methods=['post'])\n    def deep_create(self, request):\n        serializer = self.get_serializer()\n        results = serializer.deep_update_or_create(self.queryset.model, request.data, raise_exception=True)\n        return Response(results, status=status.HTTP_201_CREATED)\n\n# or just this:\n\nclass DeepCreateBookViewSets(DeepCreateViewSet):\n    queryset = Book.objects\n```\nThe `deep_update_or_create` function get a list of data and return either a list of primary_keys or a list of representations\nbased on the optional parameter verbose.\n\nThe optional parameter are:\n- `verbose` Define the amount of returned ny the function, by default it is `True`:\n    - If `verbose=True` it will return a list of representations, in this case it will return almost the same as the `request.data` but with `id` and `book` inside the `chapters` dicts.\n    - If `verbose=False` it will return a list of primary_keys, in this case it will return `[\"My Book\"]`.\n- `delete_models`: List of all the model to delete the previously linked instances not present in the `request.data`\n\nIf a validation error occurred during the creation process it will return the representations, regardless of `verbose`, with only the problematic fields + a new `ERROR` field, even for the models\n\nThe `update` or `create` in the `deep_update_or_create` function are based on the primary key of the dict:\n- With primary key:  The function will search for this data in the database:\n    - If the key exist: it will update this instance with the given data (the update will be with `partial=True`), but only one time. If other dict are found with this primary key for this model, the updating process will be skipped.\n    - If the key does not exist: It will create a new instance of this model with the given data, but only one time. If other dict are found with this primary key for this model, the creation process will be skipped.\n- Without primary key: The function will not search for this data in the database and will directly create it.\n\nWith this you can do things like this:\n```JSON\n[\n  {\n    \"title\": \"My Book\",\n    \"description\": \"My first try to write something\",\n    \"tags\": [\n      {\n        \"name\": \"action\",\n        \"description\": \"battles!!!!!\"\n      },\n      \"adventure\"\n    ],\n    \"chapters\": [],\n    \"cover\": {\n      \"id\": -1,\n      \"url\": \"http://example.com/000000\"\n    },\n  },\n  {\n    \"title\": \"My Book 2\",\n    \"description\": \"My second try to write something\",\n    \"tags\": [\n      {\n        \"name\": \"action\",\n        \"description\": \"battles!!!!!\"\n      }\n    ],\n    \"chapters\": [],\n    \"cover\": {\n      \"id\": -1,\n      \"url\": \"http://example.com/000000\"\n    },\n  }\n]\n```\nIn the case that `action` and `-1` do not exist in the database:\nThe `action` Tag will be created only one time and will be linked to both `My Book` and `My Book 2`,\nAnd the `-1` Image will be created one time with the new primary key `42` (for example) and be linked to both `My Book` and `My Book 2`.\n\n### The types of relationships that are supported include:\n\n- `one_to_one`: One instance of a model is related to one instance of another model.\n- `one_to_many`: One instance of a model is related to many instances of another model.\n- `many_to_one`: Many instances of a model are related to one instance of another model.\n- `many_to_many`: Many instances of a model are related to many instances of another model.\n\nAnd in reverse with:\n\n- `related_name`: The name to use for the relation from the related object back to this one.\n\n# Contributing\n\nContributions are welcome! Please read the contributing guidelines before getting started.\n\n# License\n\nThis project is licensed under the terms of the MIT license.\n",
    "bugtrack_url": null,
    "license": null,
    "summary": "A package to create deep serializer for django rest framework",
    "version": "0.1.17",
    "project_urls": {
        "Homepage": "https://github.com/Horou/djangorestframework-deepserializer"
    },
    "split_keywords": [
        "django",
        " django rest framework",
        " deep",
        " depth",
        " serializer",
        " viewset",
        " nested",
        " nested serializer"
    ],
    "urls": [
        {
            "comment_text": "",
            "digests": {
                "blake2b_256": "4110b39a64e7243c79a5eb48192565150a1f0ddc115acccb6519f5d72858dab4",
                "md5": "1bfee56d9d8521315ea23884bc6dcd52",
                "sha256": "e12a731a117f1cb75b9576076aa116192d6911848e011fe0a8f0959dc4c8a186"
            },
            "downloads": -1,
            "filename": "djangorestframework_deepserializer-0.1.17-py3-none-any.whl",
            "has_sig": false,
            "md5_digest": "1bfee56d9d8521315ea23884bc6dcd52",
            "packagetype": "bdist_wheel",
            "python_version": "py3",
            "requires_python": null,
            "size": 14938,
            "upload_time": "2024-04-14T11:30:25",
            "upload_time_iso_8601": "2024-04-14T11:30:25.805265Z",
            "url": "https://files.pythonhosted.org/packages/41/10/b39a64e7243c79a5eb48192565150a1f0ddc115acccb6519f5d72858dab4/djangorestframework_deepserializer-0.1.17-py3-none-any.whl",
            "yanked": false,
            "yanked_reason": null
        },
        {
            "comment_text": "",
            "digests": {
                "blake2b_256": "d459f6f272d75d2e138da59fce25f30a33462fd561259fac0b29169fe15cc099",
                "md5": "b62d026883accef5c1dd9f9ef7e4b702",
                "sha256": "93390b6fc3680667a906a111c21c41ca0475d43f3db341dff032b2bcfed53b3d"
            },
            "downloads": -1,
            "filename": "djangorestframework_deepserializer-0.1.17.tar.gz",
            "has_sig": false,
            "md5_digest": "b62d026883accef5c1dd9f9ef7e4b702",
            "packagetype": "sdist",
            "python_version": "source",
            "requires_python": null,
            "size": 16696,
            "upload_time": "2024-04-14T11:30:27",
            "upload_time_iso_8601": "2024-04-14T11:30:27.454376Z",
            "url": "https://files.pythonhosted.org/packages/d4/59/f6f272d75d2e138da59fce25f30a33462fd561259fac0b29169fe15cc099/djangorestframework_deepserializer-0.1.17.tar.gz",
            "yanked": false,
            "yanked_reason": null
        }
    ],
    "upload_time": "2024-04-14 11:30:27",
    "github": true,
    "gitlab": false,
    "bitbucket": false,
    "codeberg": false,
    "github_user": "Horou",
    "github_project": "djangorestframework-deepserializer",
    "travis_ci": false,
    "coveralls": false,
    "github_actions": true,
    "lcname": "djangorestframework-deepserializer"
}
        
Elapsed time: 0.21221s