ValidationError с API-интерфейсом REST Django и generic_relations для универсального доступа для записи по внешнему ключу

Я использую Django REST API и модуль generic_relations вместе с ним для создания своего рода варианта комментариев Django, который управляется не шаблонами, а функциями REST API.

Класс Model называется Annotation и имеет вид:

class Annotation(BaseCommentAbstractModel):

    paragraph_id = models.PositiveIntegerField(null=False)

    body = models.TextField()
    '''
    Annotations can be written only by logged in users. It is to prevent hit and run comments by people under anonymity.
    '''
    user = models.ForeignKey(DjangoUser, null=False, verbose_name=_('user'), related_name = "annotations")

Теперь существует другая модель для контента, которая может быть практически любой, так что, скажем, это простое приложение для публикации:

class BlogContent(models.Model):
    created = models.DateTimeField(auto_now_add=True)
    title = models.CharField(max_length=100, blank=True, default='')
    body = models.TextField()
    ...
    annotation = GenericRelation(Annotation, content_type_field='content_type', object_id_field='object_pk')

Сериализаторы для того же выглядят так:

from generic_relations.relations import GenericRelatedField 

class BlogContentSerializer(serializers.ModelSerializer):
    author_id = serializers.Field(source='author_id.username')
    annotation = serializers.RelatedField(many=True)

    class Meta:
        model = BlogContent
        fields = ('title','created','author_id',
              'data','published_flag', 'url_path','slug','annotation')

class AnnotationSerializer(serializers.ModelSerializer):
    user = serializers.Field(source='user.username')
    content_object = GenericRelatedField({
        BlogContent: serializers.HyperlinkedRelatedField(view_name='annotation:blogcontent-detail'),
        }, read_only=False)

    class Meta:
        model = Annotation
        fields = ('paragraph_id', 'body', 'user', 'content_object')


class UserSerializer(serializers.ModelSerializer):
    blogcontent= serializers.PrimaryKeyRelatedField(many=True) 
    annotations = serializers.PrimaryKeyRelatedField(many=True)

    class Meta:
        model = User
        fields = ('username', 'id', 'blogcontent', 'annotations')

Теперь проблема в строке: content_object = GenericRelatedField({ BlogContent: serializers.HyperlinkedRelatedField(view_name='annotation:blogcontent-detail'), }, read_only=False)

Если общие отношения должны использоваться только для чтения, путем установки read_only=True, тогда REST API хорошо просматривается, но как только я переключаю его на False, сервер возвращает ValidationError

ValidationError at /en/annotation/
[u'Invalid model - model not available.']

Просматривая логи и код, кажется, что в rest_framework.renderers.py в get_raw_data_form(self, view, method, request) Не находит объект в звонке obj = getattr(view, 'object', None),

Это в свою очередь при оценке field_to_native функционировать как serializer = self.determine_deserializer_for_data(value) в generic_relationsЗаставляет ничего не передавать в качестве value и, следовательно, код ломается.

Кто-нибудь еще сталкивался с такой ошибкой и / или мог бы помочь мне исправить ее? Пожалуйста помоги!

Трассировка выглядит следующим образом:

/home/craft/pirateenv/lib/python2.7/site-packages/django/core/handlers/base.py in get_response
            response = response.render() ...
▶ Local vars
/home/craft/pirateenv/lib/python2.7/site-packages/django/template/response.py in render
        self.content = self.rendered_content ...
▶ Local vars
/home/craft/pirateenv/lib/python2.7/site-packages/rest_framework/response.py in rendered_content
    ret = renderer.render(self.data, media_type, context) ...
▶ Local vars
/home/craft/pirateenv/lib/python2.7/site-packages/rest_framework/renderers.py in render
    context = self.get_context(data, accepted_media_type, renderer_context) ...
▶ Local vars
/home/craft/pirateenv/lib/python2.7/site-packages/rest_framework/renderers.py in get_context
    raw_data_post_form = self.get_raw_data_form(view, 'POST', request) ...
▶ Local vars
/home/craft/pirateenv/lib/python2.7/site-packages/rest_framework/renderers.py in get_raw_data_form
            content = renderer.render(serializer.data, accepted, context) ...
▶ Local vars
/home/craft/pirateenv/lib/python2.7/site-packages/rest_framework/serializers.py in data
            self._data = self.to_native(obj) ...
▶ Local vars
/home/craft/pirateenv/lib/python2.7/site-packages/rest_framework/serializers.py in to_native
        value = field.field_to_native(obj, field_name) ...
▶ Local vars
/home/craft/pirateenv/lib/python2.7/site-packages/generic_relations/relations.py in field_to_native
    serializer = self.determine_deserializer_for_data(value) ...
▶ Local vars
/home/craft/pirateenv/lib/python2.7/site-packages/generic_relations/relations.py in determine_deserializer_for_data
        raise ValidationError(self.error_messages['no_model_match']) ...
▶ Local vars

1 ответ

Решение

Эта проблема в хранилище также упоминает ту же ошибку, но в другом контексте, предлагаемое решение, тем не менее, работает и для меня, пока автор не выпустит исправление.

Решение состоит в том, чтобы добавить if value: пункт после вызова super метод.

Другие вопросы по тегам