Skip to content

Commit c314198

Browse files
authored
gh-98253: Break potential reference cycles in external code worsened by typing.py lru_cache (#98591)
1 parent 8bb2303 commit c314198

File tree

2 files changed

+21
-3
lines changed

2 files changed

+21
-3
lines changed

Lib/typing.py

+11-3
Original file line numberDiff line numberDiff line change
@@ -325,20 +325,28 @@ def _flatten_literal_params(parameters):
325325

326326

327327
_cleanups = []
328+
_caches = {}
328329

329330

330331
def _tp_cache(func=None, /, *, typed=False):
331332
"""Internal wrapper caching __getitem__ of generic types with a fallback to
332333
original function for non-hashable arguments.
333334
"""
334335
def decorator(func):
335-
cached = functools.lru_cache(typed=typed)(func)
336-
_cleanups.append(cached.cache_clear)
336+
# The callback 'inner' references the newly created lru_cache
337+
# indirectly by performing a lookup in the global '_caches' dictionary.
338+
# This breaks a reference that can be problematic when combined with
339+
# C API extensions that leak references to types. See GH-98253.
340+
341+
cache = functools.lru_cache(typed=typed)(func)
342+
_caches[func] = cache
343+
_cleanups.append(cache.cache_clear)
344+
del cache
337345

338346
@functools.wraps(func)
339347
def inner(*args, **kwds):
340348
try:
341-
return cached(*args, **kwds)
349+
return _caches[func](*args, **kwds)
342350
except TypeError:
343351
pass # All real errors (not unhashable args) are raised below.
344352
return func(*args, **kwds)
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,10 @@
1+
The implementation of the typing module is now more resilient to reference
2+
leaks in binary extension modules.
3+
4+
Previously, a reference leak in a typed C API-based extension module could leak
5+
internals of the typing module, which could in turn introduce leaks in
6+
essentially any other package with typed function signatures. Although the
7+
typing package is not the original source of the problem, such non-local
8+
dependences exacerbate debugging of large-scale projects, and the
9+
implementation was therefore changed to reduce harm by providing better
10+
isolation.

0 commit comments

Comments
 (0)