我正在尝试将已经作为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)))
答案 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()