drf-extra-fields


Namedrf-extra-fields JSON
Version 3.7.0 PyPI version JSON
download
home_pagehttps://github.com/Hipo/drf-extra-fields
SummaryAdditional fields for Django Rest Framework.
upload_time2023-08-08 18:16:34
maintainer
docs_urlNone
authorhipo
requires_python>=3.7
licenseApache-2.0
keywords
VCS
bugtrack_url
requirements Django djangorestframework filetype
Travis-CI No Travis.
coveralls test coverage No coveralls.
            DRF-EXTRA-FIELDS
================

Extra Fields for Django Rest Framework

[![Build Status](https://github.com/Hipo/drf-extra-fields/actions/workflows/tests.yml/badge.svg?branch=master)](https://github.com/Hipo/drf-extra-fields/actions)
[![codecov](https://codecov.io/gh/Hipo/drf-extra-fields/branch/master/graph/badge.svg)](https://codecov.io/gh/Hipo/drf-extra-fields)
[![PyPI Version](https://img.shields.io/pypi/v/drf-extra-fields.svg)](https://pypi.org/project/drf-extra-fields)
[![Python Versions](https://img.shields.io/pypi/pyversions/drf-extra-fields.svg)](https://pypi.org/project/drf-extra-fields)

Latest Changes
==============
- **v3.7.0**
  - `psycopg` (psycopg 3) is now supported and it's used automatically instead of `psycopg2` if available.
- **v3.6.0**
  - File objects without an actual file-system path can now be used in `Base64ImageField`, `Base64FileField` and `HybridImageField`
- **v3.5.0**
  - Development environment fixes & improvements.
  - Since `Python 3.6` support is ended, the codebase is refactored/modernized for `Python 3.7`.
  - `WebP` is added to default `ALLOWED_TYPES` of the `Base64ImageField`.
  - Deprecated `imghdr` library is replaced with `filetype`.
  - Unintended `Pillow` dependency is removed.
- **v3.4.0**
  - :warning: **BACKWARD INCOMPATIBLE** :warning:
    - Support for `Django 3.0` and `Django 3.1` is ended.
  - `Django 4.0` is now supported.
- **v3.3.0**
  - :warning: **BACKWARD INCOMPATIBLE** :warning:
    - Support for `Python 3.6` is ended.
- **v3.2.1**
  - A typo in the `python_requires` argument of `setup.py` that prevents installation for `Python 3.6` is fixed.
- **v3.2.0**
  - :warning: **BACKWARD INCOMPATIBLE** :warning:
    - Support for `Python 3.5` is ended.
  - `Python 3.9` and `Python 3.10` are now supported.
  - `Django 3.2` is now supported.
- **v3.1.1**
  - `psycopg2` dependency is made optional.
- **v3.1.0**
  - **Possible Breaking Change**:
    -  In this version we have changed file class used in `Base64FileField` from `ContentFile` to `SimpleUploadedFile` (you may see the change [here](https://github.com/Hipo/drf-extra-fields/pull/149/files#diff-5f77bcb61083cd9c026f6dfb3b77bf8fa824c45e620cdb7826ad713bde7b65f8L72-R85)).
  - `child_attrs` property is added to [RangeFields](https://github.com/Hipo/drf-extra-fields#rangefield).

Usage
================

Install the package

```bash
pip install drf-extra-fields
```

**Note:**
- **This package renamed as "drf-extra-fields", earlier it was named as django-extra-fields.**
- Install version 0.1 for Django Rest Framework 2.*
- Install version 0.3 or greater for Django Rest Framework 3.*

Fields:
----------------


## Base64ImageField

An image representation for Base64ImageField

Inherited from `ImageField`


**Signature:** `Base64ImageField()`

 - It takes a base64 image as a string.
 - A base64 image:  `data:image/gif;base64,R0lGODlhAQABAIAAAAAAAP///yH5BAEAAAAALAAAAAABAAEAAAIBRAA7`
 - Base64ImageField accepts the entire string or just the part after base64, `R0lGODlhAQABAIAAAAAAAP///yH5BAEAAAAALAAAAAABAAEAAAIBRAA7`
 - It takes the optional parameter `represent_in_base64` (`False` by default), if set to `True` it will allow for base64-encoded downloads of an `ImageField`.
 - You can inherit the `Base64ImageField` class and set allowed extensions (`ALLOWED_TYPES` list), or customize the validation messages (`INVALID_FILE_MESSAGE`, `INVALID_TYPE_MESSAGE`)


**Example:**

```python
# serializer

from drf_extra_fields.fields import Base64ImageField

class UploadedBase64ImageSerializer(serializers.Serializer):
    file = Base64ImageField(required=False)
    created = serializers.DateTimeField()

# use the serializer
file = 'R0lGODlhAQABAIAAAP///////yH5BAEKAAEALAAAAAABAAEAAAICTAEAOw=='
serializer = UploadedBase64ImageSerializer(data={'created': now, 'file': file})
```


## Base64FileField

A file representation for Base64FileField

Inherited from `FileField`


**Signature:** `Base64FileField()`

 - It takes a base64 file as a string.
 - Other options like for `Base64ImageField`
 - You have to provide your own full implementation of this class. You have to implement file validation in `get_file_extension` method and set `ALLOWED_TYPES` list.


**Example:**

```python
class PDFBase64File(Base64FileField):
    ALLOWED_TYPES = ['pdf']

    def get_file_extension(self, filename, decoded_file):
        try:
            PyPDF2.PdfFileReader(io.BytesIO(decoded_file))
        except PyPDF2.utils.PdfReadError as e:
            logger.warning(e)
        else:
            return 'pdf'
```


## PointField

Point field for GeoDjango


**Signature:** `PointField()`

 - It takes a dictionary contains latitude and longitude keys like below

    {
     "latitude": 49.8782482189424,
     "longitude": 24.452545489
    }
 - It takes the optional parameter `str_points` (False by default), if set to True it serializes the longitude/latitude
 values as strings
 - It takes the optional parameter `srid` (None by default), if set the Point created object will have its srid attribute set to the same value.

**Example:**

```python
# serializer

from drf_extra_fields.geo_fields import PointField

class PointFieldSerializer(serializers.Serializer):
    point = PointField(required=False)
    created = serializers.DateTimeField()

# use the serializer
point = {
    "latitude": 49.8782482189424,
    "longitude": 24.452545489
    }
serializer = PointFieldSerializer(data={'created': now, 'point': point})
```


# RangeField

The Range Fields map to Django's PostgreSQL specific [Range Fields](https://docs.djangoproject.com/en/stable/ref/contrib/postgres/fields/#range-fields).

Each accepts an optional parameter `child_attrs`, which allows passing parameters to the child field.

For example, calling `IntegerRangeField(child_attrs={"allow_null": True})` allows deserializing data with a null value for `lower` and/or `upper`:

```python
from rest_framework import serializers
from drf_extra_fields.fields import IntegerRangeField


class RangeSerializer(serializers.Serializer):
    ranges = IntegerRangeField(child_attrs={"allow_null": True})


serializer = RangeSerializer(data={'ranges': {'lower': 0, 'upper': None}})

```

## IntegerRangeField

```python
from rest_framework import serializers
from drf_extra_fields.fields import IntegerRangeField


class RangeSerializer(serializers.Serializer):
    ranges = IntegerRangeField()


serializer = RangeSerializer(data={'ranges': {'lower': 0, 'upper': 1}})

```

## FloatRangeField

```python
from rest_framework import serializers
from drf_extra_fields.fields import FloatRangeField


class RangeSerializer(serializers.Serializer):
    ranges = FloatRangeField()


serializer = RangeSerializer(data={'ranges': {'lower': 0., 'upper': 1.}})

```

## DecimalRangeField

```python
from rest_framework import serializers
from drf_extra_fields.fields import DecimalRangeField


class RangeSerializer(serializers.Serializer):
    ranges = DecimalRangeField()


serializer = RangeSerializer(data={'ranges': {'lower': 0., 'upper': 1.}}, )

```

## DateRangeField

```python
import datetime

from rest_framework import serializers
from drf_extra_fields.fields import DateRangeField


class RangeSerializer(serializers.Serializer):
    ranges = DateRangeField()


serializer = RangeSerializer(data={'ranges': {'lower': datetime.date(2015, 1, 1), 'upper': datetime.date(2015, 2, 1)}})

```

## DateTimeRangeField

```python
import datetime

from rest_framework import serializers
from drf_extra_fields.fields import DateTimeRangeField


class RangeSerializer(serializers.Serializer):
    ranges = DateTimeRangeField()


serializer = RangeSerializer(data={'ranges': {'lower': datetime.datetime(2015, 1, 1, 0), 'upper': datetime.datetime(2015, 2, 1, 0)}})

```

## PresentablePrimaryKeyRelatedField

Represents related object with a serializer.

`presentation_serializer` could also be a string that represents a dotted path of a serializer, this is useful when you want to represent a related field with the same serializer.

```python
from drf_extra_fields.relations import PresentablePrimaryKeyRelatedField

class UserSerializer(serializers.ModelSerializer):
    class Meta:
        model = User
        fields = (
            'id',
            "username",
        )

class PostSerializer(serializers.ModelSerializer):
    user = PresentablePrimaryKeyRelatedField(
        queryset=User.objects.all(),
        presentation_serializer=UserSerializer,
        presentation_serializer_kwargs={
            'example': [
                'of',
                'passing',
                'kwargs',
                'to',
                'serializer',
            ]
        },
        read_source=None
    )
    class Meta:
        model = Post
        fields = (
            "id",
            "title",
            "user",
        )
```

**Serializer data:**
```
{
    "user": 1,
    "title": "test"
}
```

**Serialized data with PrimaryKeyRelatedField:**
```
{
    "id":1,
    "user": 1,
    "title": "test"
}
```

**Serialized data with PresentablePrimaryKeyRelatedField:**
```
{
    "id":1,
    "user": {
        "id": 1,
        "username": "test"
    },
    "title": "test"
}
```


## PresentableSlugRelatedField

Represents related object retrieved using slug with a serializer.

```python
from drf_extra_fields.relations import PresentableSlugRelatedField

class CategorySerializer(serializers.ModelSerializer):
    class Meta:
        model = Category
        fields = (
            "id",
            "slug",
            "name"
        )

class ProductSerializer(serializers.ModelSerializer):
    category = PresentableSlugRelatedField(
        slug_field="slug",
        queryset=Category.objects.all(),
        presentation_serializer=CategorySerializer,
        presentation_serializer_kwargs={
            'example': [
                'of',
                'passing',
                'kwargs',
                'to',
                'serializer',
            ]
        },
        read_source=None
    )
    class Meta:
        model = Product
        fields = (
            "id",
            "name",
            "category",
        )
```

**Serializer data:**
```
{
    "category": "vegetables",
    "name": "Tomato"
}
```

**Serialized data with SlugRelatedField:**
```
{
    "id": 1,
    "name": "Tomato",
    "category": "vegetables"
}
```

**Serialized data with PresentableSlugRelatedField:**
```
{
    "id": 1,
    "name": "Tomato",
    "category": {
        "id": 1,
        "slug": "vegetables",
        "name": "Vegetables"
    }
}
```

### read_source parameter
This parameter allows you to use different `source` for read operations and doesn't change field name for write operations. This is only used while representing the data. 

## HybridImageField
A django-rest-framework field for handling image-uploads through raw post data, with a fallback to multipart form data.

It first tries Base64ImageField. if it fails then tries ImageField.

```python
from rest_framework import serializers
from drf_extra_fields.fields import HybridImageField


class HybridImageSerializer(serializers.Serializer):
    image = HybridImageField()
```

drf-yasg fix for BASE64 Fields:
----------------
The [drf-yasg](https://github.com/axnsan12/drf-yasg) project seems to generate wrong documentation on Base64ImageField or Base64FileField. It marks those fields as readonly. Here is the workaround code for correct the generated document. (More detail on issue [#66](https://github.com/Hipo/drf-extra-fields/issues/66))

```python
class PDFBase64FileField(Base64FileField):
    ALLOWED_TYPES = ['pdf']

    class Meta:
        swagger_schema_fields = {
            'type': 'string',
            'title': 'File Content',
            'description': 'Content of the file base64 encoded',
            'read_only': False  # <-- FIX
        }

    def get_file_extension(self, filename, decoded_file):
        try:
            PyPDF2.PdfFileReader(io.BytesIO(decoded_file))
        except PyPDF2.utils.PdfReadError as e:
            logger.warning(e)
        else:
            return 'pdf'
```


## LowercaseEmailField
An enhancement over django-rest-framework's EmailField to allow case-insensitive serialization and deserialization of e-mail addresses.

```python
from rest_framework import serializers
from drf_extra_fields.fields import LowercaseEmailField


class EmailSerializer(serializers.Serializer):
    email = LowercaseEmailField()

```

CONTRIBUTION
=================

**TESTS**
- Make sure that you add the test for contributed field to test/test_fields.py
and run with command before sending a pull request:

```bash
$ pip install tox  # if not already installed
$ tox
```

Or, if you prefer using Docker (recommended):

```bash
tools/run_development.sh
tox
```

**README**
- Make sure that you add the documentation for the field added to README.md


LICENSE
====================

Copyright DRF EXTRA FIELDS HIPO

Licensed under the Apache License, Version 2.0 (the "License");
you may not use this file except in compliance with the License.
You may obtain a copy of the License at

    http://www.apache.org/licenses/LICENSE-2.0

Unless required by applicable law or agreed to in writing, software
distributed under the License is distributed on an "AS IS" BASIS,
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
See the License for the specific language governing permissions and
limitations under the License.

            

Raw data

            {
    "_id": null,
    "home_page": "https://github.com/Hipo/drf-extra-fields",
    "name": "drf-extra-fields",
    "maintainer": "",
    "docs_url": null,
    "requires_python": ">=3.7",
    "maintainer_email": "",
    "keywords": "",
    "author": "hipo",
    "author_email": "pypi@hipolabs.com",
    "download_url": "https://files.pythonhosted.org/packages/a4/1d/311a0f359546c2de7b5251b2b802a02e6513a89be3e4280c98ae2edfbdd9/drf-extra-fields-3.7.0.tar.gz",
    "platform": null,
    "description": "DRF-EXTRA-FIELDS\n================\n\nExtra Fields for Django Rest Framework\n\n[![Build Status](https://github.com/Hipo/drf-extra-fields/actions/workflows/tests.yml/badge.svg?branch=master)](https://github.com/Hipo/drf-extra-fields/actions)\n[![codecov](https://codecov.io/gh/Hipo/drf-extra-fields/branch/master/graph/badge.svg)](https://codecov.io/gh/Hipo/drf-extra-fields)\n[![PyPI Version](https://img.shields.io/pypi/v/drf-extra-fields.svg)](https://pypi.org/project/drf-extra-fields)\n[![Python Versions](https://img.shields.io/pypi/pyversions/drf-extra-fields.svg)](https://pypi.org/project/drf-extra-fields)\n\nLatest Changes\n==============\n- **v3.7.0**\n  - `psycopg` (psycopg 3) is now supported and it's used automatically instead of `psycopg2` if available.\n- **v3.6.0**\n  - File objects without an actual file-system path can now be used in `Base64ImageField`, `Base64FileField` and `HybridImageField`\n- **v3.5.0**\n  - Development environment fixes & improvements.\n  - Since `Python 3.6` support is ended, the codebase is refactored/modernized for `Python 3.7`.\n  - `WebP` is added to default `ALLOWED_TYPES` of the `Base64ImageField`.\n  - Deprecated `imghdr` library is replaced with `filetype`.\n  - Unintended `Pillow` dependency is removed.\n- **v3.4.0**\n  - :warning: **BACKWARD INCOMPATIBLE** :warning:\n    - Support for `Django 3.0` and `Django 3.1` is ended.\n  - `Django 4.0` is now supported.\n- **v3.3.0**\n  - :warning: **BACKWARD INCOMPATIBLE** :warning:\n    - Support for `Python 3.6` is ended.\n- **v3.2.1**\n  - A typo in the `python_requires` argument of `setup.py` that prevents installation for `Python 3.6` is fixed.\n- **v3.2.0**\n  - :warning: **BACKWARD INCOMPATIBLE** :warning:\n    - Support for `Python 3.5` is ended.\n  - `Python 3.9` and `Python 3.10` are now supported.\n  - `Django 3.2` is now supported.\n- **v3.1.1**\n  - `psycopg2` dependency is made optional.\n- **v3.1.0**\n  - **Possible Breaking Change**:\n    -  In this version we have changed file class used in `Base64FileField` from `ContentFile` to `SimpleUploadedFile` (you may see the change [here](https://github.com/Hipo/drf-extra-fields/pull/149/files#diff-5f77bcb61083cd9c026f6dfb3b77bf8fa824c45e620cdb7826ad713bde7b65f8L72-R85)).\n  - `child_attrs` property is added to [RangeFields](https://github.com/Hipo/drf-extra-fields#rangefield).\n\nUsage\n================\n\nInstall the package\n\n```bash\npip install drf-extra-fields\n```\n\n**Note:**\n- **This package renamed as \"drf-extra-fields\", earlier it was named as django-extra-fields.**\n- Install version 0.1 for Django Rest Framework 2.*\n- Install version 0.3 or greater for Django Rest Framework 3.*\n\nFields:\n----------------\n\n\n## Base64ImageField\n\nAn image representation for Base64ImageField\n\nInherited from `ImageField`\n\n\n**Signature:** `Base64ImageField()`\n\n - It takes a base64 image as a string.\n - A base64 image:  `data:image/gif;base64,R0lGODlhAQABAIAAAAAAAP///yH5BAEAAAAALAAAAAABAAEAAAIBRAA7`\n - Base64ImageField accepts the entire string or just the part after base64, `R0lGODlhAQABAIAAAAAAAP///yH5BAEAAAAALAAAAAABAAEAAAIBRAA7`\n - It takes the optional parameter `represent_in_base64` (`False` by default), if set to `True` it will allow for base64-encoded downloads of an `ImageField`.\n - You can inherit the `Base64ImageField` class and set allowed extensions (`ALLOWED_TYPES` list), or customize the validation messages (`INVALID_FILE_MESSAGE`, `INVALID_TYPE_MESSAGE`)\n\n\n**Example:**\n\n```python\n# serializer\n\nfrom drf_extra_fields.fields import Base64ImageField\n\nclass UploadedBase64ImageSerializer(serializers.Serializer):\n    file = Base64ImageField(required=False)\n    created = serializers.DateTimeField()\n\n# use the serializer\nfile = 'R0lGODlhAQABAIAAAP///////yH5BAEKAAEALAAAAAABAAEAAAICTAEAOw=='\nserializer = UploadedBase64ImageSerializer(data={'created': now, 'file': file})\n```\n\n\n## Base64FileField\n\nA file representation for Base64FileField\n\nInherited from `FileField`\n\n\n**Signature:** `Base64FileField()`\n\n - It takes a base64 file as a string.\n - Other options like for `Base64ImageField`\n - You have to provide your own full implementation of this class. You have to implement file validation in `get_file_extension` method and set `ALLOWED_TYPES` list.\n\n\n**Example:**\n\n```python\nclass PDFBase64File(Base64FileField):\n    ALLOWED_TYPES = ['pdf']\n\n    def get_file_extension(self, filename, decoded_file):\n        try:\n            PyPDF2.PdfFileReader(io.BytesIO(decoded_file))\n        except PyPDF2.utils.PdfReadError as e:\n            logger.warning(e)\n        else:\n            return 'pdf'\n```\n\n\n## PointField\n\nPoint field for GeoDjango\n\n\n**Signature:** `PointField()`\n\n - It takes a dictionary contains latitude and longitude keys like below\n\n    {\n     \"latitude\": 49.8782482189424,\n     \"longitude\": 24.452545489\n    }\n - It takes the optional parameter `str_points` (False by default), if set to True it serializes the longitude/latitude\n values as strings\n - It takes the optional parameter `srid` (None by default), if set the Point created object will have its srid attribute set to the same value.\n\n**Example:**\n\n```python\n# serializer\n\nfrom drf_extra_fields.geo_fields import PointField\n\nclass PointFieldSerializer(serializers.Serializer):\n    point = PointField(required=False)\n    created = serializers.DateTimeField()\n\n# use the serializer\npoint = {\n    \"latitude\": 49.8782482189424,\n    \"longitude\": 24.452545489\n    }\nserializer = PointFieldSerializer(data={'created': now, 'point': point})\n```\n\n\n# RangeField\n\nThe Range Fields map to Django's PostgreSQL specific [Range Fields](https://docs.djangoproject.com/en/stable/ref/contrib/postgres/fields/#range-fields).\n\nEach accepts an optional parameter `child_attrs`, which allows passing parameters to the child field.\n\nFor example, calling `IntegerRangeField(child_attrs={\"allow_null\": True})` allows deserializing data with a null value for `lower` and/or `upper`:\n\n```python\nfrom rest_framework import serializers\nfrom drf_extra_fields.fields import IntegerRangeField\n\n\nclass RangeSerializer(serializers.Serializer):\n    ranges = IntegerRangeField(child_attrs={\"allow_null\": True})\n\n\nserializer = RangeSerializer(data={'ranges': {'lower': 0, 'upper': None}})\n\n```\n\n## IntegerRangeField\n\n```python\nfrom rest_framework import serializers\nfrom drf_extra_fields.fields import IntegerRangeField\n\n\nclass RangeSerializer(serializers.Serializer):\n    ranges = IntegerRangeField()\n\n\nserializer = RangeSerializer(data={'ranges': {'lower': 0, 'upper': 1}})\n\n```\n\n## FloatRangeField\n\n```python\nfrom rest_framework import serializers\nfrom drf_extra_fields.fields import FloatRangeField\n\n\nclass RangeSerializer(serializers.Serializer):\n    ranges = FloatRangeField()\n\n\nserializer = RangeSerializer(data={'ranges': {'lower': 0., 'upper': 1.}})\n\n```\n\n## DecimalRangeField\n\n```python\nfrom rest_framework import serializers\nfrom drf_extra_fields.fields import DecimalRangeField\n\n\nclass RangeSerializer(serializers.Serializer):\n    ranges = DecimalRangeField()\n\n\nserializer = RangeSerializer(data={'ranges': {'lower': 0., 'upper': 1.}}, )\n\n```\n\n## DateRangeField\n\n```python\nimport datetime\n\nfrom rest_framework import serializers\nfrom drf_extra_fields.fields import DateRangeField\n\n\nclass RangeSerializer(serializers.Serializer):\n    ranges = DateRangeField()\n\n\nserializer = RangeSerializer(data={'ranges': {'lower': datetime.date(2015, 1, 1), 'upper': datetime.date(2015, 2, 1)}})\n\n```\n\n## DateTimeRangeField\n\n```python\nimport datetime\n\nfrom rest_framework import serializers\nfrom drf_extra_fields.fields import DateTimeRangeField\n\n\nclass RangeSerializer(serializers.Serializer):\n    ranges = DateTimeRangeField()\n\n\nserializer = RangeSerializer(data={'ranges': {'lower': datetime.datetime(2015, 1, 1, 0), 'upper': datetime.datetime(2015, 2, 1, 0)}})\n\n```\n\n## PresentablePrimaryKeyRelatedField\n\nRepresents related object with a serializer.\n\n`presentation_serializer` could also be a string that represents a dotted path of a serializer, this is useful when you want to represent a related field with the same serializer.\n\n```python\nfrom drf_extra_fields.relations import PresentablePrimaryKeyRelatedField\n\nclass UserSerializer(serializers.ModelSerializer):\n    class Meta:\n        model = User\n        fields = (\n            'id',\n            \"username\",\n        )\n\nclass PostSerializer(serializers.ModelSerializer):\n    user = PresentablePrimaryKeyRelatedField(\n        queryset=User.objects.all(),\n        presentation_serializer=UserSerializer,\n        presentation_serializer_kwargs={\n            'example': [\n                'of',\n                'passing',\n                'kwargs',\n                'to',\n                'serializer',\n            ]\n        },\n        read_source=None\n    )\n    class Meta:\n        model = Post\n        fields = (\n            \"id\",\n            \"title\",\n            \"user\",\n        )\n```\n\n**Serializer data:**\n```\n{\n    \"user\": 1,\n    \"title\": \"test\"\n}\n```\n\n**Serialized data with PrimaryKeyRelatedField:**\n```\n{\n    \"id\":1,\n    \"user\": 1,\n    \"title\": \"test\"\n}\n```\n\n**Serialized data with PresentablePrimaryKeyRelatedField:**\n```\n{\n    \"id\":1,\n    \"user\": {\n        \"id\": 1,\n        \"username\": \"test\"\n    },\n    \"title\": \"test\"\n}\n```\n\n\n## PresentableSlugRelatedField\n\nRepresents related object retrieved using slug with a serializer.\n\n```python\nfrom drf_extra_fields.relations import PresentableSlugRelatedField\n\nclass CategorySerializer(serializers.ModelSerializer):\n    class Meta:\n        model = Category\n        fields = (\n            \"id\",\n            \"slug\",\n            \"name\"\n        )\n\nclass ProductSerializer(serializers.ModelSerializer):\n    category = PresentableSlugRelatedField(\n        slug_field=\"slug\",\n        queryset=Category.objects.all(),\n        presentation_serializer=CategorySerializer,\n        presentation_serializer_kwargs={\n            'example': [\n                'of',\n                'passing',\n                'kwargs',\n                'to',\n                'serializer',\n            ]\n        },\n        read_source=None\n    )\n    class Meta:\n        model = Product\n        fields = (\n            \"id\",\n            \"name\",\n            \"category\",\n        )\n```\n\n**Serializer data:**\n```\n{\n    \"category\": \"vegetables\",\n    \"name\": \"Tomato\"\n}\n```\n\n**Serialized data with SlugRelatedField:**\n```\n{\n    \"id\": 1,\n    \"name\": \"Tomato\",\n    \"category\": \"vegetables\"\n}\n```\n\n**Serialized data with PresentableSlugRelatedField:**\n```\n{\n    \"id\": 1,\n    \"name\": \"Tomato\",\n    \"category\": {\n        \"id\": 1,\n        \"slug\": \"vegetables\",\n        \"name\": \"Vegetables\"\n    }\n}\n```\n\n### read_source parameter\nThis parameter allows you to use different `source` for read operations and doesn't change field name for write operations. This is only used while representing the data. \n\n## HybridImageField\nA django-rest-framework field for handling image-uploads through raw post data, with a fallback to multipart form data.\n\nIt first tries Base64ImageField. if it fails then tries ImageField.\n\n```python\nfrom rest_framework import serializers\nfrom drf_extra_fields.fields import HybridImageField\n\n\nclass HybridImageSerializer(serializers.Serializer):\n    image = HybridImageField()\n```\n\ndrf-yasg fix for BASE64 Fields:\n----------------\nThe [drf-yasg](https://github.com/axnsan12/drf-yasg) project seems to generate wrong documentation on Base64ImageField or Base64FileField. It marks those fields as readonly. Here is the workaround code for correct the generated document. (More detail on issue [#66](https://github.com/Hipo/drf-extra-fields/issues/66))\n\n```python\nclass PDFBase64FileField(Base64FileField):\n    ALLOWED_TYPES = ['pdf']\n\n    class Meta:\n        swagger_schema_fields = {\n            'type': 'string',\n            'title': 'File Content',\n            'description': 'Content of the file base64 encoded',\n            'read_only': False  # <-- FIX\n        }\n\n    def get_file_extension(self, filename, decoded_file):\n        try:\n            PyPDF2.PdfFileReader(io.BytesIO(decoded_file))\n        except PyPDF2.utils.PdfReadError as e:\n            logger.warning(e)\n        else:\n            return 'pdf'\n```\n\n\n## LowercaseEmailField\nAn enhancement over django-rest-framework's EmailField to allow case-insensitive serialization and deserialization of e-mail addresses.\n\n```python\nfrom rest_framework import serializers\nfrom drf_extra_fields.fields import LowercaseEmailField\n\n\nclass EmailSerializer(serializers.Serializer):\n    email = LowercaseEmailField()\n\n```\n\nCONTRIBUTION\n=================\n\n**TESTS**\n- Make sure that you add the test for contributed field to test/test_fields.py\nand run with command before sending a pull request:\n\n```bash\n$ pip install tox  # if not already installed\n$ tox\n```\n\nOr, if you prefer using Docker (recommended):\n\n```bash\ntools/run_development.sh\ntox\n```\n\n**README**\n- Make sure that you add the documentation for the field added to README.md\n\n\nLICENSE\n====================\n\nCopyright DRF EXTRA FIELDS HIPO\n\nLicensed under the Apache License, Version 2.0 (the \"License\");\nyou may not use this file except in compliance with the License.\nYou may obtain a copy of the License at\n\n    http://www.apache.org/licenses/LICENSE-2.0\n\nUnless required by applicable law or agreed to in writing, software\ndistributed under the License is distributed on an \"AS IS\" BASIS,\nWITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\nSee the License for the specific language governing permissions and\nlimitations under the License.\n",
    "bugtrack_url": null,
    "license": "Apache-2.0",
    "summary": "Additional fields for Django Rest Framework.",
    "version": "3.7.0",
    "project_urls": {
        "Homepage": "https://github.com/Hipo/drf-extra-fields"
    },
    "split_keywords": [],
    "urls": [
        {
            "comment_text": "",
            "digests": {
                "blake2b_256": "c19d30ec8fff60178a954b9fb3dff06aef0fb26ffeab8a102f8ce37b024e2a61",
                "md5": "d9c5f8489dd2d762928568c0592e1b79",
                "sha256": "3f3b1a6ec0eea02c9cabd62fe10e1673e1e42c5daa27fa9d4cea758c52c5cc30"
            },
            "downloads": -1,
            "filename": "drf_extra_fields-3.7.0-py3-none-any.whl",
            "has_sig": false,
            "md5_digest": "d9c5f8489dd2d762928568c0592e1b79",
            "packagetype": "bdist_wheel",
            "python_version": "py3",
            "requires_python": ">=3.7",
            "size": 17764,
            "upload_time": "2023-08-08T18:16:32",
            "upload_time_iso_8601": "2023-08-08T18:16:32.868262Z",
            "url": "https://files.pythonhosted.org/packages/c1/9d/30ec8fff60178a954b9fb3dff06aef0fb26ffeab8a102f8ce37b024e2a61/drf_extra_fields-3.7.0-py3-none-any.whl",
            "yanked": false,
            "yanked_reason": null
        },
        {
            "comment_text": "",
            "digests": {
                "blake2b_256": "a41d311a0f359546c2de7b5251b2b802a02e6513a89be3e4280c98ae2edfbdd9",
                "md5": "789e3130dc2aaefad3885b514e54b1db",
                "sha256": "d7e58b8f60432233328b8a64831e50ea5d05932d3cb63ade8fe66c0d3d21ac5b"
            },
            "downloads": -1,
            "filename": "drf-extra-fields-3.7.0.tar.gz",
            "has_sig": false,
            "md5_digest": "789e3130dc2aaefad3885b514e54b1db",
            "packagetype": "sdist",
            "python_version": "source",
            "requires_python": ">=3.7",
            "size": 25355,
            "upload_time": "2023-08-08T18:16:34",
            "upload_time_iso_8601": "2023-08-08T18:16:34.947791Z",
            "url": "https://files.pythonhosted.org/packages/a4/1d/311a0f359546c2de7b5251b2b802a02e6513a89be3e4280c98ae2edfbdd9/drf-extra-fields-3.7.0.tar.gz",
            "yanked": false,
            "yanked_reason": null
        }
    ],
    "upload_time": "2023-08-08 18:16:34",
    "github": true,
    "gitlab": false,
    "bitbucket": false,
    "codeberg": false,
    "github_user": "Hipo",
    "github_project": "drf-extra-fields",
    "travis_ci": false,
    "coveralls": false,
    "github_actions": true,
    "requirements": [
        {
            "name": "Django",
            "specs": [
                [
                    ">=",
                    "2.2"
                ]
            ]
        },
        {
            "name": "djangorestframework",
            "specs": [
                [
                    ">=",
                    "3.9.2"
                ]
            ]
        },
        {
            "name": "filetype",
            "specs": [
                [
                    ">=",
                    "1.2.0"
                ]
            ]
        }
    ],
    "tox": true,
    "lcname": "drf-extra-fields"
}
        
Elapsed time: 0.38607s