mirror of
				https://github.com/django/django.git
				synced 2025-10-24 22:26:08 +00:00 
			
		
		
		
	Fixed #4948, a race condition in file saving. Thanks to Martin von Löwis, who diagnosed the problem and pointed the way to a fix.
git-svn-id: http://code.djangoproject.com/svn/django/trunk@8306 bcc190cf-cafb-0310-a4f2-bffc1f526a37
This commit is contained in:
		| @@ -40,20 +40,24 @@ try: | |||||||
| except (ImportError, AttributeError): | except (ImportError, AttributeError): | ||||||
|     pass |     pass | ||||||
|  |  | ||||||
|  | def fd(f): | ||||||
|  |     """Get a filedescriptor from something which could be a file or an fd.""" | ||||||
|  |     return hasattr(f, 'fileno') and f.fileno() or f | ||||||
|  |  | ||||||
| if system_type == 'nt': | if system_type == 'nt': | ||||||
|     def lock(file, flags): |     def lock(file, flags): | ||||||
|         hfile = win32file._get_osfhandle(file.fileno()) |         hfile = win32file._get_osfhandle(fd(file)) | ||||||
|         win32file.LockFileEx(hfile, flags, 0, -0x10000, __overlapped) |         win32file.LockFileEx(hfile, flags, 0, -0x10000, __overlapped) | ||||||
|  |  | ||||||
|     def unlock(file): |     def unlock(file): | ||||||
|         hfile = win32file._get_osfhandle(file.fileno()) |         hfile = win32file._get_osfhandle(fd(file)) | ||||||
|         win32file.UnlockFileEx(hfile, 0, -0x10000, __overlapped) |         win32file.UnlockFileEx(hfile, 0, -0x10000, __overlapped) | ||||||
| elif system_type == 'posix': | elif system_type == 'posix': | ||||||
|     def lock(file, flags): |     def lock(file, flags): | ||||||
|         fcntl.flock(file.fileno(), flags) |         fcntl.flock(fd(file), flags) | ||||||
|  |  | ||||||
|     def unlock(file): |     def unlock(file): | ||||||
|         fcntl.flock(file.fileno(), fcntl.LOCK_UN) |         fcntl.flock(fd(file), fcntl.LOCK_UN) | ||||||
| else: | else: | ||||||
|     # File locking is not supported. |     # File locking is not supported. | ||||||
|     LOCK_EX = LOCK_SH = LOCK_NB = None |     LOCK_EX = LOCK_SH = LOCK_NB = None | ||||||
|   | |||||||
| @@ -44,16 +44,17 @@ def file_move_safe(old_file_name, new_file_name, chunk_size = 1024*64, allow_ove | |||||||
|         pass |         pass | ||||||
|  |  | ||||||
|     # If the built-in didn't work, do it the hard way. |     # If the built-in didn't work, do it the hard way. | ||||||
|     new_file = open(new_file_name, 'wb') |     fd = os.open(new_file_name, os.O_WRONLY | os.O_CREAT | os.O_EXCL | getattr(os, 'O_BINARY', 0)) | ||||||
|     locks.lock(new_file, locks.LOCK_EX) |     try: | ||||||
|     old_file = open(old_file_name, 'rb') |         locks.lock(fd, locks.LOCK_EX) | ||||||
|     current_chunk = None |         old_file = open(old_file_name, 'rb') | ||||||
|  |         current_chunk = None | ||||||
|     while current_chunk != '': |         while current_chunk != '': | ||||||
|         current_chunk = old_file.read(chunk_size) |             current_chunk = old_file.read(chunk_size) | ||||||
|         new_file.write(current_chunk) |             os.write(fd, current_chunk) | ||||||
|  |     finally: | ||||||
|     new_file.close() |         locks.unlock(fd) | ||||||
|     old_file.close() |         os.close(fd) | ||||||
|  |         old_file.close() | ||||||
|  |  | ||||||
|     os.remove(old_file_name) |     os.remove(old_file_name) | ||||||
|   | |||||||
| @@ -39,9 +39,9 @@ class Storage(object): | |||||||
|         # Get the proper name for the file, as it will actually be saved. |         # Get the proper name for the file, as it will actually be saved. | ||||||
|         if name is None: |         if name is None: | ||||||
|             name = content.name |             name = content.name | ||||||
|         name = self.get_available_name(name) |  | ||||||
|          |          | ||||||
|         self._save(name, content) |         name = self.get_available_name(name) | ||||||
|  |         name = self._save(name, content) | ||||||
|  |  | ||||||
|         # Store filenames with forward slashes, even on Windows |         # Store filenames with forward slashes, even on Windows | ||||||
|         return force_unicode(name.replace('\\', '/')) |         return force_unicode(name.replace('\\', '/')) | ||||||
| @@ -136,18 +136,40 @@ class FileSystemStorage(Storage): | |||||||
|         elif not os.path.isdir(directory): |         elif not os.path.isdir(directory): | ||||||
|             raise IOError("%s exists and is not a directory." % directory) |             raise IOError("%s exists and is not a directory." % directory) | ||||||
|  |  | ||||||
|         if hasattr(content, 'temporary_file_path'): |         # There's a potential race condition between get_available_name and | ||||||
|             # This file has a file path that we can move. |         # saving the file; it's possible that two threads might return the | ||||||
|             file_move_safe(content.temporary_file_path(), full_path) |         # same name, at which point all sorts of fun happens. So we need to | ||||||
|             content.close() |         # try to create the file, but if it already exists we have to go back | ||||||
|         else: |         # to get_available_name() and try again. | ||||||
|             # This is a normal uploadedfile that we can stream. |  | ||||||
|             fp = open(full_path, 'wb') |         while True: | ||||||
|             locks.lock(fp, locks.LOCK_EX) |             try: | ||||||
|             for chunk in content.chunks(): |                 # This file has a file path that we can move. | ||||||
|                 fp.write(chunk) |                 if hasattr(content, 'temporary_file_path'): | ||||||
|             locks.unlock(fp) |                     file_move_safe(content.temporary_file_path(), full_path) | ||||||
|             fp.close() |                     content.close() | ||||||
|  |  | ||||||
|  |                 # This is a normal uploadedfile that we can stream. | ||||||
|  |                 else: | ||||||
|  |                     # This fun binary flag incantation makes os.open throw an | ||||||
|  |                     # OSError if the file already exists before we open it. | ||||||
|  |                     fd = os.open(full_path, os.O_WRONLY | os.O_CREAT | os.O_EXCL | getattr(os, 'O_BINARY', 0)) | ||||||
|  |                     try: | ||||||
|  |                         locks.lock(fd, locks.LOCK_EX) | ||||||
|  |                         for chunk in content.chunks(): | ||||||
|  |                             os.write(fd, chunk) | ||||||
|  |                     finally: | ||||||
|  |                         locks.unlock(fd) | ||||||
|  |                         os.close(fd) | ||||||
|  |             except OSError: | ||||||
|  |                 # Ooops, we need a new file name. | ||||||
|  |                 name = self.get_available_name(name) | ||||||
|  |                 full_path = self.path(name) | ||||||
|  |             else: | ||||||
|  |                 # OK, the file save worked. Break out of the loop. | ||||||
|  |                 break | ||||||
|  |                  | ||||||
|  |         return name | ||||||
|  |  | ||||||
|     def delete(self, name): |     def delete(self, name): | ||||||
|         name = self.path(name) |         name = self.path(name) | ||||||
|   | |||||||
| @@ -64,3 +64,38 @@ u'custom_storage.2' | |||||||
| >>> custom_storage.delete(first) | >>> custom_storage.delete(first) | ||||||
| >>> custom_storage.delete(second) | >>> custom_storage.delete(second) | ||||||
| """ | """ | ||||||
|  |  | ||||||
|  | # Tests for a race condition on file saving (#4948). | ||||||
|  | # This is written in such a way that it'll always pass on platforms  | ||||||
|  | # without threading. | ||||||
|  |  | ||||||
|  | import time | ||||||
|  | from unittest import TestCase | ||||||
|  | from django.core.files.base import ContentFile | ||||||
|  | from models import temp_storage | ||||||
|  | try: | ||||||
|  |     import threading | ||||||
|  | except ImportError: | ||||||
|  |     import dummy_threading as threading | ||||||
|  |  | ||||||
|  | class SlowFile(ContentFile): | ||||||
|  |     def chunks(self): | ||||||
|  |         time.sleep(1) | ||||||
|  |         return super(ContentFile, self).chunks() | ||||||
|  |  | ||||||
|  | class FileSaveRaceConditionTest(TestCase): | ||||||
|  |     def setUp(self): | ||||||
|  |         self.thread = threading.Thread(target=self.save_file, args=['conflict']) | ||||||
|  |      | ||||||
|  |     def save_file(self, name): | ||||||
|  |         name = temp_storage.save(name, SlowFile("Data")) | ||||||
|  |      | ||||||
|  |     def test_race_condition(self): | ||||||
|  |         self.thread.start() | ||||||
|  |         name = self.save_file('conflict') | ||||||
|  |         self.thread.join() | ||||||
|  |         self.assert_(temp_storage.exists('conflict')) | ||||||
|  |         self.assert_(temp_storage.exists('conflict_')) | ||||||
|  |         temp_storage.delete('conflict') | ||||||
|  |         temp_storage.delete('conflict_') | ||||||
|  |  | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user