{{{
FileNotFoundError: [Errno 2] No such file or directory:
'/app/var/cache/d729e4cf4ba88cba5a0f48e0396ec48a.djcache'
[...]
File "django/core/cache/backends/base.py", line 229, in get_or_set
self.add(key, default, timeout=timeout, version=version)
File "django/core/cache/backends/filebased.py", line 26, in add
if self.has_key(key, version):
File "django/core/cache/backends/filebased.py", line 94, in has_key
with open(fname, "rb") as f:
}}}
The code is:
{{{
def has_key(self, key, version=None):
fname = self._key_to_file(key, version)
if os.path.exists(fname):
with open(fname, "rb") as f:
return not self._is_expired(f)
return False
}}}
Between the `exists()` check and `open()`, it's possible for the file to
be deleted. In fact, the `_is_expired()` method itself deletes the file if
it finds it to be expired. So if many threads race to read an expired
cache at once, it's not that unlikely to hit this window.
--
Ticket URL: <https://code.djangoproject.com/ticket/34209>
Django <https://code.djangoproject.com/>
The Web framework for perfectionists with deadlines.
* owner: nobody => Marti Raudsepp
* status: new => assigned
--
Ticket URL: <https://code.djangoproject.com/ticket/34209#comment:1>
Old description:
> I received the exception from Django's cache framework:
>
> {{{
> FileNotFoundError: [Errno 2] No such file or directory:
> '/app/var/cache/d729e4cf4ba88cba5a0f48e0396ec48a.djcache'
> [...]
> File "django/core/cache/backends/base.py", line 229, in get_or_set
> self.add(key, default, timeout=timeout, version=version)
> File "django/core/cache/backends/filebased.py", line 26, in add
> if self.has_key(key, version):
> File "django/core/cache/backends/filebased.py", line 94, in has_key
> with open(fname, "rb") as f:
> }}}
>
> The code is:
> {{{
> def has_key(self, key, version=None):
> fname = self._key_to_file(key, version)
> if os.path.exists(fname):
> with open(fname, "rb") as f:
> return not self._is_expired(f)
> return False
> }}}
>
> Between the `exists()` check and `open()`, it's possible for the file to
> be deleted. In fact, the `_is_expired()` method itself deletes the file
> if it finds it to be expired. So if many threads race to read an expired
> cache at once, it's not that unlikely to hit this window.
New description:
I received the exception from Django's cache framework:
{{{
FileNotFoundError: [Errno 2] No such file or directory:
'/app/var/cache/d729e4cf4ba88cba5a0f48e0396ec48a.djcache'
[...]
File "django/core/cache/backends/base.py", line 229, in get_or_set
self.add(key, default, timeout=timeout, version=version)
File "django/core/cache/backends/filebased.py", line 26, in add
if self.has_key(key, version):
File "django/core/cache/backends/filebased.py", line 94, in has_key
with open(fname, "rb") as f:
}}}
The code is:
{{{#!python
def has_key(self, key, version=None):
fname = self._key_to_file(key, version)
if os.path.exists(fname):
with open(fname, "rb") as f:
return not self._is_expired(f)
return False
}}}
Between the `exists()` check and `open()`, it's possible for the file to
be deleted. In fact, the `_is_expired()` method itself deletes the file if
it finds it to be expired. So if many threads race to read an expired
cache at once, it's not that unlikely to hit this window.
--
--
Ticket URL: <https://code.djangoproject.com/ticket/34209#comment:2>
* type: Uncategorized => Bug
* stage: Unreviewed => Accepted
Comment:
Thanks for the report.
--
Ticket URL: <https://code.djangoproject.com/ticket/34209#comment:3>
* needs_tests: 0 => 1
--
Ticket URL: <https://code.djangoproject.com/ticket/34209#comment:4>
Comment (by Marti Raudsepp):
Added a test.
--
Ticket URL: <https://code.djangoproject.com/ticket/34209#comment:5>
* needs_tests: 1 => 0
* stage: Accepted => Ready for checkin
--
Ticket URL: <https://code.djangoproject.com/ticket/34209#comment:6>
* status: assigned => closed
* resolution: => fixed
Comment:
In [changeset:"32268456d6494483f9a737dc7c32bbbf1eceab8c" 32268456]:
{{{
#!CommitTicketReference repository=""
revision="32268456d6494483f9a737dc7c32bbbf1eceab8c"
Fixed #34209 -- Prevented FileBasedCache.has_key() crash caused by a race
condition.
}}}
--
Ticket URL: <https://code.djangoproject.com/ticket/34209#comment:7>
Comment (by Marti Raudsepp):
This fix doesn't qualify for backporting?
--
Ticket URL: <https://code.djangoproject.com/ticket/34209#comment:8>