Commit c82bfd87 authored by Victor Stinner's avatar Victor Stinner

Issue #18664, #18408: Rewrite PyErr_WriteUnraisable() to handle errors

 * Catch PyFile_WriteString() and PyFile_WriteObject() errors
 * Clear the current exception on _PyObject_GetAttrId() failure
 * Use PyUnicode_CompareWithASCIIString() and PyFile_WriteObject() instead of
   _PyUnicode_AsString() and strcmp() to avoid Unicode encoding error. stderr
   has a more tolerant error handler than utf-8/strict.
parent e5132102
...@@ -825,18 +825,30 @@ PyErr_WriteUnraisable(PyObject *obj) ...@@ -825,18 +825,30 @@ PyErr_WriteUnraisable(PyObject *obj)
{ {
_Py_IDENTIFIER(__module__); _Py_IDENTIFIER(__module__);
PyObject *f, *t, *v, *tb; PyObject *f, *t, *v, *tb;
PyObject *moduleName = NULL;
char* className;
PyErr_Fetch(&t, &v, &tb); PyErr_Fetch(&t, &v, &tb);
f = PySys_GetObject("stderr"); f = PySys_GetObject("stderr");
if (f != NULL && f != Py_None) { if (f == NULL || f == Py_None)
goto done;
if (obj) { if (obj) {
PyFile_WriteString("Exception ignored in: ", f); if (PyFile_WriteString("Exception ignored in: ", f) < 0)
PyFile_WriteObject(obj, f, 0); goto done;
PyFile_WriteString("\n", f); if (PyFile_WriteObject(obj, f, 0) < 0)
goto done;
if (PyFile_WriteString("\n", f) < 0)
goto done;
} }
PyTraceBack_Print(tb, f);
if (t) { if (PyTraceBack_Print(tb, f) < 0)
PyObject* moduleName; goto done;
char* className;
if (!t)
goto done;
assert(PyExceptionClass_Check(t)); assert(PyExceptionClass_Check(t));
className = PyExceptionClass_Name(t); className = PyExceptionClass_Name(t);
if (className != NULL) { if (className != NULL) {
...@@ -846,33 +858,43 @@ PyErr_WriteUnraisable(PyObject *obj) ...@@ -846,33 +858,43 @@ PyErr_WriteUnraisable(PyObject *obj)
} }
moduleName = _PyObject_GetAttrId(t, &PyId___module__); moduleName = _PyObject_GetAttrId(t, &PyId___module__);
if (moduleName == NULL) if (moduleName == NULL) {
PyFile_WriteString("<unknown>", f); PyErr_Clear();
if (PyFile_WriteString("<unknown>", f) < 0)
goto done;
}
else { else {
char* modstr = _PyUnicode_AsString(moduleName); if (PyUnicode_CompareWithASCIIString(moduleName, "builtins") != 0) {
if (modstr && if (PyFile_WriteObject(moduleName, f, Py_PRINT_RAW) < 0)
strcmp(modstr, "builtins") != 0) goto done;
{ if (PyFile_WriteString(".", f) < 0)
PyFile_WriteString(modstr, f); goto done;
PyFile_WriteString(".", f);
} }
} }
if (className == NULL) if (className == NULL) {
PyFile_WriteString("<unknown>", f); if (PyFile_WriteString("<unknown>", f) < 0)
else goto done;
PyFile_WriteString(className, f);
if (v && v != Py_None) {
PyFile_WriteString(": ", f);
PyFile_WriteObject(v, f, Py_PRINT_RAW);
} }
PyFile_WriteString("\n", f); else {
Py_XDECREF(moduleName); if (PyFile_WriteString(className, f) < 0)
goto done;
} }
PyErr_Clear(); /* Just in case */
if (v && v != Py_None) {
if (PyFile_WriteString(": ", f) < 0)
goto done;
if (PyFile_WriteObject(v, f, Py_PRINT_RAW) < 0)
goto done;
} }
if (PyFile_WriteString("\n", f) < 0)
goto done;
done:
Py_XDECREF(moduleName);
Py_XDECREF(t); Py_XDECREF(t);
Py_XDECREF(v); Py_XDECREF(v);
Py_XDECREF(tb); Py_XDECREF(tb);
PyErr_Clear(); /* Just in case */
} }
extern PyObject *PyModule_GetWarningsModule(void); extern PyObject *PyModule_GetWarningsModule(void);
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment