14
14
# along with this program; if not, write to the Free Software
15
15
# Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
17
18
"""Handling and reporting crashes.
19
20
A crash is an exception propagated up almost to the top level of Bazaar.
21
22
If we have apport <https://launchpad.net/apport/>, we store a report of the
22
crash using apport into its /var/crash spool directory, from where the user
23
crash using apport into it's /var/crash spool directory, from where the user
23
24
can either manually send it to Launchpad. In some cases (at least Ubuntu
24
25
development releases), Apport may pop up a window asking if they want
32
33
In principle apport can run on any platform though as of Feb 2010 there seem
33
34
to be some portability bugs.
35
To force this off in brz turn set APPORT_DISABLE in the environment or
36
To force this off in bzr turn set APPORT_DISABLE in the environment or
39
from __future__ import absolute_import
41
# for interactive testing, try the 'brz assert-fail' command
40
# for interactive testing, try the 'bzr assert-fail' command
42
41
# or see http://code.launchpad.net/~mbp/bzr/bzr-fail
44
43
# to test with apport it's useful to set
51
from StringIO import StringIO
66
63
def report_bug(exc_info, stderr):
67
64
if ('no_apport' in debug.debug_flags) or \
68
os.environ.get('APPORT_DISABLE', None):
65
os.environ.get('APPORT_DISABLE', None):
69
66
return report_bug_legacy(exc_info, stderr)
71
68
if report_bug_to_apport(exc_info, stderr):
72
69
# wrote a file; if None then report the old way
74
except ImportError as e:
71
except ImportError, e:
75
72
trace.mutter("couldn't find apport bug-reporting library: %s" % e)
76
except Exception as e:
77
75
# this should only happen if apport is installed but it didn't
78
76
# work, eg because of an io error writing the crash file
79
trace.mutter("brz: failed to report crash using apport: %r" % e)
80
trace.log_exception_quietly()
77
stderr.write("bzr: failed to report crash using apport:\n "
81
80
return report_bug_legacy(exc_info, stderr)
85
84
"""Report a bug by just printing a message to the user."""
86
85
trace.print_exception(exc_info, err_file)
87
86
err_file.write('\n')
91
err_file.write(textwrap.fill(
92
l, width=78, subsequent_indent=' ') + '\n')
93
print_wrapped('brz %s on python %s (%s)\n' %
95
breezy._format_version_tuple(sys.version_info),
96
platform.platform(aliased=1)))
97
print_wrapped('arguments: %r\n' % sys.argv)
98
print_wrapped(textwrap.fill(
99
'plugins: ' + plugin.format_concise_plugin_list(),
101
subsequent_indent=' ',
87
err_file.write('bzr %s on python %s (%s)\n' % \
89
bzrlib._format_version_tuple(sys.version_info),
90
platform.platform(aliased=1)))
91
err_file.write('arguments: %r\n' % sys.argv)
104
93
'encoding: %r, fsenc: %r, lang: %r\n' % (
105
94
osutils.get_user_encoding(), sys.getfilesystemencoding(),
106
95
os.environ.get('LANG')))
107
# We used to show all the plugins here, but it's too verbose.
96
err_file.write("plugins:\n")
97
err_file.write(_format_plugin_list())
110
100
"*** Bazaar has encountered an internal error. This probably indicates a\n"
111
101
" bug in Bazaar. You can help us fix it by filing a bug report at\n"
112
" https://bugs.launchpad.net/brz/+filebug\n"
102
" https://bugs.launchpad.net/bzr/+filebug\n"
113
103
" including this traceback and a description of the problem.\n"
122
112
# this function is based on apport_package_hook.py, but omitting some of the
123
113
# Ubuntu-specific policy about what to report and when
125
# This import is apparently not used, but we're doing it so that if the
126
# import fails, the exception will be caught at a higher level and we'll
127
# report the error by other means.
128
import apport # noqa: F401
115
# if the import fails, the exception will be caught at a higher level and
116
# we'll report the error by other means
130
119
crash_filename = _write_apport_report_to_file(exc_info)
132
121
if crash_filename is None:
133
122
stderr.write("\n"
134
"apport is set to ignore crashes in this version of brz.\n"
123
"apport is set to ignore crashes in this version of bzr.\n"
137
126
trace.print_exception(exc_info, stderr)
138
127
stderr.write("\n"
139
"You can report this problem to Bazaar's developers by running\n"
141
"if a bug-reporting window does not automatically appear.\n"
128
"You can report this problem to Bazaar's developers by running\n"
130
"if a bug-reporting window does not automatically appear.\n"
143
132
# XXX: on Windows, Mac, and other platforms where we might have the
144
133
# apport libraries but not have an apport always running, we could
145
134
# synchronously file now
154
143
exc_type, exc_object, exc_tb = exc_info
157
# add_proc_info sets the ExecutablePath, InterpreterPath, etc.
146
# add_proc_info gives you the memory map of the process, which is not so
147
# useful for Bazaar but does tell you what binary libraries are loaded.
148
# More importantly it sets the ExecutablePath, InterpreterPath, etc.
158
149
pr.add_proc_info()
159
# It also adds ProcMaps which for us is rarely useful and mostly noise, so
162
150
pr.add_user_info()
164
152
# Package and SourcePackage are needed so that apport will report about even
165
# non-packaged versions of brz; also this reports on their packaged
153
# non-packaged versions of bzr; also this reports on their packaged
166
154
# dependencies which is useful.
167
pr['SourcePackage'] = 'brz'
168
pr['Package'] = 'brz'
155
pr['SourcePackage'] = 'bzr'
156
pr['Package'] = 'bzr'
170
158
pr['CommandLine'] = pprint.pformat(sys.argv)
171
pr['BrzVersion'] = breezy.__version__
172
pr['PythonVersion'] = breezy._format_version_tuple(sys.version_info)
159
pr['BzrVersion'] = bzrlib.__version__
160
pr['PythonVersion'] = bzrlib._format_version_tuple(sys.version_info)
173
161
pr['Platform'] = platform.platform(aliased=1)
174
162
pr['UserEncoding'] = osutils.get_user_encoding()
175
163
pr['FileSystemEncoding'] = sys.getfilesystemencoding()
176
pr['Locale'] = os.environ.get('LANG', 'C')
177
pr['BrzPlugins'] = _format_plugin_list()
164
pr['Locale'] = os.environ.get('LANG')
165
pr['BzrPlugins'] = _format_plugin_list()
178
166
pr['PythonLoadedModules'] = _format_module_list()
179
pr['BrzDebugFlags'] = pprint.pformat(debug.debug_flags)
167
pr['BzrDebugFlags'] = pprint.pformat(debug.debug_flags)
181
169
# actually we'd rather file directly against the upstream product, but
182
170
# apport does seem to count on there being one in there; we might need to
183
171
# redirect it elsewhere anyhow
184
pr['SourcePackage'] = 'brz'
185
pr['Package'] = 'brz'
172
pr['SourcePackage'] = 'bzr'
173
pr['Package'] = 'bzr'
187
# tell apport to file directly against the brz package using
175
# tell apport to file directly against the bzr package using
188
176
# <https://bugs.launchpad.net/bzr/+bug/391015>
190
178
# XXX: unfortunately apport may crash later if the crashdb definition
191
179
# file isn't present
192
pr['CrashDb'] = 'brz'
180
pr['CrashDb'] = 'bzr'
194
182
tb_file = StringIO()
195
183
traceback.print_exception(exc_type, exc_object, exc_tb, file=tb_file)
198
186
_attach_log_tail(pr)
200
# We want to use the 'brz' crashdb so that it gets sent directly upstream,
188
# We want to use the 'bzr' crashdb so that it gets sent directly upstream,
201
189
# which is a reasonable default for most internal errors. However, if we
202
190
# set it here then apport will crash later if it doesn't know about that
203
# crashdb. Instead, we rely on the brz package installing both a
191
# crashdb. Instead, we rely on the bzr package installing both a
204
192
# source hook telling crashes to go to this crashdb, and a crashdb
205
193
# configuration describing it.
207
195
# these may contain some sensitive info (smtp_passwords)
208
196
# TODO: strip that out and attach the rest
210
# attach_file_if_exists(report,
211
# os.path.join(dot_brz, 'breezy.conf', 'BrzConfig')
212
# attach_file_if_exists(report,
213
# os.path.join(dot_brz, 'locations.conf', 'BrzLocations')
198
#attach_file_if_exists(report,
199
# os.path.join(dot_bzr, 'bazaar.conf', 'BzrConfig')
200
#attach_file_if_exists(report,
201
# os.path.join(dot_bzr, 'locations.conf', 'BzrLocations')
215
203
# strip username, hostname, etc
228
216
def _attach_log_tail(pr):
230
brz_log = open(trace._get_brz_log_filename(), 'rt')
231
except (IOError, OSError) as e:
232
pr['BrzLogTail'] = repr(e)
218
bzr_log = open(trace._get_bzr_log_filename(), 'rt')
219
except (IOError, OSError), e:
220
pr['BzrLogTail'] = repr(e)
235
lines = brz_log.readlines()
236
pr['BrzLogTail'] = ''.join(lines[-40:])
223
lines = bzr_log.readlines()
224
pr['BzrLogTail'] = ''.join(lines[-40:])
241
229
def _open_crash_file():
242
crash_dir = bedding.crash_dir()
230
crash_dir = config.crash_dir()
243
231
if not osutils.isdir(crash_dir):
244
232
# on unix this should be /var/crash and should already exist; on
245
233
# Windows or if it's manually configured it might need to be created,
246
234
# and then it should be private
247
os.makedirs(crash_dir, mode=0o600)
235
os.makedirs(crash_dir, mode=0600)
248
236
date_string = time.strftime('%Y-%m-%dT%H:%M', time.gmtime())
249
237
# XXX: getuid doesn't work on win32, but the crash directory is per-user
250
238
if sys.platform == 'win32':
253
241
user_part = '.%d' % os.getuid()
254
242
filename = osutils.pathjoin(
259
247
# be careful here that people can't play tmp-type symlink mischief in the
260
248
# world-writable directory
261
249
return filename, os.fdopen(
263
os.O_WRONLY | os.O_CREAT | os.O_EXCL,
251
os.O_WRONLY|os.O_CREAT|os.O_EXCL,
268
256
def _format_plugin_list():
269
return ''.join(plugin.describe_plugins(show_paths=True))
258
for name, a_plugin in sorted(plugin.plugins().items()):
259
plugin_lines.append(" %-20s %s [%s]" %
260
(name, a_plugin.path(), a_plugin.__version__))
261
return '\n'.join(plugin_lines)
272
264
def _format_module_list():