The Anatomy of a Traceback
Nir Cohen @ strigo.io
nir0s@github
@thinkops
BEFORE TRACEBACK
...
Traceback (most recent call last):
File "test.py", line 115, in <module>
i = Moose()
File "test.py", line 17, in __init__
self.a_moose_once_bit_my_sister()
File "test.py", line 20, in a_moose_once_bit_my_sister
_mynd_you()
File "test.py", line 110, in _mynd_you
test2.moose_bites_kan_be_pretty_nasti()
File "...test2.py", line 10, in moose_bites_kan_be_pretty_nasti
moose_func(ex)
File "...test2.py", line 14, in moose_func
raise raise_moose(ex)
File "...test2.py", line 18, in raise_moose
raise MooseError(ex)
test2.MooseError: [Errno 2] No such file or directory: \
'certainly_a_non_existing_moose'
The traceback object
try:
raise Exception
except Exception as e:
# py3 specific. In py2 would use:
# _, __, tb = sys.exc_info()
# raise Exception(tb)
raise Exception(e.__traceback__)
...
Traceback (most recent call last):
File "test.py", line 7, in <module>
raise Exception(e.__traceback__)
Exception: <traceback object at 0x7ff5e7f38e08>
Call stack
The call to all functions during execution, where a call to a single function is called a Stack Frame
tb object properties
- `tb_frame` - frame object at this level
- `tb_lasti` - index of last attempted instruction in bytecode
- `tb_lineno` - current line number in Python source code
- `tb_next` - next inner traceback object
The frame object
- `f_back` - next outer frame object
- `f_builtins` - builtins namespace seen by this frame
- `f_code` - code object being executed by this frame
- `f_globals` - global namespace seen by this frame
- `f_lasti` - ...
- `f_lineno` - ...
- `f_locals` - local namespace seen by this frame
- `f_trace` - tracing function for this frame, or `None`
The code object
Too much information.
(See https://goo.gl/lCCxXr)
Use case
Let's say we want to create a logger which will take a parsed/reformatted traceback and ship it for analysis.
from .logger.tracelogger import parse, send
tracelogger.setup(host=..., port=..., format='json')
try:
raise Exception
except Exception as e:
send(parse(traceback.extract_tb(e.__traceback__)))
Replacing the excepthook
def new_excepthook(tpe, value, tb):
# tpe - exception type (e.g. MyException)
# value - the value of the exception
# (e.g. No such file or directory ...)
# tb - the traceback object
do_something_with_exception_type(tpe)
do_something_with_except_value(value)
reformat_traceback(tb)
sys.excepthook = new_excepthook
Extract some info
import traceback
def reformat_traceback(tb):
# Again, a bit different in py2
tb_entries = traceback.extract_tb(e.__traceback__)
for entry in tb_entries:
print((entry.filename, entry.lineno, entry.name, entry.line))
...
('test.py', 140, 'i = Moose()', '<module>')
('test.py', 42, 'self.a_moose_once_bit_my_sister()', '__init__')
('test.py', 45, '_mynd_you()', 'a_moose_once_bit_my_sister')
('test.py', 135, 'test2.moose_bites_kan_be_pretty_nasti()', '_mynd_you')
('.../test2.py', 10, 'moose_func(ex)', 'moose_bites_kan_be_pretty_nasti')
('.../test2.py', 14, 'raise raise_moose(ex)', 'moose_func')
('.../test2.py', 18, 'raise MooseError(ex)', 'raise_moose')
Another use case
Collecting all tracebacks from multiple threads and handling in the main thread
backtrace
Formats tracebacks
Usage (Python)
import backtrace
backtrace.hook(
reverse=False,
align=True,
strip_path=True,
enable_on_envvar_only=False,
on_tty=False,
conservative=False,
styles={})
Usage (Bash)
$ python raise_exception.py 2>&1 | backtrace
...
Traceback (Most recent call last):
115 test.py <module> --> i = Moose()
17 test.py __init__ --> self.a_moose_once_bit_my_sister()
20 test.py a_moose_once_bit_my_sister --> _mynd_you()
110 test.py _mynd_you --> test2.moose_bites_kan_be_pretty_nasti()
...
Project Page
References
- traceback.py source: https://goo.gl/70bGY8
- traceback.c source: https://goo.gl/0AC9Me
- sysmodule source ref: https://goo.gl/URJBb6
- frame object source ref: https://goo.gl/1FR5ZG
- https://late.am/post/2012/03/26/exploring-python-code-objects.html
- Stack vs. Frame: https://goo.gl/abuRHm
- GDB Frames: https://goo.gl/zqc0zR
- https://docs.python.org/3/library/traceback.html
The Anatomy of a Traceback
By Nir Cohen
The Anatomy of a Traceback
- 1,855