通过REST API处理图像



我试图用django-rest-framework创建一个REST API,它将处理一些事情,一个是从前端发送和接收图像。我找到了一篇试图解释这个想法的文章,但它在views.py中使用了基于类的方法,理想情况下,我希望坚持使用基于函数的方法,因为我已经用这种方式做了一些工作(不包括JWT授权),我希望它继续下去。我不知道如何使我的后端清晰的接收和发送图像,你能试着为我提供一些代码片段(或者更好的是,文章)关于如何做到这一点?提前感谢!

需要提到的一件事是,理想情况下,我希望有一个端点来处理创建一个新对象,该对象将附带一个图像(具体为植物)和一个端点来处理更新(更改)对象的图像。

我models.py:

class Plant(models.Model):
user = models.ForeignKey(User, on_delete=models.CASCADE, null=True)
name = models.CharField(max_length=150)
date_added = models.DateField(auto_now_add=True)
description = models.TextField()
img = models.ImageField(blank=True, null=True, upload_to=upload_path)
plant_species = models.CharField(max_length=150)
last_watered = models.IntegerField(default=0)
how_often = models.IntegerField(default=0)
tracked=models.BooleanField(default=True)

我views.py:

class MyTokenObtainPairSerializer(TokenObtainPairSerializer):

@classmethod
def get_token(cls, user):
token = super().get_token(user)
token['username'] = user.username
return token
class MyTokenObtainPairView(TokenObtainPairView):
serializer_class = MyTokenObtainPairSerializer

@api_view(['GET'])
def getRoutes(request):
routes = [
'/api/token',
'/api/token/refresh',
'/api/plants_data',
'/api/update_plant/<str:pk>'
]
return Response(routes)
@api_view(['GET'])
@permission_classes([IsAuthenticated])
def getPlants(request):
user = request.user
plants = user.plant_set.all()
serializer = PlantSerializer(plants, many=True)
return Response(serializer.data)
@api_view(['POST'])
@permission_classes([IsAuthenticated])
def updatePlantTracking(request, pk):
plant = Plant.objects.get(id=pk)
serializer = PlantSerializer(instance=plant, data=request.data, partial=True)
if serializer.is_valid():
serializer.save()
return Response(serializer.data)
return Response(serializer.errors, status=400)

从前端发送json字符串的图像或任何媒体文件,并创建一个序列化器字段来处理json字符串。

import base64
import uuid
from rest_framework.fields import Field
from rest_framework import serializers
class Base64ContentField(Field):
"""
For image, send base64 string as json
"""
def to_internal_value(self, data):
try:
format, datastr = data.split(';base64,')
ext = format.split('/')[-1]
file = ContentFile(base64.b64decode(datastr), name=str(uuid.uuid4())+'.'+ext)
except:
raise serializers.ValidationError('Error in decoding base64 data')
return file
def to_representation(self, value):
if not value:
return None
return value.url

你的PlantSerializer看起来像

class PlantSerializer(serializers.ModelSerializer):
img = Base64ContentField(required=False)
class Meta:
model = Plant
fields = '__all__'

带img字段的post数据看起来像这样

{
"name": "some name",
"img": "",
# ... other fields
}

您可以从这个链接将图像转换为base64;在前端像reactJS他们有包来转换图像为base64字符串

最新更新