python - Django 休息框架 : Serialize data from nested json fields to plain object

标签 python json django serialization django-rest-framework

我想将非平面结构序列化为一个平面对象。 这是我收到的 API 调用的示例(不幸的是我无法控制它):

{
"webhookEvent": "jira:issue_updated",
"user": {
        "id": 2434,
        "name": "Ben",
    },
"issue": {
      "id": "33062",
      "key": "jira-project-key-111",
      "fields": {
          "summary": "The week ahead",
      },
"changelog": {
    "id": "219580",
    "items": [{
         "field": "status",
         "fieldtype": "jira",
         "from": "10127",
         "fromString": "Submitted",
         "to": "10128",
         "toString": "Staged"
    }]
},
"timestamp": 1423234723378
}

我想将它序列化为这样的模型:

class Issue(models.Model):
    jira_id = models.IntegerField()
    jira_id = models.CharField()
    summary = models.CharField()

class Change(models.Model):
    issue = models.ForeignKey(Issue)
    timestamp = models.DataTimeField()

如您所见,模型 Issue 的字段 summaryidkey 位于同一对象上code> 不同于 JSON 数据。

我的序列化器是下一个:

    class ChangeSerializer(serializers.ModelSerializer):
        """Receives complex data from jira and converts into objects."""

        issue = JiraIssueSerializer()
        timestamp = TimestampField(source='created_at')

        class Meta:
            model = Change
            fields = ('issue', 'timestamp')

        def create(self, validated_data):
            super(serializers.ModelSerializer, self).create(validated_data=validated_data)
            jira_issue = JiraIssueSerializer(data=validated_data)
            issue = Issue.objects.get(jira_issue)
            self.created_at = datetime.utcnow()
            change = Change(**validated_data)
            return change



    class JiraIssueSerializer(serializers.ModelSerializer):
        """Issue serializer."""
        id = serializers.IntegerField(source='jira_id')
        key = serializers.CharField(source='jira_key')
        summary = serializers.CharField()   ### I want field to work!
        # fields = serializers.DictField(child=serializers.CharField())

        class Meta:
            model = Issue
            fields = ('id', 'key',
               'summary',
            )

        def to_internal_value(self, data):
           # ret = super(serializers.ModelSerializer,   self).to_internal_value(data)
           ret = {}
           # ret = super().to_internal_value(data)
           ret['jira_id'] = data.get('id', None)
           ret['jira_key'] = data.get('key', None)
           jira_issue_fields_data = data.get('fields')
           if jira_issue_fields_data or 1:
               summary = jira_issue_fields_data.get('summary', None)
               ret.update(summary=summary)
            print('to_internal_value', ret)
            return ret

         def to_representation(self, instance):
            ret = {}
            ret = super().to_representation(instance)
            fields = {}
            fields['summary'] = instance.summary
            ret.update(fields=fields)
            print(ret)
            return ret

我在 JSON 中的 issue 对象中工作得很好。 但是如何向 JiraIssueSerializer 添加一些字段,例如 summary?它们不是issue对象的直接字段,而是位于子结构fields中。 我看到了这些方式:

  • 再制作一个模型 Fields 来保留它们,但这很荒谬。我不需要它,我的 API 严格依赖于外部结构。

  • 制作一些 .to_internal_fields() 转换器。但在这种情况下,我必须手动验证和准备我的 Issue 中的所有字段,并重复多次。此外,如果字段 summary 未在 Serializer 中登记,后者将无法使用它或验证失败。

  • DictField 添加到 Serializer(在代码中注释)并从中获取数据。好吗?如何验证它?同样,我有 n 个干净的代码结构。

接下来我想解析并保存变更日志数据。

如何更好地处理这样的结构?

谢谢!

最佳答案

最后在django-rest-framework的测试中找到了解决方案。
https://github.com/tomchristie/django-rest-framework/blob/master/tests/test_serializer.py#L149

您可以轻松定义嵌套的序列化程序,它们将充当容器并将数据提取到普通对象中。像这样:

    class NestedSerializer1(serializers.Serializer):
        a = serializers.IntegerField()
        b = serializers.IntegerField()

    class NestedSerializer2(serializers.Serializer):
        c = serializers.IntegerField()
        d = serializers.IntegerField()

    class TestSerializer(serializers.Serializer):
        nested1 = NestedSerializer1(source='*')
        nested2 = NestedSerializer2(source='*')

    data = {
        'nested1': {'a': 1, 'b': 2},
        'nested2': {'c': 3, 'd': 4}
     }

     serializer = TestSerializer(data=self.data)
     assert serializer.is_valid()

     assert serializer.validated_data == {
        'a': 1, 
        'b': 2,
        'c': 3, 
        'd': 4
    }

关于python - Django 休息框架 : Serialize data from nested json fields to plain object,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/37324341/

相关文章:

python - 打开文件对象的大小

Django admin 外键下拉字段列表仅 "test object"

javascript - AngularJS 嵌套 ul 列表和 ng-repeat

javascript - 如何将JSON数据加载到Jqgrid中?

python - 创建 super 用户时 Django manage.pysync_db 崩溃

python - 如何保持多台机器与 South 和 Git 同步

Python 子进程允许用户交互

python - 当两点之间存在视线时,如何计算父节点和邻居节点之间的距离?

python - 使用连字符传递关键字参数

java - jackson : Partially update bean with json String