mirror of
				https://github.com/django/django.git
				synced 2025-10-26 07:06:08 +00:00 
			
		
		
		
	Fixed #7712, #9404, #10249, #10300: a light refactor and cleanup of file storage and the File object. Thanks to Armin Ronacher and Alex Gaynor.
				
					
				
			git-svn-id: http://code.djangoproject.com/svn/django/trunk@10717 bcc190cf-cafb-0310-a4f2-bffc1f526a37
This commit is contained in:
		| @@ -1,20 +1,23 @@ | |||||||
| import os | import os | ||||||
|  |  | ||||||
| from django.utils.encoding import smart_str, smart_unicode |  | ||||||
|  |  | ||||||
| try: | try: | ||||||
|     from cStringIO import StringIO |     from cStringIO import StringIO | ||||||
| except ImportError: | except ImportError: | ||||||
|     from StringIO import StringIO |     from StringIO import StringIO | ||||||
|  |  | ||||||
| class File(object): | from django.utils.encoding import smart_str, smart_unicode | ||||||
|  | from django.core.files.utils import FileProxyMixin | ||||||
|  |  | ||||||
|  | class File(FileProxyMixin): | ||||||
|     DEFAULT_CHUNK_SIZE = 64 * 2**10 |     DEFAULT_CHUNK_SIZE = 64 * 2**10 | ||||||
|  |  | ||||||
|     def __init__(self, file): |     def __init__(self, file, name=None): | ||||||
|         self.file = file |         self.file = file | ||||||
|         self._name = file.name |         if name is None: | ||||||
|         self._mode = file.mode |             name = getattr(file, 'name', None) | ||||||
|         self._closed = False |         self.name = name | ||||||
|  |         self.mode = getattr(file, 'mode', None) | ||||||
|  |         self.closed = False | ||||||
|  |  | ||||||
|     def __str__(self): |     def __str__(self): | ||||||
|         return smart_str(self.name or '') |         return smart_str(self.name or '') | ||||||
| @@ -26,25 +29,11 @@ class File(object): | |||||||
|         return "<%s: %s>" % (self.__class__.__name__, self or "None") |         return "<%s: %s>" % (self.__class__.__name__, self or "None") | ||||||
|  |  | ||||||
|     def __nonzero__(self): |     def __nonzero__(self): | ||||||
|         return not not self.name |         return bool(self.name) | ||||||
|  |  | ||||||
|     def __len__(self): |     def __len__(self): | ||||||
|         return self.size |         return self.size | ||||||
|  |  | ||||||
|     def _get_name(self): |  | ||||||
|         if not hasattr(self, '_name'): |  | ||||||
|             raise ValueError("This operation requires the file to have a name.") |  | ||||||
|         return self._name |  | ||||||
|     name = property(_get_name) |  | ||||||
|  |  | ||||||
|     def _get_mode(self): |  | ||||||
|         return self._mode |  | ||||||
|     mode = property(_get_mode) |  | ||||||
|  |  | ||||||
|     def _get_closed(self): |  | ||||||
|         return self._closed |  | ||||||
|     closed = property(_get_closed) |  | ||||||
|  |  | ||||||
|     def _get_size(self): |     def _get_size(self): | ||||||
|         if not hasattr(self, '_size'): |         if not hasattr(self, '_size'): | ||||||
|             if hasattr(self.file, 'size'): |             if hasattr(self.file, 'size'): | ||||||
| @@ -66,7 +55,7 @@ class File(object): | |||||||
|         ``UploadedFile.DEFAULT_CHUNK_SIZE``). |         ``UploadedFile.DEFAULT_CHUNK_SIZE``). | ||||||
|         """ |         """ | ||||||
|         if not chunk_size: |         if not chunk_size: | ||||||
|             chunk_size = self.__class__.DEFAULT_CHUNK_SIZE |             chunk_size = self.DEFAULT_CHUNK_SIZE | ||||||
|  |  | ||||||
|         if hasattr(self, 'seek'): |         if hasattr(self, 'seek'): | ||||||
|             self.seek(0) |             self.seek(0) | ||||||
| @@ -89,12 +78,6 @@ class File(object): | |||||||
|             chunk_size = self.DEFAULT_CHUNK_SIZE |             chunk_size = self.DEFAULT_CHUNK_SIZE | ||||||
|         return self.size > chunk_size |         return self.size > chunk_size | ||||||
|  |  | ||||||
|     def xreadlines(self): |  | ||||||
|         return iter(self) |  | ||||||
|  |  | ||||||
|     def readlines(self): |  | ||||||
|         return list(self.xreadlines()) |  | ||||||
|  |  | ||||||
|     def __iter__(self): |     def __iter__(self): | ||||||
|         # Iterate over this file-like object by newlines |         # Iterate over this file-like object by newlines | ||||||
|         buffer_ = None |         buffer_ = None | ||||||
| @@ -121,43 +104,22 @@ class File(object): | |||||||
|             self.seek(0) |             self.seek(0) | ||||||
|         elif os.path.exists(self.file.name): |         elif os.path.exists(self.file.name): | ||||||
|             self.file = open(self.file.name, mode or self.file.mode) |             self.file = open(self.file.name, mode or self.file.mode) | ||||||
|  |             self.closed = False | ||||||
|         else: |         else: | ||||||
|             raise ValueError("The file cannot be reopened.") |             raise ValueError("The file cannot be reopened.") | ||||||
|  |  | ||||||
|     def seek(self, position): |  | ||||||
|         self.file.seek(position) |  | ||||||
|  |  | ||||||
|     def tell(self): |  | ||||||
|         return self.file.tell() |  | ||||||
|  |  | ||||||
|     def read(self, num_bytes=None): |  | ||||||
|         if num_bytes is None: |  | ||||||
|             return self.file.read() |  | ||||||
|         return self.file.read(num_bytes) |  | ||||||
|  |  | ||||||
|     def write(self, content): |  | ||||||
|         if not self.mode.startswith('w'): |  | ||||||
|             raise IOError("File was not opened with write access.") |  | ||||||
|         self.file.write(content) |  | ||||||
|  |  | ||||||
|     def flush(self): |  | ||||||
|         if not self.mode.startswith('w'): |  | ||||||
|             raise IOError("File was not opened with write access.") |  | ||||||
|         self.file.flush() |  | ||||||
|  |  | ||||||
|     def close(self): |     def close(self): | ||||||
|         self.file.close() |         self.file.close() | ||||||
|         self._closed = True |         self.closed = True | ||||||
|  |  | ||||||
| class ContentFile(File): | class ContentFile(File): | ||||||
|     """ |     """ | ||||||
|     A File-like object that takes just raw content, rather than an actual file. |     A File-like object that takes just raw content, rather than an actual file. | ||||||
|     """ |     """ | ||||||
|     def __init__(self, content): |     def __init__(self, content): | ||||||
|         self.file = StringIO(content or '') |         content = content or '' | ||||||
|         self.size = len(content or '') |         super(ContentFile, self).__init__(StringIO(content)) | ||||||
|         self.file.seek(0) |         self.size = len(content) | ||||||
|         self._closed = False |  | ||||||
|  |  | ||||||
|     def __str__(self): |     def __str__(self): | ||||||
|         return 'Raw content' |         return 'Raw content' | ||||||
| @@ -166,6 +128,6 @@ class ContentFile(File): | |||||||
|         return True |         return True | ||||||
|  |  | ||||||
|     def open(self, mode=None): |     def open(self, mode=None): | ||||||
|         if self._closed: |         if self.closed: | ||||||
|             self._closed = False |             self.closed = False | ||||||
|         self.seek(0) |         self.seek(0) | ||||||
|   | |||||||
| @@ -11,11 +11,12 @@ NamedTemporaryFile uses the O_TEMPORARY flag, and thus cannot be reopened [1]. | |||||||
|  |  | ||||||
| import os | import os | ||||||
| import tempfile | import tempfile | ||||||
|  | from django.core.files.utils import FileProxyMixin | ||||||
|  |  | ||||||
| __all__ = ('NamedTemporaryFile', 'gettempdir',) | __all__ = ('NamedTemporaryFile', 'gettempdir',) | ||||||
|  |  | ||||||
| if os.name == 'nt': | if os.name == 'nt': | ||||||
|     class TemporaryFile(object): |     class TemporaryFile(FileProxyMixin): | ||||||
|         """ |         """ | ||||||
|         Temporary file object constructor that works in Windows and supports |         Temporary file object constructor that works in Windows and supports | ||||||
|         reopening of the temporary file in windows. |         reopening of the temporary file in windows. | ||||||
| @@ -48,12 +49,6 @@ if os.name == 'nt': | |||||||
|         def __del__(self): |         def __del__(self): | ||||||
|             self.close() |             self.close() | ||||||
|  |  | ||||||
|         # Proxy to the file object. |  | ||||||
|         def __getattr__(self, name): |  | ||||||
|             return getattr(self.file, name) |  | ||||||
|         def __iter__(self): |  | ||||||
|             return iter(self.file) |  | ||||||
|  |  | ||||||
|     NamedTemporaryFile = TemporaryFile |     NamedTemporaryFile = TemporaryFile | ||||||
| else: | else: | ||||||
|     NamedTemporaryFile = tempfile.NamedTemporaryFile |     NamedTemporaryFile = tempfile.NamedTemporaryFile | ||||||
|   | |||||||
| @@ -26,14 +26,15 @@ class UploadedFile(File): | |||||||
|     """ |     """ | ||||||
|     DEFAULT_CHUNK_SIZE = 64 * 2**10 |     DEFAULT_CHUNK_SIZE = 64 * 2**10 | ||||||
|  |  | ||||||
|     def __init__(self, name=None, content_type=None, size=None, charset=None): |     def __init__(self, file=None, name=None, content_type=None, size=None, charset=None): | ||||||
|         self.name = name |         super(UploadedFile, self).__init__(file, name) | ||||||
|         self.size = 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__, smart_str(self.name), self.content_type) |         return "<%s: %s (%s)>" % ( | ||||||
|  |             self.__class__.__name__, smart_str(self.name), self.content_type) | ||||||
|  |  | ||||||
|     def _get_name(self): |     def _get_name(self): | ||||||
|         return self._name |         return self._name | ||||||
| @@ -53,95 +54,66 @@ class UploadedFile(File): | |||||||
|  |  | ||||||
|     name = property(_get_name, _set_name) |     name = property(_get_name, _set_name) | ||||||
|  |  | ||||||
|     # Abstract methods; subclasses *must* define read() and probably should |  | ||||||
|     # define open/close. |  | ||||||
|     def read(self, num_bytes=None): |  | ||||||
|         raise NotImplementedError() |  | ||||||
|  |  | ||||||
|     def open(self): |  | ||||||
|         pass |  | ||||||
|  |  | ||||||
|     def close(self): |  | ||||||
|         pass |  | ||||||
|  |  | ||||||
| class TemporaryUploadedFile(UploadedFile): | 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, name, content_type, size, charset): | ||||||
|         super(TemporaryUploadedFile, self).__init__(name, content_type, size, charset) |  | ||||||
|         if settings.FILE_UPLOAD_TEMP_DIR: |         if settings.FILE_UPLOAD_TEMP_DIR: | ||||||
|             self._file = tempfile.NamedTemporaryFile(suffix='.upload', dir=settings.FILE_UPLOAD_TEMP_DIR) |             file = tempfile.NamedTemporaryFile(suffix='.upload', | ||||||
|  |                 dir=settings.FILE_UPLOAD_TEMP_DIR) | ||||||
|         else: |         else: | ||||||
|             self._file = tempfile.NamedTemporaryFile(suffix='.upload') |             file = tempfile.NamedTemporaryFile(suffix='.upload') | ||||||
|  |         super(TemporaryUploadedFile, self).__init__(file, name, content_type, size, charset) | ||||||
|  |  | ||||||
|     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._file.name |         return self.file.name | ||||||
|  |  | ||||||
|     # Most methods on this object get proxied to NamedTemporaryFile. |  | ||||||
|     # We can't directly subclass because NamedTemporaryFile is actually a |  | ||||||
|     # factory function |  | ||||||
|     def read(self, *args):          return self._file.read(*args) |  | ||||||
|     def seek(self, *args):          return self._file.seek(*args) |  | ||||||
|     def write(self, s):             return self._file.write(s) |  | ||||||
|     def tell(self, *args):          return self._file.tell(*args) |  | ||||||
|     def __iter__(self):             return iter(self._file) |  | ||||||
|     def readlines(self, size=None): return self._file.readlines(size) |  | ||||||
|     def xreadlines(self):           return self._file.xreadlines() |  | ||||||
|     def close(self): |     def close(self): | ||||||
|         try: |         try: | ||||||
|             return self._file.close() |             try: | ||||||
|         except OSError, e: |                 return self.file.close() | ||||||
|             if e.errno == 2: |             except OSError, e: | ||||||
|                 # Means the file was moved or deleted before the tempfile could unlink it. |                 if e.errno != 2: | ||||||
|                 # Still sets self._file.close_called and calls self._file.file.close() |                     # Means the file was moved or deleted before the tempfile | ||||||
|                 # before the exception |                     # could unlink it.  Still sets self.file.close_called and | ||||||
|                 return |                     # calls self.file.file.close() before the exception | ||||||
|             else: |                     raise | ||||||
|                 raise e |         finally: | ||||||
|  |             self.closed = True | ||||||
|  |  | ||||||
| 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, name, content_type, size, charset): |     def __init__(self, file, field_name, name, content_type, size, charset): | ||||||
|         super(InMemoryUploadedFile, self).__init__(name, content_type, size, charset) |         super(InMemoryUploadedFile, self).__init__(file, name, content_type, size, charset) | ||||||
|         self._file = file |  | ||||||
|         self.field_name = field_name |         self.field_name = field_name | ||||||
|         self._file.seek(0) |  | ||||||
|  |  | ||||||
|     def open(self): |     def open(self): | ||||||
|         self._file.seek(0) |         self.closed = False | ||||||
|  |         self.file.seek(0) | ||||||
|  |  | ||||||
|     def chunks(self, chunk_size=None): |     def chunks(self, chunk_size=None): | ||||||
|         self._file.seek(0) |         self.file.seek(0) | ||||||
|         yield self.read() |         yield self.read() | ||||||
|  |  | ||||||
|     def multiple_chunks(self, chunk_size=None): |     def multiple_chunks(self, chunk_size=None): | ||||||
|         # Since it's in memory, we'll never have multiple chunks. |         # Since it's in memory, we'll never have multiple chunks. | ||||||
|         return False |         return False | ||||||
|  |  | ||||||
|     # proxy methods to StringIO |  | ||||||
|     def read(self, *args): return self._file.read(*args) |  | ||||||
|     def seek(self, *args): return self._file.seek(*args) |  | ||||||
|     def tell(self, *args): return self._file.tell(*args) |  | ||||||
|     def close(self):       return self._file.close() |  | ||||||
|  |  | ||||||
| class SimpleUploadedFile(InMemoryUploadedFile): | class SimpleUploadedFile(InMemoryUploadedFile): | ||||||
|     """ |     """ | ||||||
|     A simple representation of a file, which just has content, size, and a name. |     A simple representation of a file, which just has content, size, and a name. | ||||||
|     """ |     """ | ||||||
|     def __init__(self, name, content, content_type='text/plain'): |     def __init__(self, name, content, content_type='text/plain'): | ||||||
|         self._file = StringIO(content or '') |         content = content or '' | ||||||
|         self.name = name |         super(SimpleUploadedFile, self).__init__(StringIO(content), None, name, | ||||||
|         self.field_name = None |                                                  content_type, len(content), None) | ||||||
|         self.size = len(content or '') |  | ||||||
|         self.content_type = content_type |  | ||||||
|         self.charset = None |  | ||||||
|         self._file.seek(0) |  | ||||||
|  |  | ||||||
|     def from_dict(cls, file_dict): |     def from_dict(cls, file_dict): | ||||||
|         """ |         """ | ||||||
| @@ -154,5 +126,4 @@ class SimpleUploadedFile(InMemoryUploadedFile): | |||||||
|         return cls(file_dict['filename'], |         return cls(file_dict['filename'], | ||||||
|                    file_dict['content'], |                    file_dict['content'], | ||||||
|                    file_dict.get('content-type', 'text/plain')) |                    file_dict.get('content-type', 'text/plain')) | ||||||
|  |  | ||||||
|     from_dict = classmethod(from_dict) |     from_dict = classmethod(from_dict) | ||||||
|   | |||||||
							
								
								
									
										29
									
								
								django/core/files/utils.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										29
									
								
								django/core/files/utils.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,29 @@ | |||||||
|  | class FileProxyMixin(object): | ||||||
|  |     """ | ||||||
|  |     A mixin class used to forward file methods to an underlaying file | ||||||
|  |     object.  The internal file object has to be called "file":: | ||||||
|  |  | ||||||
|  |         class FileProxy(FileProxyMixin): | ||||||
|  |             def __init__(self, file): | ||||||
|  |                 self.file = file | ||||||
|  |     """ | ||||||
|  |  | ||||||
|  |     encoding = property(lambda self: self.file.encoding) | ||||||
|  |     fileno = property(lambda self: self.file.fileno) | ||||||
|  |     flush = property(lambda self: self.file.flush) | ||||||
|  |     isatty = property(lambda self: self.file.isatty) | ||||||
|  |     newlines = property(lambda self: self.file.newlines) | ||||||
|  |     read = property(lambda self: self.file.read) | ||||||
|  |     readinto = property(lambda self: self.file.readinto) | ||||||
|  |     readline = property(lambda self: self.file.readline) | ||||||
|  |     readlines = property(lambda self: self.file.readlines) | ||||||
|  |     seek = property(lambda self: self.file.seek) | ||||||
|  |     softspace = property(lambda self: self.file.softspace) | ||||||
|  |     tell = property(lambda self: self.file.tell) | ||||||
|  |     truncate = property(lambda self: self.file.truncate) | ||||||
|  |     write = property(lambda self: self.file.write) | ||||||
|  |     writelines = property(lambda self: self.file.writelines) | ||||||
|  |     xreadlines = property(lambda self: self.file.xreadlines) | ||||||
|  |  | ||||||
|  |     def __iter__(self): | ||||||
|  |         return iter(self.file) | ||||||
| @@ -17,11 +17,10 @@ from django.db.models.loading import cache | |||||||
|  |  | ||||||
| class FieldFile(File): | class FieldFile(File): | ||||||
|     def __init__(self, instance, field, name): |     def __init__(self, instance, field, name): | ||||||
|  |         super(FieldFile, self).__init__(None, name) | ||||||
|         self.instance = instance |         self.instance = instance | ||||||
|         self.field = field |         self.field = field | ||||||
|         self.storage = field.storage |         self.storage = field.storage | ||||||
|         self._name = name or u'' |  | ||||||
|         self._closed = False |  | ||||||
|         self._committed = True |         self._committed = True | ||||||
|  |  | ||||||
|     def __eq__(self, other): |     def __eq__(self, other): | ||||||
| @@ -48,10 +47,17 @@ class FieldFile(File): | |||||||
|  |  | ||||||
|     def _get_file(self): |     def _get_file(self): | ||||||
|         self._require_file() |         self._require_file() | ||||||
|         if not hasattr(self, '_file'): |         if not hasattr(self, '_file') or self._file is None: | ||||||
|             self._file = self.storage.open(self.name, 'rb') |             self._file = self.storage.open(self.name, 'rb') | ||||||
|         return self._file |         return self._file | ||||||
|     file = property(_get_file) |  | ||||||
|  |     def _set_file(self, file): | ||||||
|  |         self._file = file | ||||||
|  |  | ||||||
|  |     def _del_file(self): | ||||||
|  |         del self._file | ||||||
|  |  | ||||||
|  |     file = property(_get_file, _set_file, _del_file) | ||||||
|  |  | ||||||
|     def _get_path(self): |     def _get_path(self): | ||||||
|         self._require_file() |         self._require_file() | ||||||
| @@ -65,6 +71,8 @@ class FieldFile(File): | |||||||
|  |  | ||||||
|     def _get_size(self): |     def _get_size(self): | ||||||
|         self._require_file() |         self._require_file() | ||||||
|  |         if not self._committed: | ||||||
|  |             return len(self.file) | ||||||
|         return self.storage.size(self.name) |         return self.storage.size(self.name) | ||||||
|     size = property(_get_size) |     size = property(_get_size) | ||||||
|  |  | ||||||
| @@ -80,7 +88,7 @@ class FieldFile(File): | |||||||
|  |  | ||||||
|     def save(self, name, content, save=True): |     def save(self, name, content, save=True): | ||||||
|         name = self.field.generate_filename(self.instance, name) |         name = self.field.generate_filename(self.instance, name) | ||||||
|         self._name = self.storage.save(name, content) |         self.name = self.storage.save(name, content) | ||||||
|         setattr(self.instance, self.field.name, self.name) |         setattr(self.instance, self.field.name, self.name) | ||||||
|  |  | ||||||
|         # Update the filesize cache |         # Update the filesize cache | ||||||
| @@ -97,11 +105,11 @@ class FieldFile(File): | |||||||
|         # presence of self._file |         # presence of self._file | ||||||
|         if hasattr(self, '_file'): |         if hasattr(self, '_file'): | ||||||
|             self.close() |             self.close() | ||||||
|             del self._file |             del self.file | ||||||
|  |  | ||||||
|         self.storage.delete(self.name) |         self.storage.delete(self.name) | ||||||
|  |  | ||||||
|         self._name = None |         self.name = None | ||||||
|         setattr(self.instance, self.field.name, self.name) |         setattr(self.instance, self.field.name, self.name) | ||||||
|  |  | ||||||
|         # Delete the filesize cache |         # Delete the filesize cache | ||||||
| @@ -113,12 +121,18 @@ class FieldFile(File): | |||||||
|             self.instance.save() |             self.instance.save() | ||||||
|     delete.alters_data = True |     delete.alters_data = True | ||||||
|  |  | ||||||
|  |     def close(self): | ||||||
|  |         file = getattr(self, '_file', None) | ||||||
|  |         if file is not None: | ||||||
|  |             file.close() | ||||||
|  |             self.closed = True | ||||||
|  |  | ||||||
|     def __getstate__(self): |     def __getstate__(self): | ||||||
|         # FieldFile needs access to its associated model field and an instance |         # FieldFile needs access to its associated model field and an instance | ||||||
|         # it's attached to in order to work properly, but the only necessary |         # it's attached to in order to work properly, but the only necessary | ||||||
|         # data to be pickled is the file's name itself. Everything else will |         # data to be pickled is the file's name itself. Everything else will | ||||||
|         # be restored later, by FileDescriptor below. |         # be restored later, by FileDescriptor below. | ||||||
|         return {'_name': self.name, '_closed': False, '_committed': True} |         return {'name': self.name, 'closed': False, '_committed': True, '_file': None} | ||||||
|  |  | ||||||
| class FileDescriptor(object): | class FileDescriptor(object): | ||||||
|     def __init__(self, field): |     def __init__(self, field): | ||||||
| @@ -134,12 +148,8 @@ class FileDescriptor(object): | |||||||
|         elif isinstance(file, File) and not isinstance(file, FieldFile): |         elif isinstance(file, File) and not isinstance(file, FieldFile): | ||||||
|             # Other types of files may be assigned as well, but they need to |             # Other types of files may be assigned as well, but they need to | ||||||
|             # have the FieldFile interface added to them |             # have the FieldFile interface added to them | ||||||
|             file_copy = copy.copy(file) |             file_copy = self.field.attr_class(instance, self.field, file.name) | ||||||
|             file_copy.__class__ = type(file.__class__.__name__,  |             file_copy.file = file | ||||||
|                                        (file.__class__, self.field.attr_class), {}) |  | ||||||
|             file_copy.instance = instance |  | ||||||
|             file_copy.field = self.field |  | ||||||
|             file_copy.storage = self.field.storage |  | ||||||
|             file_copy._committed = False |             file_copy._committed = False | ||||||
|             instance.__dict__[self.field.name] = file_copy |             instance.__dict__[self.field.name] = file_copy | ||||||
|         elif isinstance(file, FieldFile) and not hasattr(file, 'field'): |         elif isinstance(file, FieldFile) and not hasattr(file, 'field'): | ||||||
|   | |||||||
| @@ -6,6 +6,7 @@ and where files should be stored. | |||||||
| """ | """ | ||||||
|  |  | ||||||
| import shutil | import shutil | ||||||
|  | import random | ||||||
| import tempfile | import tempfile | ||||||
| from django.db import models | from django.db import models | ||||||
| from django.core.files.base import ContentFile | from django.core.files.base import ContentFile | ||||||
| @@ -26,7 +27,6 @@ class Storage(models.Model): | |||||||
|     def random_upload_to(self, filename): |     def random_upload_to(self, filename): | ||||||
|         # This returns a different result each time, |         # This returns a different result each time, | ||||||
|         # to make sure it only gets called once. |         # to make sure it only gets called once. | ||||||
|         import random |  | ||||||
|         return '%s/%s' % (random.randint(100, 999), filename) |         return '%s/%s' % (random.randint(100, 999), filename) | ||||||
|  |  | ||||||
|     normal = models.FileField(storage=temp_storage, upload_to='tests') |     normal = models.FileField(storage=temp_storage, upload_to='tests') | ||||||
|   | |||||||
| @@ -29,7 +29,7 @@ if Image: | |||||||
|         mug_width = models.PositiveSmallIntegerField() |         mug_width = models.PositiveSmallIntegerField() | ||||||
|  |  | ||||||
|     __test__ = {'API_TESTS': """ |     __test__ = {'API_TESTS': """ | ||||||
|  | >>> from django.core.files import File | ||||||
| >>> image_data = open(os.path.join(os.path.dirname(__file__), "test.png"), 'rb').read() | >>> image_data = open(os.path.join(os.path.dirname(__file__), "test.png"), 'rb').read() | ||||||
| >>> p = Person(name="Joe") | >>> p = Person(name="Joe") | ||||||
| >>> p.mugshot.save("mug", ContentFile(image_data)) | >>> p.mugshot.save("mug", ContentFile(image_data)) | ||||||
| @@ -76,14 +76,15 @@ True | |||||||
| # It won't have an opened file. This is a bit brittle since it depends on the | # It won't have an opened file. This is a bit brittle since it depends on the | ||||||
| # the internals of FieldFile, but there's no other way of telling if the | # the internals of FieldFile, but there's no other way of telling if the | ||||||
| # file's been opened or not. | # file's been opened or not. | ||||||
| >>> hasattr(p3.mugshot, '_file') | >>> p3.mugshot._file is not None | ||||||
| False | False | ||||||
|  |  | ||||||
| # After asking for the size, the file should still be closed. | # After asking for the size, the file should still be closed. | ||||||
| >>> _ = p3.mugshot.size | >>> _ = p3.mugshot.size | ||||||
| >>> hasattr(p3.mugshot, '_file') | >>> p3.mugshot._file is not None | ||||||
| False | False | ||||||
|  |  | ||||||
|  | >>> p = Person.objects.create(name="Bob", mugshot=File(p3.mugshot.file)) | ||||||
|  |  | ||||||
| >>> shutil.rmtree(temp_storage_dir) | >>> shutil.rmtree(temp_storage_dir) | ||||||
| """} | """} | ||||||
|  |  | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user