mirror of
				https://github.com/django/django.git
				synced 2025-10-24 22:26:08 +00:00 
			
		
		
		
	Fixed #7614: the quickening has come, and there now is only one UploadedFile. On top of that, UploadedFile's interface has been improved:
* The API now more closely matches a proper file API. This unfortunately means a few backwards-incompatible renamings; see BackwardsIncompatibleChanges. This refs #7593. * While we were at it, renamed chunk() to chunks() to clarify that it's an iterator. * Temporary uploaded files now property use the tempfile library behind the scenes which should ensure better cleanup of tempfiles (refs #7593 again). Thanks to Mike Axiak for the bulk of this patch. git-svn-id: http://code.djangoproject.com/svn/django/trunk@7859 bcc190cf-cafb-0310-a4f2-bffc1f526a37
This commit is contained in:
		| @@ -3,12 +3,40 @@ Classes representing uploaded files. | |||||||
| """ | """ | ||||||
|  |  | ||||||
| import os | import os | ||||||
|  | import tempfile | ||||||
|  | import warnings | ||||||
| try: | try: | ||||||
|     from cStringIO import StringIO |     from cStringIO import StringIO | ||||||
| except ImportError: | except ImportError: | ||||||
|     from StringIO import StringIO |     from StringIO import StringIO | ||||||
|  |  | ||||||
| __all__ = ('UploadedFile', 'TemporaryUploadedFile', 'InMemoryUploadedFile') | from django.conf import settings | ||||||
|  |  | ||||||
|  | __all__ = ('UploadedFile', 'TemporaryUploadedFile', 'InMemoryUploadedFile', 'SimpleUploadedFile') | ||||||
|  |  | ||||||
|  | # Because we fooled around with it a bunch, UploadedFile has a bunch | ||||||
|  | # of deprecated properties. This little shortcut helps define 'em | ||||||
|  | # without too much code duplication. | ||||||
|  | def deprecated_property(old, new, readonly=False): | ||||||
|  |     def issue_warning(): | ||||||
|  |         warnings.warn( | ||||||
|  |             message = "UploadedFile.%s is deprecated; use UploadedFile.%s instead." % (old, new), | ||||||
|  |             category = DeprecationWarning, | ||||||
|  |             stacklevel = 3 | ||||||
|  |         ) | ||||||
|  |      | ||||||
|  |     def getter(self): | ||||||
|  |         issue_warning() | ||||||
|  |         return getattr(self, new) | ||||||
|  |          | ||||||
|  |     def setter(self, value): | ||||||
|  |         issue_warning() | ||||||
|  |         setattr(self, new, value) | ||||||
|  |          | ||||||
|  |     if readonly: | ||||||
|  |         return property(getter) | ||||||
|  |     else: | ||||||
|  |         return property(getter, setter) | ||||||
|  |  | ||||||
| class UploadedFile(object): | class UploadedFile(object): | ||||||
|     """ |     """ | ||||||
| @@ -20,34 +48,34 @@ class UploadedFile(object): | |||||||
|     """ |     """ | ||||||
|     DEFAULT_CHUNK_SIZE = 64 * 2**10 |     DEFAULT_CHUNK_SIZE = 64 * 2**10 | ||||||
|  |  | ||||||
|     def __init__(self, file_name=None, content_type=None, file_size=None, charset=None): |     def __init__(self, name=None, content_type=None, size=None, charset=None): | ||||||
|         self.file_name = file_name |         self.name = name | ||||||
|         self.file_size = file_size |         self.size = size | ||||||
|         self.content_type = content_type |         self.content_type = content_type | ||||||
|         self.charset = charset |         self.charset = charset | ||||||
|  |  | ||||||
|     def __repr__(self): |     def __repr__(self): | ||||||
|         return "<%s: %s (%s)>" % (self.__class__.__name__, self.file_name, self.content_type) |         return "<%s: %s (%s)>" % (self.__class__.__name__, self.name, self.content_type) | ||||||
|  |  | ||||||
|     def _set_file_name(self, name): |     def _get_name(self): | ||||||
|  |         return self._name | ||||||
|  |  | ||||||
|  |     def _set_name(self, name): | ||||||
|         # Sanitize the file name so that it can't be dangerous. |         # Sanitize the file name so that it can't be dangerous. | ||||||
|         if name is not None: |         if name is not None: | ||||||
|             # Just use the basename of the file -- anything else is dangerous. |             # Just use the basename of the file -- anything else is dangerous. | ||||||
|             name = os.path.basename(name) |             name = os.path.basename(name) | ||||||
|              |  | ||||||
|             # File names longer than 255 characters can cause problems on older OSes. |             # File names longer than 255 characters can cause problems on older OSes. | ||||||
|             if len(name) > 255: |             if len(name) > 255: | ||||||
|                 name, ext = os.path.splitext(name) |                 name, ext = os.path.splitext(name) | ||||||
|                 name = name[:255 - len(ext)] + ext |                 name = name[:255 - len(ext)] + ext | ||||||
|                  |  | ||||||
|         self._file_name = name |  | ||||||
|          |  | ||||||
|     def _get_file_name(self): |  | ||||||
|         return self._file_name |  | ||||||
|          |  | ||||||
|     file_name = property(_get_file_name, _set_file_name) |  | ||||||
|  |  | ||||||
|     def chunk(self, chunk_size=None): |         self._name = name | ||||||
|  |  | ||||||
|  |     name = property(_get_name, _set_name) | ||||||
|  |  | ||||||
|  |     def chunks(self, chunk_size=None): | ||||||
|         """ |         """ | ||||||
|         Read the file and yield chucks of ``chunk_size`` bytes (defaults to |         Read the file and yield chucks of ``chunk_size`` bytes (defaults to | ||||||
|         ``UploadedFile.DEFAULT_CHUNK_SIZE``). |         ``UploadedFile.DEFAULT_CHUNK_SIZE``). | ||||||
| @@ -58,12 +86,18 @@ class UploadedFile(object): | |||||||
|         if hasattr(self, 'seek'): |         if hasattr(self, 'seek'): | ||||||
|             self.seek(0) |             self.seek(0) | ||||||
|         # Assume the pointer is at zero... |         # Assume the pointer is at zero... | ||||||
|         counter = self.file_size |         counter = self.size | ||||||
|  |  | ||||||
|         while counter > 0: |         while counter > 0: | ||||||
|             yield self.read(chunk_size) |             yield self.read(chunk_size) | ||||||
|             counter -= chunk_size |             counter -= chunk_size | ||||||
|  |  | ||||||
|  |     # Deprecated properties | ||||||
|  |     file_name = deprecated_property(old="file_name", new="name") | ||||||
|  |     file_size = deprecated_property(old="file_size", new="size") | ||||||
|  |     data = deprecated_property(old="data", new="read", readonly=True) | ||||||
|  |     chunk = deprecated_property(old="chunk", new="chunks", readonly=True) | ||||||
|  |  | ||||||
|     def multiple_chunks(self, chunk_size=None): |     def multiple_chunks(self, chunk_size=None): | ||||||
|         """ |         """ | ||||||
|         Returns ``True`` if you can expect multiple chunks. |         Returns ``True`` if you can expect multiple chunks. | ||||||
| @@ -74,9 +108,9 @@ class UploadedFile(object): | |||||||
|         """ |         """ | ||||||
|         if not chunk_size: |         if not chunk_size: | ||||||
|             chunk_size = UploadedFile.DEFAULT_CHUNK_SIZE |             chunk_size = UploadedFile.DEFAULT_CHUNK_SIZE | ||||||
|         return self.file_size < chunk_size |         return self.size > chunk_size | ||||||
|  |  | ||||||
|     # Abstract methods; subclasses *must* default read() and probably should |     # Abstract methods; subclasses *must* define read() and probably should | ||||||
|     # define open/close. |     # define open/close. | ||||||
|     def read(self, num_bytes=None): |     def read(self, num_bytes=None): | ||||||
|         raise NotImplementedError() |         raise NotImplementedError() | ||||||
| @@ -87,23 +121,49 @@ class UploadedFile(object): | |||||||
|     def close(self): |     def close(self): | ||||||
|         pass |         pass | ||||||
|  |  | ||||||
|  |     def xreadlines(self): | ||||||
|  |         return self | ||||||
|  |  | ||||||
|  |     def readlines(self): | ||||||
|  |         return list(self.xreadlines()) | ||||||
|  |  | ||||||
|  |     def __iter__(self): | ||||||
|  |         # Iterate over this file-like object by newlines | ||||||
|  |         buffer_ = None | ||||||
|  |         for chunk in self.chunks(): | ||||||
|  |             chunk_buffer = StringIO(chunk) | ||||||
|  |  | ||||||
|  |             for line in chunk_buffer: | ||||||
|  |                 if buffer_: | ||||||
|  |                     line = buffer_ + line | ||||||
|  |                     buffer_ = None | ||||||
|  |  | ||||||
|  |                 # If this is the end of a line, yield | ||||||
|  |                 # otherwise, wait for the next round | ||||||
|  |                 if line[-1] in ('\n', '\r'): | ||||||
|  |                     yield line | ||||||
|  |                 else: | ||||||
|  |                     buffer_ = line | ||||||
|  |  | ||||||
|  |         if buffer_ is not None: | ||||||
|  |             yield buffer_ | ||||||
|  |  | ||||||
|     # Backwards-compatible support for uploaded-files-as-dictionaries. |     # Backwards-compatible support for uploaded-files-as-dictionaries. | ||||||
|     def __getitem__(self, key): |     def __getitem__(self, key): | ||||||
|         import warnings |  | ||||||
|         warnings.warn( |         warnings.warn( | ||||||
|             message = "The dictionary access of uploaded file objects is deprecated. Use the new object interface instead.", |             message = "The dictionary access of uploaded file objects is deprecated. Use the new object interface instead.", | ||||||
|             category = DeprecationWarning, |             category = DeprecationWarning, | ||||||
|             stacklevel = 2 |             stacklevel = 2 | ||||||
|         ) |         ) | ||||||
|         backwards_translate = { |         backwards_translate = { | ||||||
|             'filename': 'file_name', |             'filename': 'name', | ||||||
|             'content-type': 'content_type', |             'content-type': 'content_type', | ||||||
|             } |         } | ||||||
|  |  | ||||||
|         if key == 'content': |         if key == 'content': | ||||||
|             return self.read() |             return self.read() | ||||||
|         elif key == 'filename': |         elif key == 'filename': | ||||||
|             return self.file_name |             return self.name | ||||||
|         elif key == 'content-type': |         elif key == 'content-type': | ||||||
|             return self.content_type |             return self.content_type | ||||||
|         else: |         else: | ||||||
| @@ -113,34 +173,36 @@ class TemporaryUploadedFile(UploadedFile): | |||||||
|     """ |     """ | ||||||
|     A file uploaded to a temporary location (i.e. stream-to-disk). |     A file uploaded to a temporary location (i.e. stream-to-disk). | ||||||
|     """ |     """ | ||||||
|  |     def __init__(self, name, content_type, size, charset): | ||||||
|     def __init__(self, file, file_name, content_type, file_size, charset): |         super(TemporaryUploadedFile, self).__init__(name, content_type, size, charset) | ||||||
|         super(TemporaryUploadedFile, self).__init__(file_name, content_type, file_size, charset) |         if settings.FILE_UPLOAD_TEMP_DIR: | ||||||
|         self.file = file |             self._file = tempfile.NamedTemporaryFile(suffix='.upload', dir=settings.FILE_UPLOAD_TEMP_DIR) | ||||||
|         self.path = file.name |         else: | ||||||
|         self.file.seek(0) |             self._file = tempfile.NamedTemporaryFile(suffix='.upload') | ||||||
|  |  | ||||||
|     def temporary_file_path(self): |     def temporary_file_path(self): | ||||||
|         """ |         """ | ||||||
|         Returns the full path of this file. |         Returns the full path of this file. | ||||||
|         """ |         """ | ||||||
|         return self.path |         return self.name | ||||||
|  |      | ||||||
|     def read(self, *args, **kwargs): |     # Most methods on this object get proxied to NamedTemporaryFile. | ||||||
|         return self.file.read(*args, **kwargs) |     # We can't directly subclass because NamedTemporaryFile is actually a | ||||||
|  |     # factory function | ||||||
|     def open(self): |     def read(self, *args):          return self._file.read(*args) | ||||||
|         self.seek(0) |     def seek(self, offset):         return self._file.seek(offset) | ||||||
|  |     def write(self, s):             return self._file.write(s) | ||||||
|     def seek(self, *args, **kwargs): |     def close(self):                return self._file.close() | ||||||
|         self.file.seek(*args, **kwargs) |     def __iter__(self):             return iter(self._file) | ||||||
|  |     def readlines(self, size=None): return self._file.readlines(size) | ||||||
|  |     def xreadlines(self):           return self._file.xreadlines() | ||||||
|  |  | ||||||
| class InMemoryUploadedFile(UploadedFile): | class InMemoryUploadedFile(UploadedFile): | ||||||
|     """ |     """ | ||||||
|     A file uploaded into memory (i.e. stream-to-memory). |     A file uploaded into memory (i.e. stream-to-memory). | ||||||
|     """ |     """ | ||||||
|     def __init__(self, file, field_name, file_name, content_type, file_size, charset): |     def __init__(self, file, field_name, name, content_type, size, charset): | ||||||
|         super(InMemoryUploadedFile, self).__init__(file_name, content_type, file_size, charset) |         super(InMemoryUploadedFile, self).__init__(name, content_type, size, charset) | ||||||
|         self.file = file |         self.file = file | ||||||
|         self.field_name = field_name |         self.field_name = field_name | ||||||
|         self.file.seek(0) |         self.file.seek(0) | ||||||
| @@ -154,7 +216,7 @@ class InMemoryUploadedFile(UploadedFile): | |||||||
|     def read(self, *args, **kwargs): |     def read(self, *args, **kwargs): | ||||||
|         return self.file.read(*args, **kwargs) |         return self.file.read(*args, **kwargs) | ||||||
|  |  | ||||||
|     def chunk(self, chunk_size=None): |     def chunks(self, chunk_size=None): | ||||||
|         self.file.seek(0) |         self.file.seek(0) | ||||||
|         yield self.read() |         yield self.read() | ||||||
|  |  | ||||||
| @@ -168,9 +230,9 @@ class SimpleUploadedFile(InMemoryUploadedFile): | |||||||
|     """ |     """ | ||||||
|     def __init__(self, name, content, content_type='text/plain'): |     def __init__(self, name, content, content_type='text/plain'): | ||||||
|         self.file = StringIO(content or '') |         self.file = StringIO(content or '') | ||||||
|         self.file_name = name |         self.name = name | ||||||
|         self.field_name = None |         self.field_name = None | ||||||
|         self.file_size = len(content or '') |         self.size = len(content or '') | ||||||
|         self.content_type = content_type |         self.content_type = content_type | ||||||
|         self.charset = None |         self.charset = None | ||||||
|         self.file.seek(0) |         self.file.seek(0) | ||||||
|   | |||||||
| @@ -132,21 +132,15 @@ class TemporaryFileUploadHandler(FileUploadHandler): | |||||||
|         Create the file object to append to as data is coming in. |         Create the file object to append to as data is coming in. | ||||||
|         """ |         """ | ||||||
|         super(TemporaryFileUploadHandler, self).new_file(file_name, *args, **kwargs) |         super(TemporaryFileUploadHandler, self).new_file(file_name, *args, **kwargs) | ||||||
|         self.file = TemporaryFile(settings.FILE_UPLOAD_TEMP_DIR) |         self.file = TemporaryUploadedFile(self.file_name, self.content_type, 0, self.charset) | ||||||
|         self.write = self.file.write |  | ||||||
|  |  | ||||||
|     def receive_data_chunk(self, raw_data, start): |     def receive_data_chunk(self, raw_data, start): | ||||||
|         self.write(raw_data) |         self.file.write(raw_data) | ||||||
|  |  | ||||||
|     def file_complete(self, file_size): |     def file_complete(self, file_size): | ||||||
|         self.file.seek(0) |         self.file.seek(0) | ||||||
|         return TemporaryUploadedFile( |         self.file.size = file_size | ||||||
|             file = self.file,  |         return self.file | ||||||
|             file_name = self.file_name,  |  | ||||||
|             content_type = self.content_type,  |  | ||||||
|             file_size = file_size,  |  | ||||||
|             charset = self.charset |  | ||||||
|         ) |  | ||||||
|  |  | ||||||
| class MemoryFileUploadHandler(FileUploadHandler): | class MemoryFileUploadHandler(FileUploadHandler): | ||||||
|     """ |     """ | ||||||
| @@ -189,37 +183,12 @@ class MemoryFileUploadHandler(FileUploadHandler): | |||||||
|         return InMemoryUploadedFile( |         return InMemoryUploadedFile( | ||||||
|             file = self.file, |             file = self.file, | ||||||
|             field_name = self.field_name, |             field_name = self.field_name, | ||||||
|             file_name = self.file_name, |             name = self.file_name, | ||||||
|             content_type = self.content_type, |             content_type = self.content_type, | ||||||
|             file_size = file_size, |             size = file_size, | ||||||
|             charset = self.charset |             charset = self.charset | ||||||
|         ) |         ) | ||||||
|  |  | ||||||
| class TemporaryFile(object): |  | ||||||
|     """ |  | ||||||
|     A temporary file that tries to delete itself when garbage collected. |  | ||||||
|     """ |  | ||||||
|     def __init__(self, dir): |  | ||||||
|         if not dir: |  | ||||||
|             dir = tempfile.gettempdir() |  | ||||||
|         try: |  | ||||||
|             (fd, name) = tempfile.mkstemp(suffix='.upload', dir=dir) |  | ||||||
|             self.file = os.fdopen(fd, 'w+b') |  | ||||||
|         except (OSError, IOError): |  | ||||||
|             raise OSError("Could not create temporary file for uploading, have you set settings.FILE_UPLOAD_TEMP_DIR correctly?") |  | ||||||
|         self.name = name |  | ||||||
|  |  | ||||||
|     def __getattr__(self, name): |  | ||||||
|         a = getattr(self.__dict__['file'], name) |  | ||||||
|         if type(a) != type(0): |  | ||||||
|             setattr(self, name, a) |  | ||||||
|         return a |  | ||||||
|  |  | ||||||
|     def __del__(self): |  | ||||||
|         try: |  | ||||||
|             os.unlink(self.name) |  | ||||||
|         except OSError: |  | ||||||
|             pass |  | ||||||
|  |  | ||||||
| def load_handler(path, *args, **kwargs): | def load_handler(path, *args, **kwargs): | ||||||
|     """ |     """ | ||||||
|   | |||||||
| @@ -536,7 +536,7 @@ class Model(object): | |||||||
|             # This is a normal uploadedfile that we can stream. |             # This is a normal uploadedfile that we can stream. | ||||||
|             fp = open(full_filename, 'wb') |             fp = open(full_filename, 'wb') | ||||||
|             locks.lock(fp, locks.LOCK_EX) |             locks.lock(fp, locks.LOCK_EX) | ||||||
|             for chunk in raw_field.chunk(): |             for chunk in raw_field.chunks(): | ||||||
|                 fp.write(chunk) |                 fp.write(chunk) | ||||||
|             locks.unlock(fp) |             locks.unlock(fp) | ||||||
|             fp.close() |             fp.close() | ||||||
|   | |||||||
| @@ -766,9 +766,12 @@ class FileField(Field): | |||||||
|     def get_db_prep_save(self, value): |     def get_db_prep_save(self, value): | ||||||
|         "Returns field's value prepared for saving into a database." |         "Returns field's value prepared for saving into a database." | ||||||
|         # Need to convert UploadedFile objects provided via a form to unicode for database insertion |         # Need to convert UploadedFile objects provided via a form to unicode for database insertion | ||||||
|         if value is None: |         if hasattr(value, 'name'): | ||||||
|  |             return value.name | ||||||
|  |         elif value is None: | ||||||
|             return None |             return None | ||||||
|         return unicode(value) |         else: | ||||||
|  |             return unicode(value) | ||||||
|  |  | ||||||
|     def get_manipulator_fields(self, opts, manipulator, change, name_prefix='', rel=False, follow=True): |     def get_manipulator_fields(self, opts, manipulator, change, name_prefix='', rel=False, follow=True): | ||||||
|         field_list = Field.get_manipulator_fields(self, opts, manipulator, change, name_prefix, rel, follow) |         field_list = Field.get_manipulator_fields(self, opts, manipulator, change, name_prefix, rel, follow) | ||||||
| @@ -842,7 +845,7 @@ class FileField(Field): | |||||||
|             # We don't need to raise a warning because Model._save_FIELD_file will |             # We don't need to raise a warning because Model._save_FIELD_file will | ||||||
|             # do so for us. |             # do so for us. | ||||||
|             try: |             try: | ||||||
|                 file_name = file.file_name |                 file_name = file.name | ||||||
|             except AttributeError: |             except AttributeError: | ||||||
|                 file_name = file['filename'] |                 file_name = file['filename'] | ||||||
|  |  | ||||||
| @@ -857,9 +860,9 @@ class FileField(Field): | |||||||
|         return os.path.normpath(f) |         return os.path.normpath(f) | ||||||
|  |  | ||||||
|     def save_form_data(self, instance, data): |     def save_form_data(self, instance, data): | ||||||
|         from django.newforms.fields import UploadedFile |         from django.core.files.uploadedfile import UploadedFile | ||||||
|         if data and isinstance(data, UploadedFile): |         if data and isinstance(data, UploadedFile): | ||||||
|             getattr(instance, "save_%s_file" % self.name)(data.filename, data.data, save=False) |             getattr(instance, "save_%s_file" % self.name)(data.name, data, save=False) | ||||||
|  |  | ||||||
|     def formfield(self, **kwargs): |     def formfield(self, **kwargs): | ||||||
|         defaults = {'form_class': forms.FileField} |         defaults = {'form_class': forms.FileField} | ||||||
|   | |||||||
| @@ -27,7 +27,7 @@ from django.utils.encoding import StrAndUnicode, smart_unicode, smart_str | |||||||
|  |  | ||||||
| from util import ErrorList, ValidationError | from util import ErrorList, ValidationError | ||||||
| from widgets import TextInput, PasswordInput, HiddenInput, MultipleHiddenInput, FileInput, CheckboxInput, Select, NullBooleanSelect, SelectMultiple, DateTimeInput | from widgets import TextInput, PasswordInput, HiddenInput, MultipleHiddenInput, FileInput, CheckboxInput, Select, NullBooleanSelect, SelectMultiple, DateTimeInput | ||||||
|  | from django.core.files.uploadedfile import SimpleUploadedFile as UploadedFile | ||||||
|  |  | ||||||
| __all__ = ( | __all__ = ( | ||||||
|     'Field', 'CharField', 'IntegerField', |     'Field', 'CharField', 'IntegerField', | ||||||
| @@ -419,18 +419,6 @@ except ImportError: | |||||||
|     # It's OK if Django settings aren't configured. |     # It's OK if Django settings aren't configured. | ||||||
|     URL_VALIDATOR_USER_AGENT = 'Django (http://www.djangoproject.com/)' |     URL_VALIDATOR_USER_AGENT = 'Django (http://www.djangoproject.com/)' | ||||||
|  |  | ||||||
| class UploadedFile(StrAndUnicode): |  | ||||||
|     "A wrapper for files uploaded in a FileField" |  | ||||||
|     def __init__(self, filename, data): |  | ||||||
|         self.filename = filename |  | ||||||
|         self.data = data |  | ||||||
|  |  | ||||||
|     def __unicode__(self): |  | ||||||
|         """ |  | ||||||
|         The unicode representation is the filename, so that the pre-database-insertion |  | ||||||
|         logic can use UploadedFile objects |  | ||||||
|         """ |  | ||||||
|         return self.filename |  | ||||||
|  |  | ||||||
| class FileField(Field): | class FileField(Field): | ||||||
|     widget = FileInput |     widget = FileInput | ||||||
| @@ -460,23 +448,20 @@ class FileField(Field): | |||||||
|                 category = DeprecationWarning, |                 category = DeprecationWarning, | ||||||
|                 stacklevel = 2 |                 stacklevel = 2 | ||||||
|             ) |             ) | ||||||
|  |             data = UploadedFile(data['filename'], data['content']) | ||||||
|  |  | ||||||
|         try: |         try: | ||||||
|             file_name = data.file_name |             file_name = data.name | ||||||
|             file_size = data.file_size |             file_size = data.size | ||||||
|         except AttributeError: |         except AttributeError: | ||||||
|             try: |             raise ValidationError(self.error_messages['invalid']) | ||||||
|                 file_name = data.get('filename') |  | ||||||
|                 file_size = bool(data['content']) |  | ||||||
|             except (AttributeError, KeyError): |  | ||||||
|                 raise ValidationError(self.error_messages['invalid']) |  | ||||||
|  |  | ||||||
|         if not file_name: |         if not file_name: | ||||||
|             raise ValidationError(self.error_messages['invalid']) |             raise ValidationError(self.error_messages['invalid']) | ||||||
|         if not file_size: |         if not file_size: | ||||||
|             raise ValidationError(self.error_messages['empty']) |             raise ValidationError(self.error_messages['empty']) | ||||||
|  |  | ||||||
|         return UploadedFile(file_name, data) |         return data | ||||||
|  |  | ||||||
| class ImageField(FileField): | class ImageField(FileField): | ||||||
|     default_error_messages = { |     default_error_messages = { | ||||||
|   | |||||||
| @@ -686,7 +686,7 @@ class FileUploadField(FormField): | |||||||
|             if upload_errors: |             if upload_errors: | ||||||
|                 raise validators.CriticalValidationError, upload_errors |                 raise validators.CriticalValidationError, upload_errors | ||||||
|         try: |         try: | ||||||
|             file_size = new_data.file_size |             file_size = new_data.size | ||||||
|         except AttributeError: |         except AttributeError: | ||||||
|             file_size = len(new_data['content']) |             file_size = len(new_data['content']) | ||||||
|         if not file_size: |         if not file_size: | ||||||
|   | |||||||
| @@ -1331,23 +1331,12 @@ given length. | |||||||
|     * Validates that non-empty file data has been bound to the form. |     * Validates that non-empty file data has been bound to the form. | ||||||
|     * Error message keys: ``required``, ``invalid``, ``missing``, ``empty`` |     * Error message keys: ``required``, ``invalid``, ``missing``, ``empty`` | ||||||
|  |  | ||||||
| An ``UploadedFile`` object has two attributes: | To learn more about the ``UploadedFile`` object, see the `file uploads documentation`_. | ||||||
|  |  | ||||||
|     ======================  ==================================================== |  | ||||||
|     Attribute               Description |  | ||||||
|     ======================  ==================================================== |  | ||||||
|     ``filename``            The name of the file, provided by the uploading |  | ||||||
|                             client. |  | ||||||
|                              |  | ||||||
|     ``content``             The array of bytes comprising the file content. |  | ||||||
|     ======================  ==================================================== |  | ||||||
|  |  | ||||||
| The string representation of an ``UploadedFile`` is the same as the filename |  | ||||||
| attribute. |  | ||||||
|  |  | ||||||
| When you use a ``FileField`` in a form, you must also remember to | When you use a ``FileField`` in a form, you must also remember to | ||||||
| `bind the file data to the form`_. | `bind the file data to the form`_. | ||||||
|  |  | ||||||
|  | .. _file uploads documentation: ../upload_handling/ | ||||||
| .. _`bind the file data to the form`: `Binding uploaded files to a form`_ | .. _`bind the file data to the form`: `Binding uploaded files to a form`_ | ||||||
|  |  | ||||||
| ``FilePathField`` | ``FilePathField`` | ||||||
|   | |||||||
| @@ -279,7 +279,7 @@ Default: ``''`` (Empty string) | |||||||
|  |  | ||||||
| The database backend to use. The build-in database backends are | The database backend to use. The build-in database backends are | ||||||
| ``'postgresql_psycopg2'``, ``'postgresql'``, ``'mysql'``, ``'mysql_old'``, | ``'postgresql_psycopg2'``, ``'postgresql'``, ``'mysql'``, ``'mysql_old'``, | ||||||
| ``'sqlite3'``, ``'oracle'``, and ``'oracle'``. | ``'sqlite3'``, and ``'oracle'``. | ||||||
|  |  | ||||||
| In the Django development version, you can use a database backend that doesn't | In the Django development version, you can use a database backend that doesn't | ||||||
| ship with Django by setting ``DATABASE_ENGINE`` to a fully-qualified path (i.e. | ship with Django by setting ``DATABASE_ENGINE`` to a fully-qualified path (i.e. | ||||||
|   | |||||||
| @@ -22,7 +22,7 @@ Consider a simple form containing a ``FileField``:: | |||||||
|     class UploadFileForm(forms.Form): |     class UploadFileForm(forms.Form): | ||||||
|         title = forms.CharField(max_length=50) |         title = forms.CharField(max_length=50) | ||||||
|         file  = forms.FileField() |         file  = forms.FileField() | ||||||
|          |  | ||||||
| A view handling this form will receive the file data in ``request.FILES``, which | A view handling this form will receive the file data in ``request.FILES``, which | ||||||
| is a dictionary containing a key for each ``FileField`` (or ``ImageField``, or | is a dictionary containing a key for each ``FileField`` (or ``ImageField``, or | ||||||
| other ``FileField`` subclass) in the form. So the data from the above form would | other ``FileField`` subclass) in the form. So the data from the above form would | ||||||
| @@ -64,32 +64,32 @@ methods to access the uploaded content: | |||||||
|     ``UploadedFile.read()`` |     ``UploadedFile.read()`` | ||||||
|         Read the entire uploaded data from the file. Be careful with this |         Read the entire uploaded data from the file. Be careful with this | ||||||
|         method: if the uploaded file is huge it can overwhelm your system if you |         method: if the uploaded file is huge it can overwhelm your system if you | ||||||
|         try to read it into memory. You'll probably want to use ``chunk()`` |         try to read it into memory. You'll probably want to use ``chunks()`` | ||||||
|         instead; see below. |         instead; see below. | ||||||
|          |  | ||||||
|     ``UploadedFile.multiple_chunks()`` |     ``UploadedFile.multiple_chunks()`` | ||||||
|         Returns ``True`` if the uploaded file is big enough to require |         Returns ``True`` if the uploaded file is big enough to require | ||||||
|         reading in multiple chunks. By default this will be any file |         reading in multiple chunks. By default this will be any file | ||||||
|         larger than 2.5 megabytes, but that's configurable; see below. |         larger than 2.5 megabytes, but that's configurable; see below. | ||||||
|      |  | ||||||
|     ``UploadedFile.chunk()`` |     ``UploadedFile.chunk()`` | ||||||
|         A generator returning chunks of the file. If ``multiple_chunks()`` is |         A generator returning chunks of the file. If ``multiple_chunks()`` is | ||||||
|         ``True``, you should use this method in a loop instead of ``read()``. |         ``True``, you should use this method in a loop instead of ``read()``. | ||||||
|          |  | ||||||
|         In practice, it's often easiest simply to use ``chunks()`` all the time; |         In practice, it's often easiest simply to use ``chunks()`` all the time; | ||||||
|         see the example below. |         see the example below. | ||||||
|      |  | ||||||
|     ``UploadedFile.file_name`` |     ``UploadedFile.file_name`` | ||||||
|         The name of the uploaded file (e.g. ``my_file.txt``). |         The name of the uploaded file (e.g. ``my_file.txt``). | ||||||
|          |  | ||||||
|     ``UploadedFile.file_size`` |     ``UploadedFile.file_size`` | ||||||
|         The size, in bytes, of the uploaded file. |         The size, in bytes, of the uploaded file. | ||||||
|          |  | ||||||
| There are a few other methods and attributes available on ``UploadedFile`` | There are a few other methods and attributes available on ``UploadedFile`` | ||||||
| objects; see `UploadedFile objects`_ for a complete reference. | objects; see `UploadedFile objects`_ for a complete reference. | ||||||
|  |  | ||||||
| Putting it all together, here's a common way you might handle an uploaded file:: | Putting it all together, here's a common way you might handle an uploaded file:: | ||||||
|      |  | ||||||
|     def handle_uploaded_file(f): |     def handle_uploaded_file(f): | ||||||
|         destination = open('some/file/name.txt', 'wb') |         destination = open('some/file/name.txt', 'wb') | ||||||
|         for chunk in f.chunks(): |         for chunk in f.chunks(): | ||||||
| @@ -126,27 +126,27 @@ Three `settings`_ control Django's file upload behavior: | |||||||
|         The maximum size, in bytes, for files that will be uploaded |         The maximum size, in bytes, for files that will be uploaded | ||||||
|         into memory. Files larger than ``FILE_UPLOAD_MAX_MEMORY_SIZE`` |         into memory. Files larger than ``FILE_UPLOAD_MAX_MEMORY_SIZE`` | ||||||
|         will be streamed to disk. |         will be streamed to disk. | ||||||
|          |  | ||||||
|         Defaults to 2.5 megabytes. |         Defaults to 2.5 megabytes. | ||||||
|          |  | ||||||
|     ``FILE_UPLOAD_TEMP_DIR`` |     ``FILE_UPLOAD_TEMP_DIR`` | ||||||
|         The directory where uploaded files larger than ``FILE_UPLOAD_TEMP_DIR`` |         The directory where uploaded files larger than ``FILE_UPLOAD_TEMP_DIR`` | ||||||
|         will be stored. |         will be stored. | ||||||
|          |  | ||||||
|         Defaults to your system's standard temporary directory (i.e. ``/tmp`` on |         Defaults to your system's standard temporary directory (i.e. ``/tmp`` on | ||||||
|         most Unix-like systems). |         most Unix-like systems). | ||||||
|          |  | ||||||
|     ``FILE_UPLOAD_HANDLERS`` |     ``FILE_UPLOAD_HANDLERS`` | ||||||
|         The actual handlers for uploaded files. Changing this setting |         The actual handlers for uploaded files. Changing this setting | ||||||
|         allows complete customization -- even replacement -- of |         allows complete customization -- even replacement -- of | ||||||
|         Django's upload process. See `upload handlers`_, below, |         Django's upload process. See `upload handlers`_, below, | ||||||
|         for details. |         for details. | ||||||
|          |  | ||||||
|         Defaults to:: |         Defaults to:: | ||||||
|          |  | ||||||
|             ("django.core.files.uploadhandler.MemoryFileUploadHandler", |             ("django.core.files.uploadhandler.MemoryFileUploadHandler", | ||||||
|              "django.core.files.uploadhandler.TemporaryFileUploadHandler",) |              "django.core.files.uploadhandler.TemporaryFileUploadHandler",) | ||||||
|              |  | ||||||
|         Which means "try to upload to memory first, then fall back to temporary |         Which means "try to upload to memory first, then fall back to temporary | ||||||
|         files." |         files." | ||||||
|  |  | ||||||
| @@ -161,35 +161,39 @@ All ``UploadedFile`` objects define the following methods/attributes: | |||||||
|         Returns a byte string of length ``num_bytes``, or the complete file if |         Returns a byte string of length ``num_bytes``, or the complete file if | ||||||
|         ``num_bytes`` is ``None``. |         ``num_bytes`` is ``None``. | ||||||
|  |  | ||||||
|     ``UploadedFile.chunk(self, chunk_size=None)`` |     ``UploadedFile.chunks(self, chunk_size=None)`` | ||||||
|         A generator yielding small chunks from the file. If ``chunk_size`` isn't |         A generator yielding small chunks from the file. If ``chunk_size`` isn't | ||||||
|         given, chunks will be 64 kb. |         given, chunks will be 64 KB. | ||||||
|  |  | ||||||
|     ``UploadedFile.multiple_chunks(self, chunk_size=None)`` |     ``UploadedFile.multiple_chunks(self, chunk_size=None)`` | ||||||
|         Returns ``True`` if you can expect more than one chunk when calling |         Returns ``True`` if you can expect more than one chunk when calling | ||||||
|         ``UploadedFile.chunk(self, chunk_size)``. |         ``UploadedFile.chunks(self, chunk_size)``. | ||||||
|  |  | ||||||
|     ``UploadedFile.file_size`` |     ``UploadedFile.file_size`` | ||||||
|         The size, in bytes, of the uploaded file. |         The size, in bytes, of the uploaded file. | ||||||
|      |  | ||||||
|     ``UploadedFile.file_name`` |     ``UploadedFile.file_name`` | ||||||
|         The name of the uploaded file as provided by the user. |         The name of the uploaded file as provided by the user. | ||||||
|      |  | ||||||
|     ``UploadedFile.content_type`` |     ``UploadedFile.content_type`` | ||||||
|         The content-type header uploaded with the file (e.g. ``text/plain`` or |         The content-type header uploaded with the file (e.g. ``text/plain`` or | ||||||
|         ``application/pdf``). Like any data supplied by the user, you shouldn't |         ``application/pdf``). Like any data supplied by the user, you shouldn't | ||||||
|         trust that the uploaded file is actually this type. You'll still need to |         trust that the uploaded file is actually this type. You'll still need to | ||||||
|         validate that the file contains the content that the content-type header |         validate that the file contains the content that the content-type header | ||||||
|         claims -- "trust but verify." |         claims -- "trust but verify." | ||||||
|      |  | ||||||
|     ``UploadedFile.charset`` |     ``UploadedFile.charset`` | ||||||
|         For ``text/*`` content-types, the character set (i.e. ``utf8``) supplied |         For ``text/*`` content-types, the character set (i.e. ``utf8``) supplied | ||||||
|         by the browser. Again, "trust but verify" is the best policy here. |         by the browser. Again, "trust but verify" is the best policy here. | ||||||
|  |  | ||||||
|  |     ``UploadedFile.__iter__()`` | ||||||
|  |         Iterates over the lines in the file. | ||||||
|  |  | ||||||
|     ``UploadedFile.temporary_file_path()`` |     ``UploadedFile.temporary_file_path()`` | ||||||
|         Only files uploaded onto disk will have this method; it returns the full |         Only files uploaded onto disk will have this method; it returns the full | ||||||
|         path to the temporary uploaded file. |         path to the temporary uploaded file. | ||||||
|  |  | ||||||
|  |  | ||||||
| Upload Handlers | Upload Handlers | ||||||
| =============== | =============== | ||||||
|  |  | ||||||
|   | |||||||
| @@ -803,7 +803,7 @@ False | |||||||
| >>> f.is_valid() | >>> f.is_valid() | ||||||
| True | True | ||||||
| >>> type(f.cleaned_data['file']) | >>> type(f.cleaned_data['file']) | ||||||
| <class 'django.newforms.fields.UploadedFile'> | <class 'django.core.files.uploadedfile.SimpleUploadedFile'> | ||||||
| >>> instance = f.save() | >>> instance = f.save() | ||||||
| >>> instance.file | >>> instance.file | ||||||
| u'...test1.txt' | u'...test1.txt' | ||||||
| @@ -814,7 +814,7 @@ u'...test1.txt' | |||||||
| >>> f.is_valid() | >>> f.is_valid() | ||||||
| True | True | ||||||
| >>> type(f.cleaned_data['file']) | >>> type(f.cleaned_data['file']) | ||||||
| <class 'django.newforms.fields.UploadedFile'> | <class 'django.core.files.uploadedfile.SimpleUploadedFile'> | ||||||
| >>> instance = f.save() | >>> instance = f.save() | ||||||
| >>> instance.file | >>> instance.file | ||||||
| u'...test1.txt' | u'...test1.txt' | ||||||
| @@ -906,7 +906,7 @@ u'...test3.txt' | |||||||
| >>> f.is_valid() | >>> f.is_valid() | ||||||
| True | True | ||||||
| >>> type(f.cleaned_data['image']) | >>> type(f.cleaned_data['image']) | ||||||
| <class 'django.newforms.fields.UploadedFile'> | <class 'django.core.files.uploadedfile.SimpleUploadedFile'> | ||||||
| >>> instance = f.save() | >>> instance = f.save() | ||||||
| >>> instance.image | >>> instance.image | ||||||
| u'...test.png' | u'...test.png' | ||||||
| @@ -918,7 +918,7 @@ u'...test.png' | |||||||
| >>> f.is_valid() | >>> f.is_valid() | ||||||
| True | True | ||||||
| >>> type(f.cleaned_data['image']) | >>> type(f.cleaned_data['image']) | ||||||
| <class 'django.newforms.fields.UploadedFile'> | <class 'django.core.files.uploadedfile.SimpleUploadedFile'> | ||||||
| >>> instance = f.save() | >>> instance = f.save() | ||||||
| >>> instance.image | >>> instance.image | ||||||
| u'...test.png' | u'...test.png' | ||||||
|   | |||||||
| @@ -15,7 +15,7 @@ def file_upload_view(request): | |||||||
|     if isinstance(form_data.get('file_field'), UploadedFile) and isinstance(form_data['name'], unicode): |     if isinstance(form_data.get('file_field'), UploadedFile) and isinstance(form_data['name'], unicode): | ||||||
|         # If a file is posted, the dummy client should only post the file name, |         # If a file is posted, the dummy client should only post the file name, | ||||||
|         # not the full path. |         # not the full path. | ||||||
|         if os.path.dirname(form_data['file_field'].file_name) != '': |         if os.path.dirname(form_data['file_field'].name) != '': | ||||||
|             return HttpResponseServerError()             |             return HttpResponseServerError()             | ||||||
|         return HttpResponse('') |         return HttpResponse('') | ||||||
|     else: |     else: | ||||||
| @@ -29,7 +29,7 @@ def file_upload_view_verify(request): | |||||||
|     form_data.update(request.FILES) |     form_data.update(request.FILES) | ||||||
|  |  | ||||||
|     # Check to see if unicode names worked out. |     # Check to see if unicode names worked out. | ||||||
|     if not request.FILES['file_unicode'].file_name.endswith(u'test_\u4e2d\u6587_Orl\xe9ans.jpg'): |     if not request.FILES['file_unicode'].name.endswith(u'test_\u4e2d\u6587_Orl\xe9ans.jpg'): | ||||||
|         return HttpResponseServerError() |         return HttpResponseServerError() | ||||||
|  |  | ||||||
|     for key, value in form_data.items(): |     for key, value in form_data.items(): | ||||||
| @@ -51,7 +51,7 @@ def file_upload_echo(request): | |||||||
|     """ |     """ | ||||||
|     Simple view to echo back info about uploaded files for tests. |     Simple view to echo back info about uploaded files for tests. | ||||||
|     """ |     """ | ||||||
|     r = dict([(k, f.file_name) for k, f in request.FILES.items()]) |     r = dict([(k, f.name) for k, f in request.FILES.items()]) | ||||||
|     return HttpResponse(simplejson.dumps(r)) |     return HttpResponse(simplejson.dumps(r)) | ||||||
|      |      | ||||||
| def file_upload_quota(request): | def file_upload_quota(request): | ||||||
|   | |||||||
| @@ -800,10 +800,10 @@ Traceback (most recent call last): | |||||||
| ValidationError: [u'The submitted file is empty.'] | ValidationError: [u'The submitted file is empty.'] | ||||||
|  |  | ||||||
| >>> type(f.clean(SimpleUploadedFile('name', 'Some File Content'))) | >>> type(f.clean(SimpleUploadedFile('name', 'Some File Content'))) | ||||||
| <class 'django.newforms.fields.UploadedFile'> | <class 'django.core.files.uploadedfile.SimpleUploadedFile'> | ||||||
|  |  | ||||||
| >>> type(f.clean(SimpleUploadedFile('name', 'Some File Content'), 'files/test4.pdf')) | >>> type(f.clean(SimpleUploadedFile('name', 'Some File Content'), 'files/test4.pdf')) | ||||||
| <class 'django.newforms.fields.UploadedFile'> | <class 'django.core.files.uploadedfile.SimpleUploadedFile'> | ||||||
|  |  | ||||||
| # URLField ################################################################## | # URLField ################################################################## | ||||||
|  |  | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user