Загрузочный образ Django REST Framework: "Представленные данные не были файлом"
Я склоняюсь к тому, как загружать файл в Django, и здесь я сталкиваюсь с непростой задачей с ошибкой:
Представленные данные не были файлом. Проверьте тип кодировки в форме.
Ниже представлена подробная информация.
Примечание: Я также рассмотрел Django Rest Framework ImageField, и я попробовал
serializer = ImageSerializer(data=request.data, files=request.FILES)
но я получаю
TypeError: __init__()
получил неожиданный аргумент ключевого слова 'files'
У меня есть модель Image
, с которой я бы хотел взаимодействовать с помощью структуры Django REST:
models.py
class Image(models.Model):
image = models.ImageField(upload_to='item_images')
owner = models.ForeignKey(
User, related_name='uploaded_item_images',
blank=False,
)
time_created = models.DateTimeField(auto_now_add=True)
serializers.py
class ImageSerializer(serializers.ModelSerializer):
image = serializers.ImageField(
max_length=None, use_url=True,
)
class Meta:
model = Image
fields = ("id", 'image', 'owner', 'time_created', )
settings.py
'DEFAULT_PARSER_CLASSES': (
'rest_framework.parsers.JSONParser',
'rest_framework.parsers.FormParser',
'rest_framework.parsers.MultiPartParser',
),
Передняя часть (используя AngularJS и angular-restmod
или $resource
) отправляет JSON
данные с owner
и Image
формы:
Ввод:
{"owner": 5, "image": "..."}
В бэкэнде request.data
отображается
{u'owner': 5, u'image': u'..."}
Но тогда ImageSerializer(data=request.data).errors
показывает ошибку
ReturnDict([('image', [u'The submitted data was not a file. Check the encoding type on the form.'])])
Интересно, что я должен сделать, чтобы исправить ошибку?
EDIT: JS часть
Связанные коды front end состоят из двух частей: a angular-file-dnd
directive
(доступно здесь) для удаления файла на страницу и angular-restmod
, который предоставляет операции CRUD:
<!-- The template: according to angular-file-dnd, -->
<!-- it will store the dropped image into variable $scope.image -->
<div file-dropzone="[image/png, image/jpeg, image/gif]" file="image" class='method' data-max-file-size="3" file-name="imageFileName">
<div layout='row' layout-align='center'>
<i class="fa fa-upload" style='font-size:50px;'></i>
</div>
<div class='text-large'>Drap & drop your photo here</div>
</div>
# A simple `Image` `model` to perform `POST`
$scope.image_resource = Image.$build();
$scope.upload = function() {
console.log("uploading");
$scope.image_resource.image = $scope.image;
$scope.image_resource.owner = Auth.get_profile().user_id;
return $scope.image_resource.$save();
};
Обновление проблемы: прямо сейчас я переключился на ng-file-upload
, который отправляет данные изображения в надлежащем формате.
Ответы
Ответ 1
Проблема, с которой вы сталкиваетесь, заключается в том, что инфраструктура Django REST ожидает, что файлы будут загружены в виде данных с несколькими форматами, с помощью стандартных способов загрузки файлов. Это обычно поле file
, но объект JavaScript Blob
также работает для AJAX.
Вы хотите загрузить файлы с использованием кодированной base64 строки вместо необработанного файла, который не поддерживается по умолчанию. Существуют варианты Base64ImageField
там, но наиболее перспективный пришел с помощью запроса на перенос.
Так как они были в основном разработаны для Django REST framework 2.x, я улучшил его с помощью запроса на pull и создал тот, который должен быть совместим с DRF 3.
serializers.py
from rest_framework import serializers
class Base64ImageField(serializers.ImageField):
"""
A Django REST framework field for handling image-uploads through raw post data.
It uses base64 for encoding and decoding the contents of the file.
Heavily based on
https://github.com/tomchristie/django-rest-framework/pull/1268
Updated for Django REST framework 3.
"""
def to_internal_value(self, data):
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:
self.fail('invalid_image')
# Generate file name:
file_name = str(uuid.uuid4())[:12] # 12 characters are more than enough.
# Get the file name extension:
file_extension = self.get_file_extension(file_name, decoded_file)
complete_file_name = "%s.%s" % (file_name, file_extension, )
data = ContentFile(decoded_file, name=complete_file_name)
return super(Base64ImageField, self).to_internal_value(data)
def get_file_extension(self, file_name, decoded_file):
import imghdr
extension = imghdr.what(file_name, decoded_file)
extension = "jpg" if extension == "jpeg" else extension
return extension
Это должно использоваться для замены стандартного ImageField
, предоставляемого инфраструктурой Django REST. Таким образом, ваш сериализатор станет
class ImageSerializer(serializers.ModelSerializer):
image = Base64ImageField(
max_length=None, use_url=True,
)
class Meta:
model = Image
fields = ("id", 'image', 'owner', 'time_created', )
Это должно позволить вам указать строку с кодировкой base64 или стандартный объект Blob
, который обычно ожидает инфраструктура Django REST.
Ответ 2
Я столкнулся с той же проблемой несколько дней назад. Вот мое представление django rest framework для обработки загрузки файлов
views.py
class PhotoUploadView(APIView):
parser_classes = (FileUploadParser,)
def post(self, request):
user = self.request.user
if not user:
return Response(status=status.HTTP_403_FORBIDDEN)
profile = None
data = None
photo = None
file_form = FileUploadForm(request.POST,request.FILES)
if file_form.is_valid():
photo = request.FILES['file']
else:
return Response(ajax_response(file_form),status=status.HTTP_406_NOT_ACCEPTABLE)
try:
profile = Organizer.objects.get(user=user)
profile.photo = photo
profile.save()
data = OrganizersSerializer(profile).data
except Organizer.DoesNotExist:
profile = Student.objects.get(user=user)
profile.photo = photo
profile.save()
data = StudentsSerializer(profile).data
return Response(data)
В интерфейсе я использовал angular-file-upload lib.
Вот мой ввод файла
<div ng-file-drop="" ng-file-select="" ng-model="organizer.photo" class="drop-box" drag-over-class="{accept:'dragover', reject:'dragover-err', delay:100}" ng-multiple="false" allow-dir="true" accept="image/*">
Drop Images or PDFs<div>here</div>
</div>
И вот моя служба загрузки
main.js
(function () {
'use strict';
angular
.module('trulii.utils.services')
.factory('UploadFile', UploadFile);
UploadFile.$inject = ['$cookies', '$http','$upload','$window','Authentication'];
/**
* @namespace Authentication
* @returns {Factory}
*/
function UploadFile($cookies, $http,$upload,$window,Authentication) {
/**
* @name UploadFile
* @desc The Factory to be returned
*/
var UploadFile = {
upload_file: upload_file,
};
return UploadFile;
function upload_file(file) {
return $upload.upload({
url: '/api/users/upload/photo/', // upload.PHP скрипт, node.js route, or servlet url
//method: 'POST' or 'PUT',
//headers: {'Authorization': 'xxx'}, // only for html5
//withCredentials: true,
file: file, // single file or a list of files. list is only for html5
//fileName: 'doc.jpg' or ['1.jpg', '2.jpg', ...] // to modify the name of the file(s)
//fileFormDataName: myFile, // file formData name ('Content-Disposition'), server side request form name
// could be a list of names for multiple files (html5). Default is 'file'
//formDataAppender: function(formData, key, val){} // customize how data is added to the formData.
// See #40#issuecomment-28612000 for sample code
})
}
}
})();
Ответ 3
Я обнаружил, что эта нить пытается загрузить изображения в моем проекте, это делает меня отличной загрузкой при добавлении изображения, нет проблем, проблема возникает при обновлении этой записи, не обязательно для изображения, если не для какого-либо поля, он отправляет мне статус ошибки: 500, statusText: "Internal Server Error". Я снимаю изображение и могу поставить, может кто-нибудь мне помочь в этом?