summaryrefslogtreecommitdiffstats
path: root/Lib/importlib/_bootstrap.py
diff options
context:
space:
mode:
authorFilipe LaĆ­ns <lains@riseup.net>2021-07-24 22:44:46 (GMT)
committerGitHub <noreply@github.com>2021-07-24 22:44:46 (GMT)
commit0a8ae8a50a0fea3d39ec49b220a5c7a5b70e36f8 (patch)
treea02d5e8a7613fb4187a490a1670fec65ed868ca7 /Lib/importlib/_bootstrap.py
parent717f608c4a52664a03f8f2b9c57eaa25b24a7837 (diff)
downloadcpython-0a8ae8a50a0fea3d39ec49b220a5c7a5b70e36f8.zip
cpython-0a8ae8a50a0fea3d39ec49b220a5c7a5b70e36f8.tar.gz
cpython-0a8ae8a50a0fea3d39ec49b220a5c7a5b70e36f8.tar.bz2
bpo-44717: improve AttributeError on circular imports of submodules (GH-27338)
Diffstat (limited to 'Lib/importlib/_bootstrap.py')
-rw-r--r--Lib/importlib/_bootstrap.py15
1 files changed, 13 insertions, 2 deletions
diff --git a/Lib/importlib/_bootstrap.py b/Lib/importlib/_bootstrap.py
index 527bc9c..f8b77e3 100644
--- a/Lib/importlib/_bootstrap.py
+++ b/Lib/importlib/_bootstrap.py
@@ -361,6 +361,7 @@ class ModuleSpec:
self.origin = origin
self.loader_state = loader_state
self.submodule_search_locations = [] if is_package else None
+ self._uninitialized_submodules = []
# file-location attributes
self._set_fileattr = False
@@ -987,6 +988,7 @@ _ERR_MSG = _ERR_MSG_PREFIX + '{!r}'
def _find_and_load_unlocked(name, import_):
path = None
parent = name.rpartition('.')[0]
+ parent_spec = None
if parent:
if parent not in sys.modules:
_call_with_frames_removed(import_, parent)
@@ -999,15 +1001,24 @@ def _find_and_load_unlocked(name, import_):
except AttributeError:
msg = (_ERR_MSG + '; {!r} is not a package').format(name, parent)
raise ModuleNotFoundError(msg, name=name) from None
+ parent_spec = parent_module.__spec__
+ child = name.rpartition('.')[2]
spec = _find_spec(name, path)
if spec is None:
raise ModuleNotFoundError(_ERR_MSG.format(name), name=name)
else:
- module = _load_unlocked(spec)
+ if parent_spec:
+ # Temporarily add child we are currently importing to parent's
+ # _uninitialized_submodules for circular import tracking.
+ parent_spec._uninitialized_submodules.append(child)
+ try:
+ module = _load_unlocked(spec)
+ finally:
+ if parent_spec:
+ parent_spec._uninitialized_submodules.pop()
if parent:
# Set the module as an attribute on its parent.
parent_module = sys.modules[parent]
- child = name.rpartition('.')[2]
try:
setattr(parent_module, child, module)
except AttributeError: