Skip to content

Commit f5c408c

Browse files
brettcannonadorilson
authored andcommitted
bpo-42133: update parts of the stdlib to fall back to __spec__.loader when __loader__ is missing (python#22929)
1 parent c51db35 commit f5c408c

File tree

6 files changed

+59
-14
lines changed

6 files changed

+59
-14
lines changed

Doc/whatsnew/3.10.rst

+24
Original file line numberDiff line numberDiff line change
@@ -186,6 +186,12 @@ by :func:`curses.color_content`, :func:`curses.init_color`,
186186
support is provided by the underlying ncurses library.
187187
(Contributed by Jeffrey Kintscher and Hans Petter Jansson in :issue:`36982`.)
188188

189+
doctest
190+
-------
191+
192+
When a module does not define ``__loader__``, fall back to ``__spec__.loader``.
193+
(Contributed by Brett Cannon in :issue:`42133`.)
194+
189195
encodings
190196
---------
191197
:func:`encodings.normalize_encoding` now ignores non-ASCII characters.
@@ -198,6 +204,18 @@ Added the *root_dir* and *dir_fd* parameters in :func:`~glob.glob` and
198204
:func:`~glob.iglob` which allow to specify the root directory for searching.
199205
(Contributed by Serhiy Storchaka in :issue:`38144`.)
200206

207+
inspect
208+
-------
209+
210+
When a module does not define ``__loader__``, fall back to ``__spec__.loader``.
211+
(Contributed by Brett Cannon in :issue:`42133`.)
212+
213+
linecache
214+
---------
215+
216+
When a module does not define ``__loader__``, fall back to ``__spec__.loader``.
217+
(Contributed by Brett Cannon in :issue:`42133`.)
218+
201219
os
202220
--
203221

@@ -217,6 +235,12 @@ The :mod:`shelve` module now uses :data:`pickle.DEFAULT_PROTOCOL` by default
217235
instead of :mod:`pickle` protocol ``3`` when creating shelves.
218236
(Contributed by Zackery Spytz in :issue:`34204`.)
219237

238+
site
239+
----
240+
241+
When a module does not define ``__loader__``, fall back to ``__spec__.loader``.
242+
(Contributed by Brett Cannon in :issue:`42133`.)
243+
220244
sys
221245
---
222246

Lib/doctest.py

+11-7
Original file line numberDiff line numberDiff line change
@@ -222,13 +222,17 @@ def _load_testfile(filename, package, module_relative, encoding):
222222
if module_relative:
223223
package = _normalize_module(package, 3)
224224
filename = _module_relative_path(package, filename)
225-
if getattr(package, '__loader__', None) is not None:
226-
if hasattr(package.__loader__, 'get_data'):
227-
file_contents = package.__loader__.get_data(filename)
228-
file_contents = file_contents.decode(encoding)
229-
# get_data() opens files as 'rb', so one must do the equivalent
230-
# conversion as universal newlines would do.
231-
return _newline_convert(file_contents), filename
225+
if (loader := getattr(package, '__loader__', None)) is None:
226+
try:
227+
loader = package.__spec__.loader
228+
except AttributeError:
229+
pass
230+
if hasattr(loader, 'get_data'):
231+
file_contents = loader.get_data(filename)
232+
file_contents = file_contents.decode(encoding)
233+
# get_data() opens files as 'rb', so one must do the equivalent
234+
# conversion as universal newlines would do.
235+
return _newline_convert(file_contents), filename
232236
with open(filename, encoding=encoding) as f:
233237
return f.read(), filename
234238

Lib/inspect.py

+5-2
Original file line numberDiff line numberDiff line change
@@ -707,10 +707,13 @@ def getsourcefile(object):
707707
if os.path.exists(filename):
708708
return filename
709709
# only return a non-existent filename if the module has a PEP 302 loader
710-
if getattr(getmodule(object, filename), '__loader__', None) is not None:
710+
module = getmodule(object, filename)
711+
if getattr(module, '__loader__', None) is not None:
712+
return filename
713+
elif getattr(getattr(module, "__spec__", None), "loader", None) is not None:
711714
return filename
712715
# or it is in the linecache
713-
if filename in linecache.cache:
716+
elif filename in linecache.cache:
714717
return filename
715718

716719
def getabsfile(object, _filename=None):

Lib/linecache.py

+8-3
Original file line numberDiff line numberDiff line change
@@ -165,9 +165,14 @@ def lazycache(filename, module_globals):
165165
if not filename or (filename.startswith('<') and filename.endswith('>')):
166166
return False
167167
# Try for a __loader__, if available
168-
if module_globals and '__loader__' in module_globals:
169-
name = module_globals.get('__name__')
170-
loader = module_globals['__loader__']
168+
if module_globals and '__name__' in module_globals:
169+
name = module_globals['__name__']
170+
if (loader := module_globals.get('__loader__')) is None:
171+
if spec := module_globals.get('__spec__'):
172+
try:
173+
loader = spec.loader
174+
except AttributeError:
175+
pass
171176
get_source = getattr(loader, 'get_source', None)
172177

173178
if name and get_source:

Lib/site.py

+9-2
Original file line numberDiff line numberDiff line change
@@ -105,8 +105,15 @@ def makepath(*paths):
105105
def abs_paths():
106106
"""Set all module __file__ and __cached__ attributes to an absolute path"""
107107
for m in set(sys.modules.values()):
108-
if (getattr(getattr(m, '__loader__', None), '__module__', None) not in
109-
('_frozen_importlib', '_frozen_importlib_external')):
108+
loader_module = None
109+
try:
110+
loader_module = m.__loader__.__module__
111+
except AttributeError:
112+
try:
113+
loader_module = m.__spec__.loader.__module__
114+
except AttributeError:
115+
pass
116+
if loader_module not in {'_frozen_importlib', '_frozen_importlib_external'}:
110117
continue # don't mess with a PEP 302-supplied __file__
111118
try:
112119
m.__file__ = os.path.abspath(m.__file__)
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,2 @@
1+
Update various modules in the stdlib to fall back on `__spec__.loader` when
2+
`__loader__` isn't defined on a module.

0 commit comments

Comments
 (0)