python - 序列化和反序列化 Django 枚举字段以接受数字和文本表示

标签 python django django-rest-framework

我正在尝试创建一个 enum field在 Django 中,在 GET 请求时将返回枚举的文本表示,而在 POST 或 PATCH 请求时将在保存之前将文本表示转换为相应的整数。

transform_<field>()

方法非常适合将整数枚举值转换为其相应的字符串,但除了破解

validate_<field>()

方法。

有更好的方法吗?请看下面的代码

模型文件

class Status(enum.Enum):
    RUNNING = 0
    COMPLETED = 1

    labels = {
         RUNNING: 'Running',
         COMPLETED: 'Completed'
    }

    translation = {v: k for k, v in labels.iteritems()}

class Job(models.Model):
    status = enum.EnumField(Status)

序列化器

class JobSeralizer(serializers.ModelSerailzer):
    status = seralizers.CharField(max_length=32, default=Status.QUEUED)

    def transform_status(self, obj, value):
        return JobStatus.labels[value]

    def validate_status(self, attrs, source):
        """Allow status to take numeric or character representation of status
        """
        status = attrs[source]
        if status in JobStatus.translation:
            attrs[source] = JobStatus.translation[status]
        elif status.isdigit():
            attrs[source] = int(status)
        else:
            raise serializers.ValidationError("'%s' not a valid status" % status)
        return attrs

最佳答案

如 OP 所述,您可以使用 custom fields 轻松完成此操作在 drf v3.x 中。这是用于转换值 <-> 标签(例如枚举值 <-> 文本表示)的通用自定义字段的快速示例:

class KeyValueField(serializers.Field):
    """ A field that takes a field's value as the key and returns
    the associated value for serialization """

    labels = {}
    inverted_labels = {}

    def __init__(self, labels, *args, **kwargs):
        self.labels = labels
        # Check to make sure the labels dict is reversible, otherwise
        # deserialization may produce unpredictable results
        inverted = {}
        for k, v in labels.iteritems():
            if v in inverted:
                raise ValueError(
                    'The field is not deserializable with the given labels.'
                    ' Please ensure that labels map 1:1 with values'
                )
            inverted[v] = k
        self.inverted_labels = inverted
        return super(KeyValueField, self).__init__(*args, **kwargs)

    def to_representation(self, obj):
        if type(obj) is list:
            return [self.labels.get(o, None) for o in obj]
        else:
            return self.labels.get(obj, None)

    def to_internal_value(self, data):
        if type(data) is list:
            return [self.inverted_labels.get(o, None) for o in data]
        else:
            return self.inverted_labels.get(data, None)

字段初始化看起来像这样:

class MySerializer(serializers.Serializer):
    afield = KeyValueField(labels={0:'enum text 0', 1:'enum text 1'})

关于python - 序列化和反序列化 Django 枚举字段以接受数字和文本表示,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/25450323/

相关文章:

python - 如何通过避免不需要的代码来仅从表中获取内容

python - 反向 'update' 关键字参数 '{' id' : '' }' not found. 1 pattern(s) tried: [' update/(? P<id>[0-9]+)/\\Z']

python - 如何替换或编辑 django rest 框架路由器中的查找参数?

python - pyqt向导将字段注册为字符串而不是整数

python - 什么时候应该使用 Unicode 字符串?

python - 尽可能短的路线,从任何地方开始和结束

python - 如何在 Django 中只打印当前登录的用户数据?

python - ViewSet类变量

python - 在Python 2.7中将.po文件转换为json

java - Raspberry Pi MAX31865 Python 到 Java 的转换