如何使用Django Rest API上传文件

16

有人能帮我学习如何在Django Rest API中使用POST方法上传文件吗? 例如,当我启动时...

curl -X POST 127.0.0.1:8000/api/v1/assets/ -d '{"name" = "my image   ","source"="/root/images/my_image2.jpg"}' -H "Content-Type: application/json"

我想要上传我的图片 my_image2.jpg

serializers.py:

from django.forms import widgets
from rest_framework import serializers
from .models import Asset

class AssetSerializer(serializers.ModelSerializer):
    class Meta:
        model = Asset

视图.py:

from .serializers import AssetSerializer
from django.http import Http404
from rest_framework.views import APIView
from rest_framework.response import Response
from rest_framework import status

from rest_framework import status
from rest_framework.decorators import api_view


class AssetAdd(APIView):


    def post(self, request, format=None):
        serializer = AssetSerializer(data=request.DATA)
        print serializer.data

        if serializer.is_valid():

            serializer.save()
            return Response(serializer.data, status=status.HTTP_201_CREATED)

        return Response(serializer.errors, status=status.HTTP_400_BAD_REQUEST)

模型.py

class Asset(models.Model):

    key = models.CharField(max_length=8, unique=True, editable=False)
    name = models.CharField(_('name'), max_length=200)
    source = models.FileField(_('file'), upload_to=upload_to, storage=default_storage)

    ext = models.CharField(max_length=15, editable=False)
    type = models.PositiveIntegerField(choices=ASSET_TYPE, max_length=15, editable=False)
    size = models.PositiveIntegerField(max_length=32, default=0, editable=False)

    _file_meta = models.TextField(editable=False, null=True, blank=True)

    public = models.BooleanField(default=False)
    position = models.PositiveIntegerField(default=1)

    object_id = models.PositiveIntegerField(default=1)
    content_type = models.ForeignKey(ContentType, blank=True, null=True)
    content_object = generic.GenericForeignKey('content_type', 'object_id')

    created = models.DateTimeField(auto_now_add=True)
    modified = models.DateTimeField(auto_now_add=True, auto_now=True)

我是Django REST API的新手,我读了文档http://www.django-rest-framework.org/api-guide/parsers.html#fileuploadparser,但仍然不明白如何操作。


可能是Django Rest Framework文件上传的重复问题。 - Dhia
2个回答

25
首先,您需要在视图中定义一个解析器。这是因为API需要知道要查找哪些头信息。浏览器会将文件作为表单数据传输,因此您需要同时使用MultiPartParser和FormParser。您还可以使用FileUploadParser,但需要确保客户端发送正确的HTTP头信息。

首先,您需要在视图中定义一个解析器。这是因为API需要知道要查找哪些头信息。浏览器会将文件作为表单数据传输,因此您需要同时使用MultiPartParser和FormParser。您还可以使用FileUploadParser,但需要确保客户端发送正确的HTTP头信息。

from rest_framework.parsers import MultiPartParser, FormParser

class AssetAdd(APIView):
    parser_classes = (MultiPartParser, FormParser,)

然后,在post方法中,您的文件将出现在FILES QueryDict中:

def post(self, request, format=None):
    my_file = request.FILES['file_field_name']
    filename = '/tmp/myfile'
    with open(filename, 'wb+') as temp_file:
        for chunk in my_file.chunks():
            temp_file.write(chunk)

    my_saved_file = open(filename) #there you go

请问您能否详细地解释一下吗? - blaz1988
我添加了一个名为AssetAdd的类(APIView): parser_classes = (MultiPartParser, FormParser,)def post(self, request, format=None): my_file = request.FILES['file_field_name'] print my_file - blaz1988
1
当我尝试执行{"name": "旧金山大桥", "source": "assets/34/3492/3492AA/3492AA/source.jpg"}时,却遇到错误{ "detail": "多部分表单解析错误 - 多部分中的无效边界: None" } - blaz1988
我可以从哪里导入MultiPartParser和FormParser? - LondonAppDev
1
@MarkWinterbottom 导入 MultiPartParser 和 FormParser 类,它们位于 rest_framework.parsers 模块中。 - fixmycode
显示剩余3条评论

0

请按照以下方式编写您的模型:

from django.db import models
class PicModel(models.Model):
    file_to_be_uploaded = models.FileField(upload_to='/directory/where/you/what/to/upload/to')

如果你要上传图片,则使用 ImageField,但需要安装 Pillow,命令为 pip install pillow
在你的 serializers.py 文件中,假设你正在使用 rest_framework
from rest_framework import serializers
from app_name.models import PicModel

class PicSerializer(serializers.ModelSerializer):
    class Meta:
        model = PicModel
        fields = '__all__' #or any way you may like to display you fields

在你的 views.py 中包含,假设你正在使用基于类的视图:
from app_name.models import PicModel
from app_name.serializers import PicModelSerializer
#other necessary imports

class PicList(APIView):
    def post(self, request, format=None):#or format can be json or api
        serializer = PicModelSerializer(data=request.data)
        if serializer.is_valid():
            serializer.save()
            ............more code

要测试JSON数据的写法如下:

{
    "file_to_be_uploaded":"//path//to//file//on//local//machine"
}

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接