summaryrefslogtreecommitdiffstats
path: root/Python/ceval.c
diff options
context:
space:
mode:
authorSerhiy Storchaka <storchaka@gmail.com>2023-12-07 10:19:43 (GMT)
committerGitHub <noreply@github.com>2023-12-07 10:19:43 (GMT)
commit8660fb7fd7cdcbfe58ef304f5720efe97ca7c842 (patch)
tree17cd5657cd0c6cc5795d59973a1a572a9274d60c /Python/ceval.c
parent953ee622b3901d3467e65e3484dcfa75ba6fcddf (diff)
downloadcpython-8660fb7fd7cdcbfe58ef304f5720efe97ca7c842.zip
cpython-8660fb7fd7cdcbfe58ef304f5720efe97ca7c842.tar.gz
cpython-8660fb7fd7cdcbfe58ef304f5720efe97ca7c842.tar.bz2
gh-112660: Do not clear arbitrary errors on import (GH-112661)
Previously arbitrary errors could be cleared during formatting error messages for ImportError or AttributeError for modules. Now all unexpected errors are reported.
Diffstat (limited to 'Python/ceval.c')
-rw-r--r--Python/ceval.c46
1 files changed, 31 insertions, 15 deletions
diff --git a/Python/ceval.c b/Python/ceval.c
index 1806ceb..f8fa50e 100644
--- a/Python/ceval.c
+++ b/Python/ceval.c
@@ -2614,11 +2614,10 @@ import_from(PyThreadState *tstate, PyObject *v, PyObject *name)
/* Issue #17636: in case this failed because of a circular relative
import, try to fallback on reading the module directly from
sys.modules. */
- pkgname = PyObject_GetAttr(v, &_Py_ID(__name__));
- if (pkgname == NULL) {
- goto error;
+ if (PyObject_GetOptionalAttr(v, &_Py_ID(__name__), &pkgname) < 0) {
+ return NULL;
}
- if (!PyUnicode_Check(pkgname)) {
+ if (pkgname == NULL || !PyUnicode_Check(pkgname)) {
Py_CLEAR(pkgname);
goto error;
}
@@ -2635,42 +2634,59 @@ import_from(PyThreadState *tstate, PyObject *v, PyObject *name)
Py_DECREF(pkgname);
return x;
error:
- pkgpath = PyModule_GetFilenameObject(v);
if (pkgname == NULL) {
pkgname_or_unknown = PyUnicode_FromString("<unknown module name>");
if (pkgname_or_unknown == NULL) {
- Py_XDECREF(pkgpath);
return NULL;
}
} else {
pkgname_or_unknown = pkgname;
}
+ pkgpath = NULL;
+ if (PyModule_Check(v)) {
+ pkgpath = PyModule_GetFilenameObject(v);
+ if (pkgpath == NULL) {
+ if (!PyErr_ExceptionMatches(PyExc_SystemError)) {
+ Py_DECREF(pkgname_or_unknown);
+ return NULL;
+ }
+ // module filename missing
+ _PyErr_Clear(tstate);
+ }
+ }
if (pkgpath == NULL || !PyUnicode_Check(pkgpath)) {
- _PyErr_Clear(tstate);
+ Py_CLEAR(pkgpath);
errmsg = PyUnicode_FromFormat(
"cannot import name %R from %R (unknown location)",
name, pkgname_or_unknown
);
- /* NULL checks for errmsg and pkgname done by PyErr_SetImportError. */
- _PyErr_SetImportErrorWithNameFrom(errmsg, pkgname, NULL, name);
}
else {
- PyObject *spec = PyObject_GetAttr(v, &_Py_ID(__spec__));
+ PyObject *spec;
+ int rc = PyObject_GetOptionalAttr(v, &_Py_ID(__spec__), &spec);
+ if (rc > 0) {
+ rc = _PyModuleSpec_IsInitializing(spec);
+ Py_DECREF(spec);
+ }
+ if (rc < 0) {
+ Py_DECREF(pkgname_or_unknown);
+ Py_DECREF(pkgpath);
+ return NULL;
+ }
const char *fmt =
- _PyModuleSpec_IsInitializing(spec) ?
+ rc ?
"cannot import name %R from partially initialized module %R "
"(most likely due to a circular import) (%S)" :
"cannot import name %R from %R (%S)";
- Py_XDECREF(spec);
errmsg = PyUnicode_FromFormat(fmt, name, pkgname_or_unknown, pkgpath);
- /* NULL checks for errmsg and pkgname done by PyErr_SetImportError. */
- _PyErr_SetImportErrorWithNameFrom(errmsg, pkgname, pkgpath, name);
}
+ /* NULL checks for errmsg and pkgname done by PyErr_SetImportError. */
+ _PyErr_SetImportErrorWithNameFrom(errmsg, pkgname, pkgpath, name);
Py_XDECREF(errmsg);
- Py_XDECREF(pkgname_or_unknown);
+ Py_DECREF(pkgname_or_unknown);
Py_XDECREF(pkgpath);
return NULL;
}