Fixed #8616 -- Fixed a race condition in the file-based session backend.

Thanks to warren@wandrsmith.net for the patch.


git-svn-id: http://code.djangoproject.com/svn/django/trunk@8688 bcc190cf-cafb-0310-a4f2-bffc1f526a37
This commit is contained in:
Malcolm Tredinnick 2008-08-29 01:44:11 +00:00
parent 6e841ca767
commit 02f86a1c7c
2 changed files with 41 additions and 8 deletions

View File

@ -363,6 +363,7 @@ answer newbie questions, and generally made Django that much better:
Ben Slavin <benjamin.slavin@gmail.com> Ben Slavin <benjamin.slavin@gmail.com>
sloonz <simon.lipp@insa-lyon.fr> sloonz <simon.lipp@insa-lyon.fr>
SmileyChris <smileychris@gmail.com> SmileyChris <smileychris@gmail.com>
Warren Smith <warren@wandrsmith.net>
smurf@smurf.noris.de smurf@smurf.noris.de
Vsevolod Solovyov Vsevolod Solovyov
sopel sopel

View File

@ -5,7 +5,9 @@ import tempfile
from django.conf import settings from django.conf import settings
from django.contrib.sessions.backends.base import SessionBase, CreateError from django.contrib.sessions.backends.base import SessionBase, CreateError
from django.core.exceptions import SuspiciousOperation, ImproperlyConfigured from django.core.exceptions import SuspiciousOperation, ImproperlyConfigured
from django.core.files import locks
IO_LOCK_SUFFIX = "_iolock"
class SessionStore(SessionBase): class SessionStore(SessionBase):
""" """
@ -42,8 +44,22 @@ class SessionStore(SessionBase):
return os.path.join(self.storage_path, self.file_prefix + session_key) return os.path.join(self.storage_path, self.file_prefix + session_key)
def _key_to_io_lock_file(self, session_key=None):
"""
Get the I/O lock file associated with this session key.
"""
return self._key_to_file(session_key) + IO_LOCK_SUFFIX
def load(self): def load(self):
session_data = {} session_data = {}
try:
# Open and acquire a shared lock on the I/O lock file before
# attempting to read the session file. This makes us wait to read
# the session file until another thread or process is finished
# writing it.
lock_path = self._key_to_io_lock_file()
io_lock_file = open(lock_path, "rb")
locks.lock(io_lock_file, locks.LOCK_SH)
try: try:
session_file = open(self._key_to_file(), "rb") session_file = open(self._key_to_file(), "rb")
try: try:
@ -53,6 +69,10 @@ class SessionStore(SessionBase):
self.create() self.create()
finally: finally:
session_file.close() session_file.close()
finally:
locks.unlock(io_lock_file)
io_lock_file.close()
os.unlink(lock_path)
except IOError: except IOError:
pass pass
return session_data return session_data
@ -75,12 +95,24 @@ class SessionStore(SessionBase):
# Because this may trigger a load from storage, we must do it before # Because this may trigger a load from storage, we must do it before
# truncating the file to save. # truncating the file to save.
session_data = self._get_session(no_load=must_create) session_data = self._get_session(no_load=must_create)
try:
# Open and acquire an exclusive lock on the I/O lock file before
# attempting to write the session file. This makes other threads
# or processes wait to read or write the session file until we are
# finished writing it.
lock_path = self._key_to_io_lock_file()
io_lock_file = open(lock_path, "wb")
locks.lock(io_lock_file, locks.LOCK_EX)
try: try:
fd = os.open(self._key_to_file(self.session_key), flags) fd = os.open(self._key_to_file(self.session_key), flags)
try: try:
os.write(fd, self.encode(session_data)) os.write(fd, self.encode(session_data))
finally: finally:
os.close(fd) os.close(fd)
finally:
locks.unlock(io_lock_file)
io_lock_file.close()
os.unlink(lock_path)
except OSError, e: except OSError, e:
if must_create and e.errno == errno.EEXIST: if must_create and e.errno == errno.EEXIST:
raise CreateError raise CreateError