~bzr-pqm/bzr/bzr.dev

« back to all changes in this revision

Viewing changes to bzrlib/crash.py

  • Committer: Canonical.com Patch Queue Manager
  • Date: 2010-01-20 18:55:04 UTC
  • mfrom: (4971.2.2 505762)
  • Revision ID: pqm@pqm.ubuntu.com-20100120185504-es1x5ntwauunwxvp
(nmb) Explain bound branches in "branches" help topic

Show diffs side-by-side

added added

removed removed

Lines of Context:
1
 
# Copyright (C) 2009-2011 Canonical Ltd
 
1
# Copyright (C) 2009 Canonical Ltd
2
2
#
3
3
# This program is free software; you can redistribute it and/or modify
4
4
# it under the terms of the GNU General Public License as published by
16
16
 
17
17
 
18
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
23
 
crash using apport into its /var/crash spool directory, from where the user
24
 
can either manually send it to Launchpad.  In some cases (at least Ubuntu
25
 
development releases), Apport may pop up a window asking if they want
26
 
to send it.
27
 
 
28
 
Without apport, we just write a crash report to stderr and the user can report
29
 
this manually if the wish.
30
 
 
31
 
We never send crash data across the network without user opt-in.
32
 
 
33
 
In principle apport can run on any platform though as of Feb 2010 there seem
34
 
to be some portability bugs.
35
 
 
36
 
To force this off in bzr turn set APPORT_DISABLE in the environment or 
37
 
-Dno_apport.
38
19
"""
39
20
 
40
21
# for interactive testing, try the 'bzr assert-fail' command 
41
22
# or see http://code.launchpad.net/~mbp/bzr/bzr-fail
42
 
#
43
 
# to test with apport it's useful to set
44
 
# export APPORT_IGNORE_OBSOLETE_PACKAGES=1
45
23
 
46
24
import os
47
25
import platform
61
39
 
62
40
 
63
41
def report_bug(exc_info, stderr):
64
 
    if ('no_apport' in debug.debug_flags) or \
65
 
        os.environ.get('APPORT_DISABLE', None):
66
 
        return report_bug_legacy(exc_info, stderr)
67
 
    try:
68
 
        if report_bug_to_apport(exc_info, stderr):
69
 
            # wrote a file; if None then report the old way
 
42
    if 'no_apport' not in debug.debug_flags:
 
43
        try:
 
44
            report_bug_to_apport(exc_info, stderr)
70
45
            return
71
 
    except ImportError, e:
72
 
        trace.mutter("couldn't find apport bug-reporting library: %s" % e)
73
 
        pass
74
 
    except Exception, e:
75
 
        # this should only happen if apport is installed but it didn't
76
 
        # work, eg because of an io error writing the crash file
77
 
        stderr.write("bzr: failed to report crash using apport:\n "
78
 
            "    %r\n" % e)
79
 
        pass
80
 
    return report_bug_legacy(exc_info, stderr)
 
46
        except ImportError, e:
 
47
            trace.mutter("couldn't find apport bug-reporting library: %s" % e)
 
48
            pass
 
49
        except Exception, e:
 
50
            # this should only happen if apport is installed but it didn't
 
51
            # work, eg because of an io error writing the crash file
 
52
            sys.stderr.write("bzr: failed to report crash using apport:\n "
 
53
                "    %r\n" % e)
 
54
            pass
 
55
    report_bug_legacy(exc_info, stderr)
81
56
 
82
57
 
83
58
def report_bug_legacy(exc_info, err_file):
84
59
    """Report a bug by just printing a message to the user."""
85
60
    trace.print_exception(exc_info, err_file)
86
61
    err_file.write('\n')
87
 
    import textwrap
88
 
    def print_wrapped(l):
89
 
        err_file.write(textwrap.fill(l,
90
 
            width=78, subsequent_indent='    ') + '\n')
91
 
    print_wrapped('bzr %s on python %s (%s)\n' % \
92
 
        (bzrlib.__version__,
93
 
        bzrlib._format_version_tuple(sys.version_info),
94
 
        platform.platform(aliased=1)))
95
 
    print_wrapped('arguments: %r\n' % sys.argv)
96
 
    print_wrapped(textwrap.fill(
97
 
        'plugins: ' + plugin.format_concise_plugin_list(),
98
 
        width=78,
99
 
        subsequent_indent='    ',
100
 
        ) + '\n')
101
 
    print_wrapped(
 
62
    err_file.write('bzr %s on python %s (%s)\n' % \
 
63
                       (bzrlib.__version__,
 
64
                        bzrlib._format_version_tuple(sys.version_info),
 
65
                        platform.platform(aliased=1)))
 
66
    err_file.write('arguments: %r\n' % sys.argv)
 
67
    err_file.write(
102
68
        'encoding: %r, fsenc: %r, lang: %r\n' % (
103
69
            osutils.get_user_encoding(), sys.getfilesystemencoding(),
104
70
            os.environ.get('LANG')))
105
 
    # We used to show all the plugins here, but it's too verbose.
 
71
    err_file.write("plugins:\n")
 
72
    err_file.write(_format_plugin_list())
106
73
    err_file.write(
107
 
        "\n"
 
74
        "\n\n"
108
75
        "*** Bazaar has encountered an internal error.  This probably indicates a\n"
109
76
        "    bug in Bazaar.  You can help us fix it by filing a bug report at\n"
110
77
        "        https://bugs.launchpad.net/bzr/+filebug\n"
114
81
 
115
82
def report_bug_to_apport(exc_info, stderr):
116
83
    """Report a bug to apport for optional automatic filing.
117
 
 
118
 
    :returns: The name of the crash file, or None if we didn't write one.
119
84
    """
120
 
    # this function is based on apport_package_hook.py, but omitting some of the
 
85
    # this is based on apport_package_hook.py, but omitting some of the
121
86
    # Ubuntu-specific policy about what to report and when
122
87
 
123
 
    # This import is apparently not used, but we're doing it so that if the
124
 
    # import fails, the exception will be caught at a higher level and we'll
125
 
    # report the error by other means.
 
88
    # if this fails its caught at a higher level; we don't want to open the
 
89
    # crash file unless apport can be loaded.
126
90
    import apport
127
91
 
128
 
    crash_filename = _write_apport_report_to_file(exc_info)
129
 
 
130
 
    if crash_filename is None:
131
 
        stderr.write("\n"
132
 
            "apport is set to ignore crashes in this version of bzr.\n"
133
 
            )
134
 
    else:
135
 
        trace.print_exception(exc_info, stderr)
136
 
        stderr.write("\n"
137
 
            "You can report this problem to Bazaar's developers by running\n"
138
 
            "    apport-bug %s\n"
139
 
            "if a bug-reporting window does not automatically appear.\n"
140
 
            % (crash_filename))
141
 
        # XXX: on Windows, Mac, and other platforms where we might have the
142
 
        # apport libraries but not have an apport always running, we could
143
 
        # synchronously file now
144
 
 
145
 
    return crash_filename
146
 
 
147
 
 
148
 
def _write_apport_report_to_file(exc_info):
 
92
    crash_file = _open_crash_file()
 
93
    try:
 
94
        _write_apport_report_to_file(exc_info, crash_file)
 
95
    finally:
 
96
        crash_file.close()
 
97
 
 
98
    stderr.write("bzr: ERROR: %s.%s: %s\n" 
 
99
        "\n"
 
100
        "*** Bazaar has encountered an internal error.  This probably indicates a\n"
 
101
        "    bug in Bazaar.  You can help us fix it by filing a bug report at\n"
 
102
        "        https://bugs.launchpad.net/bzr/+filebug\n"
 
103
        "    attaching the crash file\n"
 
104
        "        %s\n"
 
105
        "    and including a description of the problem.\n"
 
106
        "\n"
 
107
        "    The crash file is plain text and you can inspect or edit it to remove\n"
 
108
        "    private information.\n"
 
109
        % (exc_info[0].__module__, exc_info[0].__name__, exc_info[1],
 
110
           crash_file.name))
 
111
 
 
112
 
 
113
def _write_apport_report_to_file(exc_info, crash_file):
149
114
    import traceback
150
115
    from apport.report import Report
151
116
 
152
117
    exc_type, exc_object, exc_tb = exc_info
153
118
 
154
119
    pr = Report()
155
 
    # add_proc_info sets the ExecutablePath, InterpreterPath, etc.
156
 
    pr.add_proc_info()
157
 
    # It also adds ProcMaps which for us is rarely useful and mostly noise, so
158
 
    # let's remove it.
159
 
    del pr['ProcMaps']
 
120
    # add_proc_info gives you the memory map of the process: this seems rarely
 
121
    # useful for Bazaar and it does make the report harder to scan, though it
 
122
    # does tell you what binary modules are loaded.
 
123
    # pr.add_proc_info()
160
124
    pr.add_user_info()
161
 
 
162
 
    # Package and SourcePackage are needed so that apport will report about even
163
 
    # non-packaged versions of bzr; also this reports on their packaged
164
 
    # dependencies which is useful.
165
 
    pr['SourcePackage'] = 'bzr'
166
 
    pr['Package'] = 'bzr'
167
 
 
168
125
    pr['CommandLine'] = pprint.pformat(sys.argv)
169
126
    pr['BzrVersion'] = bzrlib.__version__
170
127
    pr['PythonVersion'] = bzrlib._format_version_tuple(sys.version_info)
176
133
    pr['PythonLoadedModules'] = _format_module_list()
177
134
    pr['BzrDebugFlags'] = pprint.pformat(debug.debug_flags)
178
135
 
179
 
    # actually we'd rather file directly against the upstream product, but
180
 
    # apport does seem to count on there being one in there; we might need to
181
 
    # redirect it elsewhere anyhow
182
 
    pr['SourcePackage'] = 'bzr'
183
 
    pr['Package'] = 'bzr'
184
 
 
185
 
    # tell apport to file directly against the bzr package using 
186
 
    # <https://bugs.launchpad.net/bzr/+bug/391015>
187
 
    #
188
 
    # XXX: unfortunately apport may crash later if the crashdb definition
189
 
    # file isn't present
190
 
    pr['CrashDb'] = 'bzr'
191
 
 
192
136
    tb_file = StringIO()
193
137
    traceback.print_exception(exc_type, exc_object, exc_tb, file=tb_file)
194
138
    pr['Traceback'] = tb_file.getvalue()
195
139
 
196
 
    _attach_log_tail(pr)
197
 
 
198
 
    # We want to use the 'bzr' crashdb so that it gets sent directly upstream,
199
 
    # which is a reasonable default for most internal errors.  However, if we
200
 
    # set it here then apport will crash later if it doesn't know about that
201
 
    # crashdb.  Instead, we rely on the bzr package installing both a
202
 
    # source hook telling crashes to go to this crashdb, and a crashdb
203
 
    # configuration describing it.
204
 
 
205
 
    # these may contain some sensitive info (smtp_passwords)
206
 
    # TODO: strip that out and attach the rest
207
 
    #
208
 
    #attach_file_if_exists(report,
209
 
    #   os.path.join(dot_bzr, 'bazaar.conf', 'BzrConfig')
210
 
    #attach_file_if_exists(report,
211
 
    #   os.path.join(dot_bzr, 'locations.conf', 'BzrLocations')
212
 
    
213
 
    # strip username, hostname, etc
214
 
    pr.anonymize()
215
 
 
216
 
    if pr.check_ignored():
217
 
        # eg configured off in ~/.apport-ignore.xml
218
 
        return None
219
 
    else:
220
 
        crash_file_name, crash_file = _open_crash_file()
221
 
        pr.write(crash_file)
222
 
        crash_file.close()
223
 
        return crash_file_name
224
 
 
225
 
 
226
 
def _attach_log_tail(pr):
227
 
    try:
228
 
        bzr_log = open(trace._get_bzr_log_filename(), 'rt')
229
 
    except (IOError, OSError), e:
230
 
        pr['BzrLogTail'] = repr(e)
231
 
        return
232
 
    try:
233
 
        lines = bzr_log.readlines()
234
 
        pr['BzrLogTail'] = ''.join(lines[-40:])
235
 
    finally:
236
 
        bzr_log.close()
 
140
    pr.write(crash_file)
237
141
 
238
142
 
239
143
def _open_crash_file():
240
144
    crash_dir = config.crash_dir()
 
145
    # user-readable only, just in case the contents are sensitive.
241
146
    if not osutils.isdir(crash_dir):
242
 
        # on unix this should be /var/crash and should already exist; on
243
 
        # Windows or if it's manually configured it might need to be created,
244
 
        # and then it should be private
245
 
        os.makedirs(crash_dir, mode=0600)
246
 
    date_string = time.strftime('%Y-%m-%dT%H:%M', time.gmtime())
247
 
    # XXX: getuid doesn't work on win32, but the crash directory is per-user
248
 
    if sys.platform == 'win32':
249
 
        user_part = ''
250
 
    else:
251
 
        user_part = '.%d' % os.getuid()
252
 
    filename = osutils.pathjoin(
253
 
        crash_dir,
254
 
        'bzr%s.%s.crash' % (
255
 
            user_part,
256
 
            date_string))
257
 
    # be careful here that people can't play tmp-type symlink mischief in the
258
 
    # world-writable directory
259
 
    return filename, os.fdopen(
260
 
        os.open(filename, 
261
 
            os.O_WRONLY|os.O_CREAT|os.O_EXCL,
262
 
            0600),
263
 
        'w')
 
147
        os.makedirs(crash_dir, mode=0700)
 
148
    filename = 'bzr-%s-%s.crash' % (
 
149
        osutils.compact_date(time.time()),
 
150
        os.getpid(),)
 
151
    return open(osutils.pathjoin(crash_dir, filename), 'wt')
264
152
 
265
153
 
266
154
def _format_plugin_list():
267
 
    return ''.join(plugin.describe_plugins(show_paths=True))
 
155
    plugin_lines = []
 
156
    for name, a_plugin in sorted(plugin.plugins().items()):
 
157
        plugin_lines.append("  %-20s %s [%s]" %
 
158
            (name, a_plugin.path(), a_plugin.__version__))
 
159
    return '\n'.join(plugin_lines)
268
160
 
269
161
 
270
162
def _format_module_list():