将解码的临时图像保存到Django Imagefield

时间:2013-02-27 15:20:58

标签: django django-1.3

我正在尝试将已经作为Base64编码文本传递给我的图像保存到Django Imagefield中。

但它似乎无法正确保存。数据库报告我的所有图像都存储为“”,它应该将它们作为文件名报告,例如:

"template_images/template_folders/myImage.png"

试图保存图像的代码如下:

elif model_field.get_internal_type() == "ImageField" or model_field.get_internal_type() == "FileField":  # Convert files from base64 back to a file.
    if field_elt.text is not None:
        setattr(instance, model_field.name, File(b64decode(field_elt.text)))

3 个答案:

答案 0 :(得分:26)

阅读this answer后,我得到了这个工作:

from base64 import b64decode
from django.core.files.base import ContentFile

image_data = b64decode(b64_text)
my_model_instance.cool_image_field = ContentFile(image_data, 'whatup.png')
my_model_instance.save()

因此,我建议您将代码更改为:

from django.core.files.base import ContentFile

# Your other code...

elif model_field.get_internal_type() == "ImageField" or model_field.get_internal_type() == "FileField":  # Convert files from base64 back to a file.
    if field_elt.text is not None:
        image_data = b64decode(field_elt.text)
        setattr(instance, model_field.name, ContentFile(image_data, 'myImage.png'))

然后,假设您的ImageField定义为upload_to参数设置为template_images/template_folders/,您应该会看到文件保存为YOUR_MEDIA_URL/template_images/template_folders/myImage.png

答案 1 :(得分:1)

我想这是最干净,最短的方式。

以下是如何在基于Django(drf的)API端的post请求中处理Base64编码的图像文件,并将其保存为ImageField。

假设你有一个模型如下:

Class MyImageModel(models.Model):
      image = models.ImageField(upload_to = 'geo_entity_pic')
      data=model.CharField()

所以对应的序列化器如下:

 from drf_extra_fields.fields import Base64ImageField

 Class MyImageModelSerializer(serializers.ModelSerializers):
      image=Base64ImageField()
      class meta:
         model=MyImageModel
         fields= ('data','image')
      def create(self, validated_data):
        image=validated_data.pop('image')
        data=validated_data.pop('data')
       return MyImageModel.objects.create(data=data,image=image)

相应的View可以如下:

elif request.method == 'POST':
    serializer = MyImageModelSerializer(data=request.data)
    if serializer.is_valid():
        serializer.save()
        return Response(serializer.data, status=201)
    return Response(serializer.errors, status=400)

注意在Serializer中我使用了模块django-extra-field中提供的Base64ImageField的实现

要安装此模块,请运行命令

pip install pip install django-extra-fields

导入相同的内容并完成!

发送(通过post方法)您的图像作为JSON对象中的Base64编码字符串以及您拥有的任何其他数据。

答案 2 :(得分:1)

基于这个SO答案的另一个好方法:https://stackoverflow.com/a/28036805/6143656尝试了它并在django 1.10中测试

我为解码的base64文件创建了一个函数。

def decode_base64_file(data):

    def get_file_extension(file_name, decoded_file):
        import imghdr

        extension = imghdr.what(file_name, decoded_file)
        extension = "jpg" if extension == "jpeg" else extension

        return extension

    from django.core.files.base import ContentFile
    import base64
    import six
    import uuid

    # Check if this is a base64 string
    if isinstance(data, six.string_types):
        # Check if the base64 string is in the "data:" format
        if 'data:' in data and ';base64,' in data:
            # Break out the header from the base64 content
            header, data = data.split(';base64,')

        # Try to decode the file. Return validation error if it fails.
        try:
            decoded_file = base64.b64decode(data)
        except TypeError:
            TypeError('invalid_image')

        # Generate file name:
        file_name = str(uuid.uuid4())[:12] # 12 characters are more than enough.
        # Get the file name extension:
        file_extension = get_file_extension(file_name, decoded_file)

        complete_file_name = "%s.%s" % (file_name, file_extension, )

        return ContentFile(decoded_file, name=complete_file_name)

然后你可以调用函数

import decode_base64_file

p = Post(content='My Picture', image=decode_based64_file(your_base64_file))
p.save()