1
# Copyright 2011 Canonical Ltd. This software is licensed under the
2
# GNU Affero General Public License version 3 (see the file LICENSE).
4
"""Replacement for some standard library traceback module functions.
6
These honor traceback supplements as defined in zope.exceptions.
23
DEBUG_EXCEPTION_FORMATTER = False
24
EXPLOSIVE_ERRORS = (SystemExit, MemoryError, KeyboardInterrupt)
27
def _try_except(callable, *args, **kwargs):
29
return callable(*args, **kwargs)
30
except EXPLOSIVE_ERRORS:
33
if DEBUG_EXCEPTION_FORMATTER:
39
"if the frame is None, make one."
42
raise ZeroDivisionError
43
except ZeroDivisionError:
44
f = sys.exc_info()[2].tb_frame.f_back.f_back
49
"Return the string as deemed suitable for the extra information."
50
return ' - %s' % string
53
def print_list(extracted_list, file=None):
54
"""Print the list of tuples as returned by extract_tb() or
55
extract_stack() as a formatted stack trace to the given file."""
58
for line in format_list(extracted_list):
62
def format_list(extracted_list):
63
"""Format a list of traceback entry tuples for printing.
65
Given a list of tuples as returned by extract_tb() or
66
extract_stack(), return a list of strings ready for printing.
67
Each string in the resulting list corresponds to the item with the
68
same index in the argument list. Each string ends in a newline;
69
the strings may contain internal newlines as well, for those items
70
whose source text line or supplement or info are not None.
73
for filename, lineno, name, line, modname, supp, info in extracted_list:
76
' File "%s", line %d, in %s' % (filename, lineno, name))
78
item.append(' %s' % line.strip())
79
# The "supp" and "info" bits are adapted from zope.exceptions.
82
if supp['source_url']:
83
item.append(_fmt(supp['source_url']))
87
_fmt('Line %(line)s, Column %(column)s' % supp))
89
item.append(_fmt('Line %(line)s' % supp))
91
item.append(_fmt('Column %(column)s' % supp))
92
if supp['expression']:
93
item.append(_fmt('Expression: %(expression)s' % supp))
95
for warning in supp['warnings']:
96
item.append(_fmt('Warning: %s' % warning))
98
item.append(supp['extra']) # We do not include a prefix.
100
item.append(_fmt(info))
101
except EXPLOSIVE_ERRORS:
104
# The values above may not stringify properly, or who knows what
105
# else. Be defensive.
106
if DEBUG_EXCEPTION_FORMATTER:
107
traceback.print_exc()
108
# else just swallow the exception.
109
item.append('') # This gives us a trailing newline.
110
list.append('\n'.join(item))
114
def print_stack(f=None, limit=None, file=None):
115
"""Print a stack trace from its invocation point.
117
The optional 'f' argument can be used to specify an alternate
118
stack frame at which to start. The optional 'limit' and 'file'
119
arguments have the same meaning as for print_exception().
121
print_list(extract_stack(_get_frame(f), limit), file)
124
def _get_limit(limit):
125
"Return the limit or the globally-set limit, if any."
127
# stdlib uses hasattr here.
128
if hasattr(sys, 'tracebacklimit'):
129
limit = sys.tracebacklimit
133
def _get_frame_data(f, lineno):
134
"Given a frame and a lineno, return data for each item of extract_*."
135
# Adapted from zope.exceptions.
136
try_except = _try_except # This is a micro-optimization.
137
modname = f.f_globals.get('__name__')
138
# Output a traceback supplement, if any.
139
supplement_dict = info = None
140
if '__traceback_supplement__' in f.f_locals:
141
# Use the supplement defined in the function.
142
tbs = f.f_locals['__traceback_supplement__']
143
elif '__traceback_supplement__' in f.f_globals:
144
# Use the supplement defined in the module.
145
tbs = f.f_globals['__traceback_supplement__']
151
supplement = try_except(factory, *args)
152
if supplement is not None:
153
# It might be nice if supplements could be dicts, for simplicity.
154
# Historically, though, they are objects.
155
# We will turn the supplement into a dict of strings, so that we
156
# have "getInfo" pre-processed and so that we are not holding on
157
# to anything from the frame.
159
getInfo = getattr(supplement, 'getInfo', None)
160
if getInfo is not None:
161
extra = try_except(getInfo)
162
extra = try_except(str, extra) if extra is not None else None
164
# The outer try-except is for the iteration.
166
for warning in getattr(supplement, 'warnings', ()):
167
if warning is not None:
168
warning = try_except(str, warning)
169
if warning is not None:
170
warnings.append(warning)
171
except EXPLOSIVE_ERRORS:
174
if DEBUG_EXCEPTION_FORMATTER:
175
traceback.print_exc()
176
supplement_dict = dict(warnings=warnings, extra=extra)
177
for key in ('source_url', 'line', 'column', 'expression'):
178
value = getattr(supplement, key, None)
179
if value is not None:
180
value = try_except(str, value)
181
supplement_dict[key] = value
182
info = f.f_locals.get('__traceback_info__', None)
183
info = try_except(str, info) if info is not None else None
184
# End part adapted from zope.exceptions.
186
filename = co.co_filename
188
linecache.checkcache(filename)
189
line = linecache.getline(filename, lineno, f.f_globals)
194
return (filename, lineno, name, line, modname, supplement_dict, info)
197
def extract_stack(f=None, limit=None):
198
"""Extract the raw traceback from the current stack frame.
200
The return value has the same format as for extract_tb(). The optional
201
'f' and 'limit' arguments have the same meaning as for print_stack().
202
Each item in the list is a septuple (filename, line number, function name,
203
text, module name, optional supplement dict, optional info string), and
204
the entries are in order from oldest to newest stack frame.
207
limit = _get_limit(limit)
210
get_frame_data = _get_frame_data # This is a micro-optimization.
211
while f is not None and (limit is None or n < limit):
212
list.append(get_frame_data(f, f.f_lineno))
219
def extract_tb(tb, limit=None):
220
"""Return list of up to limit pre-processed entries from traceback.
222
This is useful for alternate formatting of stack traces. If 'limit' is
223
omitted or None, all entries are extracted. A pre-processed stack trace
224
entry is a sextuple (filename, line number, function name, text, module
225
name, optional supplement dict, optional info string) representing the
226
information that is printed for a stack trace. The text is a string with
227
leading and trailing whitespace stripped; if the source is not available
228
it is None. The supplement dict has keys 'source_url', 'line', 'column',
229
'expression', 'warnings' (an iterable), and 'extra', any of which may be
232
# zope.exceptions handles tracebacks. This function is implemented just
233
# to show how this module's patterns might be extended to tracebacks.
234
limit = _get_limit(limit)
237
get_frame_data = _get_frame_data # This is a micro-optimization.
238
while tb is not None and (limit is None or n < limit):
239
list.append(get_frame_data(tb.tb_frame, tb.tb_lineno))