Как скопировать объект InmemoryUploadedFile на диск
-
02-10-2019 - |
Вопрос
Я пытаюсь поймать файл, отправленный с формой, и выполнить некоторые операции на нем, прежде чем он будет сохранен. Поэтому мне нужно создать копию этого файла в каталоге Temp, но я не знаю, как его достичь. Функции Shutil не копируют этот файл, поскольку к нему нет пути. Так есть ли способ сделать эту операцию каким -либо другим способом?
Мой код:
image = form.cleaned_data['image']
temp = os.path.join(settings.PROJECT_PATH, 'tmp')
sourceFile = image.name # without .name here it wasn't working either
import shutil
shutil.copy(sourceFile, temp)
Что поднимает:
Exception Type: IOError at /
Exception Value: (2, 'No such file or directory')
И отладка:
# (..)\views.py in function
67. sourceFile = image.name
68. import shutil
69. shutil.copy2(sourceFile, temp) ...
# (..)\Python26\lib\shutil.py in copy2
92. """Copy data and all stat info ("cp -p src dst").
93.
94. The destination may be a directory.
95.
96. """
97. if os.path.isdir(dst):
98. dst = os.path.join(dst, os.path.basename(src))
99. copyfile(src, dst) ...
100. copystat(src, dst)
101.
▼ Local vars
Variable Value
dst
u'(..)\\tmp\\myfile.JPG'
src
u'myfile.JPG'
# (..)\Python26\lib\shutil.py in copyfile
45. """Copy data from src to dst"""
46. if _samefile(src, dst):
47. raise Error, "`%s` and `%s` are the same file" % (src, dst)
48.
49. fsrc = None
50. fdst = None
51. try:
52. fsrc = open(src, 'rb') ...
53. fdst = open(dst, 'wb')
54. copyfileobj(fsrc, fdst)
55. finally:
56. if fdst:
57. fdst.close()
58. if fsrc:
▼ Local vars
Variable Value
dst
u'(..)\\tmp\\myfile.JPG'
fdst
None
fsrc
None
src
u'myfile.JPG'
Решение
Этот Аналогичный вопрос, это может помочь.
import os
from django.core.files.storage import default_storage
from django.core.files.base import ContentFile
from django.conf import settings
data = request.FILES['image'] # or self.files['image'] in your form
path = default_storage.save('tmp/somename.mp3', ContentFile(data.read()))
tmp_file = os.path.join(settings.MEDIA_ROOT, path)
Другие советы
Как упоминалось @UPS, при загрузке больших файлов вы не хотите засорять системную память data.read()
.
От Django Docs :
Петля над
UploadedFile.chunks()
вместо использованияread()
гарантирует, что большие файлы не перегружают память вашей системы
from django.core.files.storage import default_storage
filename = "whatever.xyz" # received file name
file_obj = request.data['file']
with default_storage.open('tmp/'+filename, 'wb+') as destination:
for chunk in file_obj.chunks():
destination.write(chunk)
Это сохранит файл на MEDIA_ROOT/tmp/
как твой default_storage
воля, если не сказано иначе.
Ваш лучший курс действия - написать пользовательский обработчик загрузки. Увидеть док Анкет Если вы добавите обработчик "file_complete", вы можете получить доступ к контенту файла независимо от файла памяти или файла температуры. Вы также можете использовать метод "CETER_DATA_CHUNCK" и написать свою копию внутри него.
С уважением
Вот еще один способ сделать это с Python mkstemp
:
### get the inmemory file
data = request.FILES.get('file') # get the file from the curl
### write the data to a temp file
tup = tempfile.mkstemp() # make a tmp file
f = os.fdopen(tup[0], 'w') # open the tmp file for writing
f.write(data.read()) # write the tmp file
f.close()
### return the path of the file
filepath = tup[1] # get the filepath
return filepath