我在使用多部分/表单数据、嵌套序列化程序和上传图像/文件时遇到了这个问题。我认为我的模型和序列化程序工作得很好,但在解析表单数据的过程中,有时会失败。
我刚刚扩展了BaseParser,将字符串解析为列表和dict。
parsers.py
class MultiPartJSONParser(BaseParser):
media_type = 'multipart/form-data'
def parse(self, stream, media_type=None, parser_context=None):
parser_context = parser_context or {}
request = parser_context['request']
encoding = parser_context.get('encoding', settings.DEFAULT_CHARSET)
meta = request.META.copy()
meta['CONTENT_TYPE'] = media_type
upload_handlers = request.upload_handlers
try:
parser = DjangoMultiPartParser(meta, stream, upload_handlers, encoding)
data, files = parser.parse()
data = data.dict()
for key in data:
if data[key]:
try:
data[key] = json.loads(data[key])
except ValueError as e:
pass
return DataAndFiles(data, files)
except MultiPartParserError as exc:
raise ParseError('Multipart form parse error - %s' % exc)
我为什么这么做?因为我的意图是传递这样一个表单数据:
键 | |
---|---|
用户 | 6 |
地址 | {"街道":"主街","数字":"1"} |
first_name | 爱丽丝 |
姓氏 | Bob |
image[file] | 。。。\image_directory |
语言 | [1,2] |
我找到了一个解决方案,它不是我想要的,但它很有效。
我发现,如果我使用MultiPartJSONParser,由于任何未知的原因,文件/图像字段都会被插入到列表中。
为了解决这个问题,我必须在调用super().create()
或super().update()
之前从views.py中的列表中提取这些字段
视图.py
class CreateProfileView(CreateAPIView):
serializer_class = ProfileSerializer
parser_classes = [MultiPartJSONParser]
def create(self, request, *args, **kwargs):
for key in request.FILES:
request.data[key] = request.data[key][0]
return super().update(request, *args, **kwargs)
class RUDProfileView(RetrieveUpdateDestroyAPIView):
serializer_class = ProfileSerializer
queryset = UserProfile.objects.all()
lookup_field = 'user'
parser_classes = [MultiPartJSONParser]
def update(self, request, *args, **kwargs):
for key in request.FILES:
request.data[key] = request.data[key][0]
return super().update(request, *args, **kwargs)