Как скопировать объект InMemoryUploadedFile на диск - PullRequest
37 голосов
/ 13 сентября 2010

Я пытаюсь поймать файл, отправленный с формой, и выполнить некоторые операции с ним перед его сохранением.Поэтому мне нужно создать копию этого файла во временном каталоге, но я не знаю, как этого достичь.Функции 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 /<br> 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'

Ответы [ 4 ]

46 голосов
/ 14 сентября 2010

Этот похож на вопрос, он может помочь.

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)
13 голосов
/ 12 мая 2015

Как упоминалось @ 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, если не указано иное.

5 голосов
/ 13 сентября 2010

Ваш лучший способ действий - написать собственный обработчик Upload.См. документы .Если вы добавите обработчик file_complete, вы сможете получить доступ к содержимому файла независимо от наличия файла памяти или файла временного пути.Вы также можете использовать метод receive_data_chunck и написать свою копию в нем.

С уважением

4 голосов
/ 20 августа 2013

Вот еще один способ сделать это с помощью 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
...