diff options
author | Filipe LaĆns <lains@riseup.net> | 2021-07-24 23:44:46 +0100 |
---|---|---|
committer | GitHub <noreply@github.com> | 2021-07-24 23:44:46 +0100 |
commit | 0a8ae8a50a0fea3d39ec49b220a5c7a5b70e36f8 (patch) | |
tree | a02d5e8a7613fb4187a490a1670fec65ed868ca7 /Objects/moduleobject.c | |
parent | 717f608c4a52664a03f8f2b9c57eaa25b24a7837 (diff) | |
download | cpython-git-0a8ae8a50a0fea3d39ec49b220a5c7a5b70e36f8.tar.gz |
bpo-44717: improve AttributeError on circular imports of submodules (GH-27338)
Diffstat (limited to 'Objects/moduleobject.c')
-rw-r--r-- | Objects/moduleobject.c | 30 |
1 files changed, 30 insertions, 0 deletions
diff --git a/Objects/moduleobject.c b/Objects/moduleobject.c index ef39bde4e4..1c37eb08c8 100644 --- a/Objects/moduleobject.c +++ b/Objects/moduleobject.c @@ -739,6 +739,30 @@ _PyModuleSpec_IsInitializing(PyObject *spec) return 0; } +/* Check if the submodule name is in the "_uninitialized_submodules" attribute + of the module spec. + */ +int +_PyModuleSpec_IsUninitializedSubmodule(PyObject *spec, PyObject *name) +{ + if (spec == NULL) { + return 0; + } + + _Py_IDENTIFIER(_uninitialized_submodules); + PyObject *value = _PyObject_GetAttrId(spec, &PyId__uninitialized_submodules); + if (value == NULL) { + return 0; + } + + int is_uninitialized = PySequence_Contains(value, name); + Py_DECREF(value); + if (is_uninitialized == -1) { + return 0; + } + return is_uninitialized; +} + static PyObject* module_getattro(PyModuleObject *m, PyObject *name) { @@ -773,6 +797,12 @@ module_getattro(PyModuleObject *m, PyObject *name) "(most likely due to a circular import)", mod_name, name); } + else if (_PyModuleSpec_IsUninitializedSubmodule(spec, name)) { + PyErr_Format(PyExc_AttributeError, + "cannot access submodule '%U' of module '%U' " + "(most likely due to a circular import)", + name, mod_name); + } else { PyErr_Format(PyExc_AttributeError, "module '%U' has no attribute '%U'", |