summaryrefslogtreecommitdiff
path: root/Objects/moduleobject.c
diff options
context:
space:
mode:
authorFilipe LaĆ­ns <lains@riseup.net>2021-07-24 23:44:46 +0100
committerGitHub <noreply@github.com>2021-07-24 23:44:46 +0100
commit0a8ae8a50a0fea3d39ec49b220a5c7a5b70e36f8 (patch)
treea02d5e8a7613fb4187a490a1670fec65ed868ca7 /Objects/moduleobject.c
parent717f608c4a52664a03f8f2b9c57eaa25b24a7837 (diff)
downloadcpython-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.c30
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'",