summaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorYury Selivanov <yury@magic.io>2016-11-09 23:55:45 (GMT)
committerYury Selivanov <yury@magic.io>2016-11-09 23:55:45 (GMT)
commit46a02db90b295bfec1712515e09aeda31bbe84e0 (patch)
treef9c528e4230c67e830ea1eadd176a0a22b63b14d
parent28f42fd4f813b4be15b47629c557ef8f958727cb (diff)
downloadcpython-46a02db90b295bfec1712515e09aeda31bbe84e0.zip
cpython-46a02db90b295bfec1712515e09aeda31bbe84e0.tar.gz
cpython-46a02db90b295bfec1712515e09aeda31bbe84e0.tar.bz2
Issue #28653: Fix a refleak in functools.lru_cache.
-rw-r--r--Lib/test/test_functools.py19
-rw-r--r--Misc/NEWS2
-rw-r--r--Modules/_functoolsmodule.c8
3 files changed, 27 insertions, 2 deletions
diff --git a/Lib/test/test_functools.py b/Lib/test/test_functools.py
index 5076644..6a3bf64 100644
--- a/Lib/test/test_functools.py
+++ b/Lib/test/test_functools.py
@@ -1162,6 +1162,25 @@ class TestLRU:
self.assertEqual(misses, 4)
self.assertEqual(currsize, 2)
+ def test_lru_type_error(self):
+ # Regression test for issue #28653.
+ # lru_cache was leaking when one of the arguments
+ # wasn't cacheable.
+
+ @functools.lru_cache(maxsize=None)
+ def infinite_cache(o):
+ pass
+
+ @functools.lru_cache(maxsize=10)
+ def limited_cache(o):
+ pass
+
+ with self.assertRaises(TypeError):
+ infinite_cache([])
+
+ with self.assertRaises(TypeError):
+ limited_cache([])
+
def test_lru_with_maxsize_none(self):
@self.module.lru_cache(maxsize=None)
def fib(n):
diff --git a/Misc/NEWS b/Misc/NEWS
index f9e2f72..c50ce12 100644
--- a/Misc/NEWS
+++ b/Misc/NEWS
@@ -457,6 +457,8 @@ Library
- Issue #28652: Make loop methods reject socket kinds they do not support.
+- Issue #28653: Fix a refleak in functools.lru_cache.
+
IDLE
----
diff --git a/Modules/_functoolsmodule.c b/Modules/_functoolsmodule.c
index d785c49..9c9ab5f 100644
--- a/Modules/_functoolsmodule.c
+++ b/Modules/_functoolsmodule.c
@@ -781,8 +781,10 @@ infinite_lru_cache_wrapper(lru_cache_object *self, PyObject *args, PyObject *kwd
if (!key)
return NULL;
hash = PyObject_Hash(key);
- if (hash == -1)
+ if (hash == -1) {
+ Py_DECREF(key);
return NULL;
+ }
result = _PyDict_GetItem_KnownHash(self->cache, key, hash);
if (result) {
Py_INCREF(result);
@@ -837,8 +839,10 @@ bounded_lru_cache_wrapper(lru_cache_object *self, PyObject *args, PyObject *kwds
if (!key)
return NULL;
hash = PyObject_Hash(key);
- if (hash == -1)
+ if (hash == -1) {
+ Py_DECREF(key);
return NULL;
+ }
link = (lru_list_elem *)_PyDict_GetItem_KnownHash(self->cache, key, hash);
if (link) {
lru_cache_extricate_link(link);