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