Can I put break points on background threads in Python?
Asked Answered
C

4

9

I'm using the PyDev for Eclipse plugin, and I'm trying to set a break point in some code that gets run in a background thread. The break point never gets hit even though the code is executing. Here's a small example:

import thread

def go(count):
    print 'count is %d.' % count # set break point here

print 'calling from main thread:'
go(13)
print 'calling from bg thread:'
thread.start_new_thread(go, (23,))

raw_input('press enter to quit.')

The break point in that example gets hit when it's called on the main thread, but not when it's called from a background thread. Is there anything I can do, or is that a limitation of the PyDev debugger?

Update

Thanks for the work arounds. I submitted a PyDev feature request, and it has been completed. It should be released with version 1.6.0. Thanks, PyDev team!

Clarino answered 24/6, 2010 at 23:50 Comment(0)
T
12

The problem is that there's no API in the thread module to know when a thread starts.

What you can do in your example is set the debugger trace function yourself (as Alex pointed) as in the code below (if you're not in the remote debugger, the pydevd.connected = True is currently required -- I'll change pydev so that this is not needed anymore). You may want to add a try..except ImportError for the pydevd import (which will fail if you're not running in the debugger)

def go(count):

   import pydevd
   pydevd.connected = True
   pydevd.settrace(suspend=False)
   print 'count is %d.' % count # set break point here

Now, on a second thought, I think that pydev can replace the start_new_thread method in the thread module providing its own function which will setup the debugger and later call the original function (just did that and it seems to be working, so, if you use the nightly that will be available in some hours, which will become the future 1.6.0, it should be working without doing anything special).

Trivium answered 14/7, 2010 at 1:35 Comment(5)
Hey, that's great news! Thanks for the new feature.Clarino
surprisingly enough i have the same issue in pydev 2.6. I have a GUI app where class A starts class B(QtCore.QThread) and debugger doesn't stop in the B methods, while the code is executed. If I'm trying to step into the b.start() call debugger sends me to the end of "main" constructCyclamen
i cannot resolve pydevd (python 2.7) so this work around is not for meCyclamen
pydevd is at eclipse/plugins/org.python.pydev_XXX/pysrc. You must add it to your pythonpath -- In the latest versions of PyDev, there's a 'pydevd' template which will add it to your path as well as doing the settrace.Trivium
@Fabio, I am using PyDev 3.9 and not seeing breakpoints in background threads created with threading. I still need to use pydevd.settrace(suspend=False) to get the to work. Should your fix have also covered threads created with threading (vs thread)?Fanni
R
4

The underlying issue is with sys.settrace, the low-level Python function used to perform all tracing and debugging -- as the docs say,

The function is thread-specific; for a debugger to support multiple threads, it must be registered using settrace() for each thread being debugged.

I believe that when you set a breakpoint in PyDev, the resulting settrace call is always happening on the main thread (I have not looked at PyDev recently so they may have added some way to work around that, but I don't recall any from the time when I did look).

A workaround you might implement yourself is, in your main thread after the breakpoint has been set, to use sys.gettrace to get PyDev's trace function, save it in a global variable, and make sure in all threads of interest to call sys.settrace with that global variable as the argument -- a tad cumbersome (more so for threads that already exist at the time the breakpoint is set!), but I can't think of any simpler alternative.

Remains answered 25/6, 2010 at 0:4 Comment(2)
This works sometimes. It seems like I have to call the sys.settrace() in a different stack frame from where the break point is. I haven't investigated fully, but it works well enough for me. Thanks.Clarino
@Don, you're welcome -- I guess PyDev's tracing function must somehow be responsible for that "different frame" issue (though it's not obvious exactly how -- there are some possibilities, but even finding out probably wouldn't help fix), but anyway I'm glad it can work for you. Be sure to add a feature request for "break in any thread" to PyDev's tracker so they know it's desired!Remains
C
2

On this question, I found a way to start the command-line debugger:

import pdb; pdb.set_trace()

It's not as easy to use as the Eclipse debugger, but it's better than nothing.

Clarino answered 25/6, 2010 at 0:16 Comment(0)
B
1

For me this worked according to one of Fabio's posts, after setting the trace with setTrace("000.000.000.000") # where 0's are the IP of your computer running Eclipse/PyDev

threading.settrace(pydevd.GetGlobalDebugger().trace_dispatch)
Bullate answered 13/11, 2013 at 8:8 Comment(1)
It works !! Thanks I'm really surprised on the small number of posts on pydevd.Artillery

© 2022 - 2025 — McMap. All rights reserved.