homepage

This issue tracker has been migrated to GitHub , and is currently read-only.
For more information, see the GitHub FAQs in the Python's Developer Guide.

classification
Title: PyTraceBack_Print() fails if signal received but PyErr_CheckSignals() not called
Type: behavior Stage: patch review
Components: Interpreter Core Versions: Python 3.2, Python 3.3, Python 2.7
process
Status: open Resolution:
Dependencies: Superseder:
Assigned To: Nosy List: amaury.forgeotdarc, pitrou, sbt
Priority: normal Keywords: patch

Created on 2011年12月28日 21:34 by sbt, last changed 2022年04月11日 14:57 by admin.

Files
File name Uploaded Description Edit
testsigint.zip sbt, 2011年12月28日 21:34
traceback_checksignals.patch sbt, 2011年12月29日 18:19 review
traceback_checksignals_2.patch sbt, 2012年01月08日 16:07 review
Messages (7)
msg150319 - (view) Author: Richard Oudkerk (sbt) * (Python committer) Date: 2011年12月28日 21:34
If SIGINT arrives while a function implemented in C is executing, then it prevents the function from raising an exception unless the function first calls PyErr_CheckSignals(). (If the function returns an object (instead of NULL) then KeyboardInterrupt is raised as expected.)
For example, the following function just spins for 5 seconds before raising RuntimeError:
 static PyObject *
 testsigint_wait(PyObject *self, PyObject *arg)
 {
 clock_t start = clock();
 while (clock() - start < 5 * CLOCKS_PER_SEC) {
 /* pass */
 }
 //PyErr_CheckSignals();
 PyErr_SetNone(PyExc_RuntimeError);
 return NULL;
 }
If I call this function and press Ctrl-C before it completes, then I get the following:
 >>> import testsigint
 >>> a = testsigint.wait()
 ^C>>> print(a)
 Traceback (most recent call last):
 File "<stdin>", line 1, in <module>
 NameError: name 'a' is not defined
So the call failed, but no exception was raised, and the variable "a" was not set!
I would have expected RuntimeError (or KeyboardInterrupt) to be raised. If I uncomment the PyErr_CheckSignals() line then I get RuntimeError as expected:
 >>> import testsigint
 >>> a = testsigint.wait()
 ^CTraceback (most recent call last):
 File "<stdin>", line 1, in <module>
 RuntimeError
Also, if I wrap the call in try...finally or try...except, I get a sensible "chained" traceback:
 >>> try:
 ... testsigint.wait()
 ... finally:
 ... print("done")
 ... 
 ^CTraceback (most recent call last):
 File "<stdin>", line 2, in <module>
 RuntimeError
 During handling of the above exception, another exception occurred:
 Traceback (most recent call last):
 File "<stdin>", line 2, in <module>
 KeyboardInterrupt
(Tested under Linux and Windows with the default branch.)
msg150328 - (view) Author: Richard Oudkerk (sbt) * (Python committer) Date: 2011年12月29日 12:25
I have tried the same with Python 2.7.1 on Linux. The problem is the same, but one gets a partial traceback with no exception:
 >>> import sys, testsigint
 >>> testsigint.wait()
 ^CTraceback (most recent call last):
 File "<stdin>", line 1, in <module>
 >>> sys.last_value
 RuntimeError()
Both on 2.7 and 3.3 sys.last_value gives RuntimeError().
msg150331 - (view) Author: Richard Oudkerk (sbt) * (Python committer) Date: 2011年12月29日 14:58
I think I have found the problem. PyTraceBack_Print() calls PyFile_WriteString(), which calls PyFile_WriteObject(), which calls PyObject_Str() which begins with 
 PyObject_Str(PyObject *v)
 {
 PyObject *res;
 if (PyErr_CheckSignals())
 return NULL;
 ...
Since PyErr_CheckSignals() returns -1, PyTraceBack_Print() fails.
(Changed title.)
msg150340 - (view) Author: Richard Oudkerk (sbt) * (Python committer) Date: 2011年12月29日 18:19
Attached is a patch for the default branch.
Before calling PyFile_WriteString() the patch saves the current exception. Then it calls PyErr_CheckSignals() and clears the current exception if any. After calling PyFile_WriteString() the exception is restored.
I am not sure this is an appropriate fix.
msg150342 - (view) Author: Antoine Pitrou (pitrou) * (Python committer) Date: 2011年12月29日 18:40
I think calling PyErr_WriteUnraisable would be more appropriate than PyErr_Clear.
I also wonder whether it's ok to ignore the exception. Pressing e.g. Ctrl-C generally shouldn't fail to stop the program, even if another exception is being processed at that moment.
(of course, you could argue this is already the case when e.g. the signal is received while in a __del__)
msg150352 - (view) Author: Richard Oudkerk (sbt) * (Python committer) Date: 2011年12月29日 22:08
> I think calling PyErr_WriteUnraisable would be more appropriate than 
> PyErr_Clear.
You mean just adding
 PyErr_CheckSignals();
 if (PyErr_Occurred())
 PyErr_WriteUnraisable(NULL);
before the call to PyFile_WriteString()? That seems to work:
 >>> from testsigint import *; wait()
 ^CException KeyboardInterrupt ignored
 Traceback (most recent call last):
 File "<stdin>", line 1, in <module>
 RuntimeError
> I also wonder whether it's ok to ignore the exception. Pressing e.g. 
> Ctrl-C generally shouldn't fail to stop the program, even if another 
> exception is being processed at that moment.
The ignoring and clearing of exceptions also happens higher (lower?) in the call stack in print_exception() and print_exception_recursive(). For example, print_exception() ends with
 /* If an error happened here, don't show it.
 XXX This is wrong, but too many callers rely on this behavior. */
 if (err != 0)
 PyErr_Clear();
 }
msg150875 - (view) Author: Richard Oudkerk (sbt) * (Python committer) Date: 2012年01月08日 16:07
Trivial 3 lines patch.
I guess there is still a race: if Ctrl-C is pressed after PyErr_CheckSignals() is called but before PyObject_Str() then the printing of any exception can still be suppressed.
History
Date User Action Args
2022年04月11日 14:57:25adminsetgithub: 57882
2015年10月13日 04:38:39martin.pantersetmessages: - msg185911
2015年10月13日 04:38:37martin.pantersetmessages: - msg185910
2013年04月03日 10:41:05sbtsetmessages: + msg185911
2013年04月03日 10:39:44sbtsetfiles: - input-ctrlc.patch
2013年04月03日 10:39:03sbtsetfiles: + input-ctrlc.patch

messages: + msg185910
2012年01月08日 16:07:57sbtsetfiles: + traceback_checksignals_2.patch

messages: + msg150875
2011年12月29日 22:08:55sbtsetmessages: + msg150352
2011年12月29日 18:40:09pitrousetnosy: + amaury.forgeotdarc, pitrou

messages: + msg150342
stage: patch review
2011年12月29日 18:19:38sbtsetfiles: + traceback_checksignals.patch
keywords: + patch
messages: + msg150340

components: + Interpreter Core
2011年12月29日 14:58:57sbtsettype: behavior
title: SIGINT prevents raising of exceptions unless PyErr_CheckSignals() called -> PyTraceBack_Print() fails if signal received but PyErr_CheckSignals() not called
messages: + msg150331
versions: + Python 2.7, Python 3.2
2011年12月29日 12:25:24sbtsetmessages: + msg150328
2011年12月28日 21:34:59sbtcreate

AltStyle によって変換されたページ (->オリジナル) /