/brz/remove-bazaar

To get this branch, use:
bzr branch http://gegoxaren.bato24.eu/bzr/brz/remove-bazaar

« back to all changes in this revision

Viewing changes to bzrlib/crash.py

  • Committer: Martin Pool
  • Date: 2009-08-20 04:45:48 UTC
  • mto: This revision was merged to the branch mainline in revision 4632.
  • Revision ID: mbp@sourcefrog.net-20090820044548-r1jgiv8golcl4auh
further tweaks to and tests of bzr apport reporting

Show diffs side-by-side

added added

removed removed

Lines of Context:
1
 
# Copyright (C) 2009, 2010 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 it's /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
 
import platform
48
25
import pprint
49
26
import sys
50
27
import time
61
38
 
62
39
 
63
40
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
 
41
    if 'no_apport' not in debug.debug_flags:
 
42
        try:
 
43
            report_bug_to_apport(exc_info, stderr)
70
44
            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)
 
45
        except Exception, e:
 
46
            # this should only happen if apport is installed but it didn't
 
47
            # work, eg because of an io error writing the crash file
 
48
            sys.stderr.write("bzr: failed to report crash using apport:\n "
 
49
                "    %r\n" % e)
 
50
            pass
 
51
    report_bug_legacy(exc_info, stderr)
81
52
 
82
53
 
83
54
def report_bug_legacy(exc_info, err_file):
87
58
    err_file.write('bzr %s on python %s (%s)\n' % \
88
59
                       (bzrlib.__version__,
89
60
                        bzrlib._format_version_tuple(sys.version_info),
90
 
                        platform.platform(aliased=1)))
 
61
                        sys.platform))
91
62
    err_file.write('arguments: %r\n' % sys.argv)
92
63
    err_file.write(
93
64
        'encoding: %r, fsenc: %r, lang: %r\n' % (
106
77
 
107
78
def report_bug_to_apport(exc_info, stderr):
108
79
    """Report a bug to apport for optional automatic filing.
109
 
 
110
 
    :returns: The name of the crash file, or None if we didn't write one.
 
80
    
 
81
    :returns: True if the bug was filed or otherwise handled; 
 
82
        False to use a fallback method.
111
83
    """
112
 
    # this function is based on apport_package_hook.py, but omitting some of the
 
84
    # this is based on apport_package_hook.py, but omitting some of the
113
85
    # Ubuntu-specific policy about what to report and when
114
 
 
115
 
    # if the import fails, the exception will be caught at a higher level and
116
 
    # we'll report the error by other means
117
 
    import apport
118
 
 
119
 
    crash_filename = _write_apport_report_to_file(exc_info)
120
 
 
121
 
    if crash_filename is None:
122
 
        stderr.write("\n"
123
 
            "apport is set to ignore crashes in this version of bzr.\n"
124
 
            )
125
 
    else:
126
 
        trace.print_exception(exc_info, stderr)
127
 
        stderr.write("\n"
128
 
            "You can report this problem to Bazaar's developers by running\n"
129
 
            "    apport-bug %s\n"
130
 
            "if a bug-reporting window does not automatically appear.\n"
131
 
            % (crash_filename))
132
 
        # XXX: on Windows, Mac, and other platforms where we might have the
133
 
        # apport libraries but not have an apport always running, we could
134
 
        # synchronously file now
135
 
 
136
 
    return crash_filename
137
 
 
138
 
 
139
 
def _write_apport_report_to_file(exc_info):
 
86
    try:
 
87
        from apport.report import Report
 
88
    except ImportError, e:
 
89
        trace.warning("couldn't find apport bug-reporting library: %s" % e)
 
90
        return False
 
91
 
 
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
    return True
 
112
 
 
113
 
 
114
def _write_apport_report_to_file(exc_info, crash_file):
 
115
    import platform
140
116
    import traceback
141
117
    from apport.report import Report
142
118
 
143
119
    exc_type, exc_object, exc_tb = exc_info
144
120
 
145
121
    pr = Report()
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.
149
 
    pr.add_proc_info()
 
122
    # add_proc_info gives you the memory map of the process: this seems rarely
 
123
    # useful for Bazaar and it does make the report harder to scan, though it
 
124
    # does tell you what binary modules are loaded.
 
125
    # pr.add_proc_info()
150
126
    pr.add_user_info()
151
 
 
152
 
    # Package and SourcePackage are needed so that apport will report about even
153
 
    # non-packaged versions of bzr; also this reports on their packaged
154
 
    # dependencies which is useful.
155
 
    pr['SourcePackage'] = 'bzr'
156
 
    pr['Package'] = 'bzr'
157
 
 
158
127
    pr['CommandLine'] = pprint.pformat(sys.argv)
159
128
    pr['BzrVersion'] = bzrlib.__version__
160
129
    pr['PythonVersion'] = bzrlib._format_version_tuple(sys.version_info)
166
135
    pr['PythonLoadedModules'] = _format_module_list()
167
136
    pr['BzrDebugFlags'] = pprint.pformat(debug.debug_flags)
168
137
 
169
 
    # actually we'd rather file directly against the upstream product, but
170
 
    # apport does seem to count on there being one in there; we might need to
171
 
    # redirect it elsewhere anyhow
172
 
    pr['SourcePackage'] = 'bzr'
173
 
    pr['Package'] = 'bzr'
174
 
 
175
 
    # tell apport to file directly against the bzr package using 
176
 
    # <https://bugs.edge.launchpad.net/bzr/+bug/391015>
177
 
    #
178
 
    # XXX: unfortunately apport may crash later if the crashdb definition
179
 
    # file isn't present
180
 
    pr['CrashDb'] = 'bzr'
181
 
 
182
138
    tb_file = StringIO()
183
139
    traceback.print_exception(exc_type, exc_object, exc_tb, file=tb_file)
184
140
    pr['Traceback'] = tb_file.getvalue()
185
141
 
186
 
    _attach_log_tail(pr)
187
 
 
188
 
    # We want to use the 'bzr' crashdb so that it gets sent directly upstream,
189
 
    # which is a reasonable default for most internal errors.  However, if we
190
 
    # 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
192
 
    # source hook telling crashes to go to this crashdb, and a crashdb
193
 
    # configuration describing it.
194
 
 
195
 
    # these may contain some sensitive info (smtp_passwords)
196
 
    # TODO: strip that out and attach the rest
197
 
    #
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')
202
 
    
203
 
    # strip username, hostname, etc
204
 
    pr.anonymize()
205
 
 
206
 
    if pr.check_ignored():
207
 
        # eg configured off in ~/.apport-ignore.xml
208
 
        return None
209
 
    else:
210
 
        crash_file_name, crash_file = _open_crash_file()
211
 
        pr.write(crash_file)
212
 
        crash_file.close()
213
 
        return crash_file_name
214
 
 
215
 
 
216
 
def _attach_log_tail(pr):
217
 
    try:
218
 
        bzr_log = open(trace._get_bzr_log_filename(), 'rt')
219
 
    except (IOError, OSError), e:
220
 
        pr['BzrLogTail'] = repr(e)
221
 
        return
222
 
    try:
223
 
        lines = bzr_log.readlines()
224
 
        pr['BzrLogTail'] = ''.join(lines[-40:])
225
 
    finally:
226
 
        bzr_log.close()
 
142
    pr.write(crash_file)
227
143
 
228
144
 
229
145
def _open_crash_file():
230
146
    crash_dir = config.crash_dir()
 
147
    # user-readable only, just in case the contents are sensitive.
231
148
    if not osutils.isdir(crash_dir):
232
 
        # on unix this should be /var/crash and should already exist; on
233
 
        # Windows or if it's manually configured it might need to be created,
234
 
        # and then it should be private
235
 
        os.makedirs(crash_dir, mode=0600)
236
 
    date_string = time.strftime('%Y-%m-%dT%H:%M', time.gmtime())
237
 
    # XXX: getuid doesn't work on win32, but the crash directory is per-user
238
 
    if sys.platform == 'win32':
239
 
        user_part = ''
240
 
    else:
241
 
        user_part = '.%d' % os.getuid()
242
 
    filename = osutils.pathjoin(
243
 
        crash_dir,
244
 
        'bzr%s.%s.crash' % (
245
 
            user_part,
246
 
            date_string))
247
 
    # be careful here that people can't play tmp-type symlink mischief in the
248
 
    # world-writable directory
249
 
    return filename, os.fdopen(
250
 
        os.open(filename, 
251
 
            os.O_WRONLY|os.O_CREAT|os.O_EXCL,
252
 
            0600),
253
 
        'w')
 
149
        os.makedirs(crash_dir, mode=0700)
 
150
    filename = 'bzr-%s-%s.crash' % (
 
151
        osutils.compact_date(time.time()),
 
152
        os.getpid(),)
 
153
    return open(osutils.pathjoin(crash_dir, filename), 'wt')
254
154
 
255
155
 
256
156
def _format_plugin_list():