FileUploadParser не получает имя файла

Я просто хочу создать REST API, который получает файл, обрабатывает его и возвращает некоторую информацию. Моя проблема в том, что я следую этому примеру: http://www.django-rest-framework.org/api-guide/parsers/#fileuploadparser

и я не могу заставить его работать с помощью Postman или curl, я думаю, что мне чего-то не хватает. Парсер всегда дает мне эти две ошибки:

  • fileupload с ошибка - никто из загрузки обработчиков может обрабатывать поток
  • отсутствует имя файла. Запрос должен включать заголовок Content-Disposition с параметром filename.

этот код:

views.py:

class FileUploadView(APIView):
    parser_classes = (FileUploadParser,)

    def post(self, request, filename, format=None):
        file_obj = request.data['file']
        # ...
        # do some stuff with uploaded file
        # ...
        return Response(status=204)

    def put(self, request, filename, format=None):
        file_obj = request.data['file']
        # ...
        # do some stuff with uploaded file
        # ...
        return Response(status=204)

urls.py

urlpatterns = [
   url(r'predict/(?P<filename>[^/]+)$', app.views.FileUploadView.as_view())
]

settings.py

"""
Django settings for GenderAPI project.

Generated by 'django-admin startproject' using Django 1.9.1.

For more information on this file, see
https://docs.djangoproject.com/en/1.9/topics/settings/

For the full list of settings and their values, see
https://docs.djangoproject.com/en/1.9/ref/settings/
"""

import os
import posixpath


LOGGING = {
    'version': 1,
    'disable_existing_loggers': False,
    'handlers': {
        'file': {
            'level': 'DEBUG',
            'class': 'logging.FileHandler',
            'filename': 'debug.log',
        },
    },
    'loggers': {
        'django': {
            'handlers': ['file'],
            'level': 'DEBUG',
            'propagate': True,
        },
    },
}

# Build paths inside the project like this: os.path.join(BASE_DIR, ...)
BASE_DIR = os.path.dirname(os.path.dirname(os.path.abspath(__file__)))


# Quick-start development settings - unsuitable for production
# See https://docs.djangoproject.com/en/1.9/howto/deployment/checklist/

# SECURITY WARNING: keep the secret key used in production secret!
SECRET_KEY = removed

# SECURITY WARNING: don't run with debug turned on in production!
DEBUG = True

ALLOWED_HOSTS = ['localhost','127.0.0.1']

REST_FRAMEWORK = {
    'DEFAULT_PARSER_CLASSES': (
        'rest_framework.parsers.FileUploadParser'
    )
}


# Application definition

INSTALLED_APPS = [    

    # Add your apps here to enable them
    'django.contrib.admin',
    'django.contrib.auth',
    'django.contrib.contenttypes',
    'django.contrib.sessions',
    'django.contrib.messages',
    'django.contrib.staticfiles',   
    'rest_framework',
    'app'  


]

MIDDLEWARE = [

    'django.middleware.security.SecurityMiddleware',
    'django.contrib.sessions.middleware.SessionMiddleware',
    'django.middleware.common.CommonMiddleware',
    'django.middleware.csrf.CsrfViewMiddleware',
    'django.contrib.auth.middleware.AuthenticationMiddleware',
    'django.contrib.auth.middleware.SessionAuthenticationMiddleware',
    'django.contrib.messages.middleware.MessageMiddleware',
    'django.middleware.clickjacking.XFrameOptionsMiddleware'
]

ROOT_URLCONF = 'GenderAPI.urls'

TEMPLATES = [
    {
        'BACKEND': 'django.template.backends.django.DjangoTemplates',
        'DIRS': [],
        'APP_DIRS': True,
        'OPTIONS': {
            'context_processors': [
                'django.template.context_processors.debug',
                'django.template.context_processors.request',
                'django.contrib.auth.context_processors.auth',
                'django.contrib.messages.context_processors.messages',
            ],
        },
    },
]

WSGI_APPLICATION = 'GenderAPI.wsgi.application'

# Database
# https://docs.djangoproject.com/en/1.9/ref/settings/#databases

DATABASES = {
    'default': {
        'ENGINE': 'django.db.backends.sqlite3',
        'NAME': os.path.join(BASE_DIR, 'db.sqlite3'),
    }
}

# Static files (CSS, JavaScript, Images)
# https://docs.djangoproject.com/en/1.9/howto/static-files/

STATIC_URL = '/static/'
STATIC_ROOT = posixpath.join(*(BASE_DIR.split(os.path.sep) + ['static']))

FILE_UPLOAD_TEMP_DIR = BASE_DIR
MEDIA_URL  = '/media/'

здесь вы можете увидеть захват почтальона (я пробовал все):

PUT /predict/pabloGrande.jpg HTTP/1.1
Host: 127.0.0.1:52276
Content-Type: multipart/form-data; boundary=----WebKitFormBoundary7MA4YWxkTrZu0gW

------WebKitFormBoundary7MA4YWxkTrZu0gW
Content-Disposition: form-data; name="file"; filename="04320cf.jpg"
Content-Type: image/jpeg


------WebKitFormBoundary7MA4YWxkTrZu0gW--

требования:

bleach==1.5.0
Django==1.11.6
djangorestframework==3.7.1
html5lib==0.9999999
Markdown==2.6.9
numpy==1.13.3
olefile==0.44
pandas==0.20.3
Pillow==4.3.0
pip==9.0.1
protobuf==3.4.0
python-dateutil==2.6.1
pytz==2017.2
scipy==1.0.0rc1
setuptools==28.8.0
six==1.11.0
tensorflow==1.3.0
tensorflow-tensorboard==0.1.8
Werkzeug==0.12.2
wheel==0.30.0

Спасибо за помощь

5 ответов


в рамках отдыха django. у нас есть такие компоненты, как Парсеры, визуализаторы и сериализаторы.

  • ответственность парсеров заключается в анализе данных, которые отправляются методами запроса GET, POST и PUT и т. д.

  • синтаксический анализатор по умолчанию, используемый в django REST, -'JSONParser'. Он анализирует только данные JSON [числа, строки, даты]. Он игнорирует данные, как файлы.

  • для того, чтобы разобрать файлы, которые нам нужны использовать Парсеры типа"MultiPartParser" или "FormParser".

    Пример Кода :

        from rest_framework.parsers import MultiPartParser
        from rest_framework.response import Response
        from rest_framework.views import APIView
    
        class ExampleView(APIView):
            """
            A view that can accept POST requests with JSON content.
            """
            parser_classes = (MultiPartParser,)
    
            def post(self, request, format=None):
                # to access files
                print request.FILES
                # to access data
                print request.data
                return Response({'received data': request.data})
    

когда мы используем собственность request.data затем parser будет анализировать данные.

ссылки: Django REST Docs, Джанго отдых Github


в вашем views.py измените парсер следующим образом

parser_classes = (JSONParser, MultiPartParser)


о второй ошибке Missing filename. Request should include a Content-Disposition header with a filename parameter. с помощью почтальона я удалил заголовок Content-Type :multipart/form-data и был успешным.

похоже, проблема заключается в том, что пользовательский заголовок Content-Type переопределяет заголовок Content-Type по умолчанию, который должен быть отправлен. Проверьте этот поток для справки Content-Type для составных сообщений


я сталкиваюсь с той же проблемой.Сообщение об ошибке проблемы показывает:

{"detail": "отсутствует имя файла. Запрос должен включать Content - заголовок диспозиции с параметром filename."} Я делаю все шаги выше моего ответа, но это не работает.Наконец,

Я нахожу причина в серверной viewset.

это показывает, как это

parser_classes = (FileUploadParser, MultiPartParser, FormParser)

и я удаляю FileUploadParser

  parser_classes = ( MultiPartParser, FormParser)

и это работает, поэтому я думаю, что вы должны уделять больше внимания этому


вам не нужно использовать MultipartParser или FormParser.

что вам нужно, это сериализатор с FileField() вот так:

serializers.py:

class FileUploadSerializer(serializers.Serializer):
    # I set use_url to False so I don't need to pass file 
    # through the url itself - defaults to True if you need it
    file = serializers.FileField(use_url=False)

Итак, когда вы пытаетесь получить доступ file здесь ниже, у вас будет дикт с ключом с именем file. Лично я, вероятно, назвал бы это чем-то более описательным, чем просто "файл", но это зависит от вас.

from .serializers import FileUploadSerializer

    class FileUploadView(APIView):

       def post(self, request):
           # set 'data' so that you can use 'is_vaid()' and raise exception
           # if the file fails validation
           serializer = FileUploadSerializer(data=request.data)
           serializer.is_valid(raise_exception=True)
           # once validated, grab the file from the request itself
           file = request.FILES['file']