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:
parent
ab1a442a01
commit
58cd4902a7
|
@ -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:
|
||||||
|
locks.lock(fd, locks.LOCK_EX)
|
||||||
old_file = open(old_file_name, 'rb')
|
old_file = open(old_file_name, 'rb')
|
||||||
current_chunk = None
|
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)
|
||||||
|
os.close(fd)
|
||||||
old_file.close()
|
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
|
||||||
|
# saving the file; it's possible that two threads might return the
|
||||||
|
# same name, at which point all sorts of fun happens. So we need to
|
||||||
|
# try to create the file, but if it already exists we have to go back
|
||||||
|
# to get_available_name() and try again.
|
||||||
|
|
||||||
|
while True:
|
||||||
|
try:
|
||||||
# This file has a file path that we can move.
|
# This file has a file path that we can move.
|
||||||
|
if hasattr(content, 'temporary_file_path'):
|
||||||
file_move_safe(content.temporary_file_path(), full_path)
|
file_move_safe(content.temporary_file_path(), full_path)
|
||||||
content.close()
|
content.close()
|
||||||
else:
|
|
||||||
# This is a normal uploadedfile that we can stream.
|
# This is a normal uploadedfile that we can stream.
|
||||||
fp = open(full_path, 'wb')
|
else:
|
||||||
locks.lock(fp, locks.LOCK_EX)
|
# 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():
|
for chunk in content.chunks():
|
||||||
fp.write(chunk)
|
os.write(fd, chunk)
|
||||||
locks.unlock(fp)
|
finally:
|
||||||
fp.close()
|
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_')
|
||||||
|
|
||||||
|
|
Loading…
Reference in New Issue