/brz/remove-bazaar

To get this branch, use:
bzr branch http://gegoxaren.bato24.eu/bzr/brz/remove-bazaar
4584.3.21 by Martin Pool
Start adding tests for apport
1
# Copyright (C) 2005, 2006, 2007, 2008, 2009 Canonical Ltd
1185.33.9 by Martin Pool
Add new selftest module.
2
#
3
# This program is free software; you can redistribute it and/or modify
4
# it under the terms of the GNU General Public License as published by
5
# the Free Software Foundation; either version 2 of the License, or
6
# (at your option) any later version.
7
#
8
# This program is distributed in the hope that it will be useful,
9
# but WITHOUT ANY WARRANTY; without even the implied warranty of
10
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
11
# GNU General Public License for more details.
12
#
13
# You should have received a copy of the GNU General Public License
14
# along with this program; if not, write to the Free Software
4183.7.1 by Sabin Iacob
update FSF mailing address
15
# Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
1185.33.9 by Martin Pool
Add new selftest module.
16
17
# "weren't nothing promised to you.  do i look like i got a promise face?"
18
19
"""Tests for trace library"""
20
1948.1.5 by John Arbash Meinel
Make sure BzrCommandError can handle unicode arguments
21
from cStringIO import StringIO
1740.5.7 by Martin Pool
Add test for formatting of EPIPE
22
import errno
1185.33.9 by Martin Pool
Add new selftest module.
23
import os
3195.1.1 by Andrew Bennetts
Always include timestamps in the trace file (i.e. remove -Dtimes in favour of having it switched on permanently)
24
import re
1185.33.9 by Martin Pool
Add new selftest module.
25
import sys
3173.1.12 by Martin Pool
Add test_push_log_file
26
import tempfile
1185.33.9 by Martin Pool
Add new selftest module.
27
1948.1.5 by John Arbash Meinel
Make sure BzrCommandError can handle unicode arguments
28
from bzrlib import (
29
    errors,
4634.118.1 by John Arbash Meinel
Fix bug #503886, errors setting up logging go to stderr.
30
    trace,
1948.1.5 by John Arbash Meinel
Make sure BzrCommandError can handle unicode arguments
31
    )
1551.9.3 by Aaron Bentley
Revert buggy apport changes
32
from bzrlib.tests import TestCaseInTempDir, TestCase
2768.1.10 by Ian Clatworthy
Add tests for new methods in trace.py
33
from bzrlib.trace import (
34
    mutter, mutter_callsite, report_exception,
35
    set_verbosity_level, get_verbosity_level, is_quiet, is_verbose, be_quiet,
3173.1.12 by Martin Pool
Add test_push_log_file
36
    pop_log_file,
37
    push_log_file,
2851.3.1 by Martin Pool
Add unit test for _rollover_trace_maybe
38
    _rollover_trace_maybe,
2768.1.10 by Ian Clatworthy
Add tests for new methods in trace.py
39
    )
1185.33.9 by Martin Pool
Add new selftest module.
40
1740.5.2 by Martin Pool
Improved tests for display of exceptions.
41
42
def _format_exception():
43
    """Format an exception as it would normally be displayed to the user"""
44
    buf = StringIO()
1551.9.3 by Aaron Bentley
Revert buggy apport changes
45
    report_exception(sys.exc_info(), buf)
46
    return buf.getvalue()
1740.5.2 by Martin Pool
Improved tests for display of exceptions.
47
48
1185.33.9 by Martin Pool
Add new selftest module.
49
class TestTrace(TestCase):
1740.5.2 by Martin Pool
Improved tests for display of exceptions.
50
1551.9.3 by Aaron Bentley
Revert buggy apport changes
51
    def test_format_sys_exception(self):
4584.3.20 by Martin Pool
Tweak trace tests to cope without the traceback being printed
52
        # Test handling of an internal/unexpected error that probably
53
        # indicates a bug in bzr.  The details of the message may vary
54
        # depending on whether apport is available or not.  See test_crash for
55
        # more.
1185.33.9 by Martin Pool
Add new selftest module.
56
        try:
57
            raise NotImplementedError, "time travel"
58
        except NotImplementedError:
59
            pass
1551.9.3 by Aaron Bentley
Revert buggy apport changes
60
        err = _format_exception()
1740.5.2 by Martin Pool
Improved tests for display of exceptions.
61
        self.assertEqualDiff(err.splitlines()[0],
1740.5.3 by Martin Pool
Cleanup more exception-formatting code
62
                'bzr: ERROR: exceptions.NotImplementedError: time travel')
1740.5.2 by Martin Pool
Improved tests for display of exceptions.
63
        self.assertContainsRe(err,
4584.3.20 by Martin Pool
Tweak trace tests to cope without the traceback being printed
64
            'Bazaar has encountered an internal error.')
1185.33.9 by Martin Pool
Add new selftest module.
65
1740.5.3 by Martin Pool
Cleanup more exception-formatting code
66
    def test_format_interrupt_exception(self):
67
        try:
68
            raise KeyboardInterrupt()
1740.5.5 by Martin Pool
Show short form for OSError and IOError too
69
        except KeyboardInterrupt:
1740.5.3 by Martin Pool
Cleanup more exception-formatting code
70
            # XXX: Some risk that a *real* keyboard interrupt won't be seen
71
            pass
1551.9.3 by Aaron Bentley
Revert buggy apport changes
72
        msg = _format_exception()
1740.5.3 by Martin Pool
Cleanup more exception-formatting code
73
        self.assertTrue(len(msg) > 0)
74
        self.assertEqualDiff(msg, 'bzr: interrupted\n')
75
4634.26.1 by Martin Pool
Cleaner message when out of memory
76
    def test_format_memory_error(self):
77
        try:
78
            raise MemoryError()
79
        except MemoryError:
80
            pass
81
        msg = _format_exception()
82
        self.assertEquals(msg,
83
            "bzr: out of memory\n")
84
1740.5.5 by Martin Pool
Show short form for OSError and IOError too
85
    def test_format_os_error(self):
86
        try:
4095.1.1 by Martin Pool
Add more distinct tests for IOError and OSError
87
            os.rmdir('nosuchfile22222')
88
        except OSError:
89
            pass
90
        msg = _format_exception()
91
        self.assertContainsRe(msg,
92
            r'^bzr: ERROR: \[Errno .*\] No such file.*nosuchfile22222')
93
94
    def test_format_io_error(self):
95
        try:
1740.5.5 by Martin Pool
Show short form for OSError and IOError too
96
            file('nosuchfile22222')
4095.1.1 by Martin Pool
Add more distinct tests for IOError and OSError
97
        except IOError:
1740.5.5 by Martin Pool
Show short form for OSError and IOError too
98
            pass
1551.9.3 by Aaron Bentley
Revert buggy apport changes
99
        msg = _format_exception()
1740.5.5 by Martin Pool
Show short form for OSError and IOError too
100
        self.assertContainsRe(msg, r'^bzr: ERROR: \[Errno .*\] No such file.*nosuchfile')
101
1948.1.5 by John Arbash Meinel
Make sure BzrCommandError can handle unicode arguments
102
    def test_format_unicode_error(self):
103
        try:
104
            raise errors.BzrCommandError(u'argument foo\xb5 does not exist')
105
        except errors.BzrCommandError:
106
            pass
1551.9.3 by Aaron Bentley
Revert buggy apport changes
107
        msg = _format_exception()
1740.5.5 by Martin Pool
Show short form for OSError and IOError too
108
1185.33.9 by Martin Pool
Add new selftest module.
109
    def test_format_exception(self):
1740.5.2 by Martin Pool
Improved tests for display of exceptions.
110
        """Short formatting of bzr exceptions"""
1185.33.9 by Martin Pool
Add new selftest module.
111
        try:
2067.3.1 by Martin Pool
Clean up BzrNewError, other exception classes and users.
112
            raise errors.NotBranchError('wibble')
1948.1.5 by John Arbash Meinel
Make sure BzrCommandError can handle unicode arguments
113
        except errors.NotBranchError:
1185.33.9 by Martin Pool
Add new selftest module.
114
            pass
1551.9.3 by Aaron Bentley
Revert buggy apport changes
115
        msg = _format_exception()
1740.5.2 by Martin Pool
Improved tests for display of exceptions.
116
        self.assertTrue(len(msg) > 0)
2745.3.2 by Daniel Watkins
Updated tests to reflect new error text.
117
        self.assertEqualDiff(msg, 'bzr: ERROR: Not a branch: \"wibble\".\n')
1185.33.63 by Martin Pool
Better display of BzrError classes that are not BzrNewErrors.
118
3497.3.2 by Martin Pool
Show short error for missing libraries
119
    def test_report_external_import_error(self):
120
        """Short friendly message for missing system modules."""
121
        try:
122
            import ImaginaryModule
123
        except ImportError, e:
124
            pass
125
        else:
126
            self.fail("somehow succeeded in importing %r" % ImaginaryModule)
127
        msg = _format_exception()
128
        self.assertEqual(msg,
129
            'bzr: ERROR: No module named ImaginaryModule\n'
130
            'You may need to install this Python library separately.\n')
131
132
    def test_report_import_syntax_error(self):
133
        try:
134
            raise ImportError("syntax error")
135
        except ImportError, e:
136
            pass
137
        msg = _format_exception()
138
        self.assertContainsRe(msg,
4584.3.20 by Martin Pool
Tweak trace tests to cope without the traceback being printed
139
            r'Bazaar has encountered an internal error')
3497.3.2 by Martin Pool
Show short error for missing libraries
140
1185.33.51 by Martin Pool
Fix trace of non-ascii messages, and add test.
141
    def test_trace_unicode(self):
142
        """Write Unicode to trace log"""
143
        self.log(u'the unicode character for benzene is \N{BENZENE RING}')
1927.3.1 by Carl Friedrich Bolz
Throw away on-disk logfile when possible.
144
        self.assertContainsRe(self._get_log(keep_log_file=True),
145
                              "the unicode character for benzene is")
3943.8.1 by Marius Kruger
remove all trailing whitespace from bzr source
146
1948.1.2 by John Arbash Meinel
Fix the test_trace functions to actually test that things are written to the log
147
    def test_trace_argument_unicode(self):
148
        """Write a Unicode argument to the trace log"""
149
        mutter(u'the unicode character for benzene is %s', u'\N{BENZENE RING}')
1927.3.4 by Carl Friedrich Bolz
Merge bzr.dev.
150
        self.assertContainsRe(self._get_log(keep_log_file=True),
151
                              'the unicode character')
1185.85.5 by John Arbash Meinel
mutter() should not fail because of unicode errors
152
1948.1.3 by John Arbash Meinel
Fix mutter() so even if args are invalid, it still works
153
    def test_trace_argument_utf8(self):
154
        """Write a Unicode argument to the trace log"""
155
        mutter(u'the unicode character for benzene is %s',
156
               u'\N{BENZENE RING}'.encode('utf-8'))
1927.3.4 by Carl Friedrich Bolz
Merge bzr.dev.
157
        self.assertContainsRe(self._get_log(keep_log_file=True),
158
                              'the unicode character')
1948.1.3 by John Arbash Meinel
Fix mutter() so even if args are invalid, it still works
159
1740.5.7 by Martin Pool
Add test for formatting of EPIPE
160
    def test_report_broken_pipe(self):
161
        try:
162
            raise IOError(errno.EPIPE, 'broken pipe foofofo')
163
        except IOError, e:
1551.9.3 by Aaron Bentley
Revert buggy apport changes
164
            msg = _format_exception()
1740.5.7 by Martin Pool
Add test for formatting of EPIPE
165
            self.assertEquals(msg, "bzr: broken pipe\n")
166
        else:
167
            self.fail("expected error not raised")
1740.5.9 by Martin Pool
[merge] bzr.dev
168
3195.1.1 by Andrew Bennetts
Always include timestamps in the trace file (i.e. remove -Dtimes in favour of having it switched on permanently)
169
    def assertLogStartsWith(self, log, string):
170
        """Like assertStartsWith, but skips the log timestamp."""
171
        self.assertContainsRe(log,
172
            '^\\d+\\.\\d+  ' + re.escape(string))
173
2725.1.1 by Robert Collins
Add -Devil flag to highlight the use of problematic API calls.
174
    def test_mutter_callsite_1(self):
175
        """mutter_callsite can capture 1 level of stack frame."""
176
        mutter_callsite(1, "foo %s", "a string")
177
        log = self._get_log(keep_log_file=True)
178
        # begin with the message
3195.1.1 by Andrew Bennetts
Always include timestamps in the trace file (i.e. remove -Dtimes in favour of having it switched on permanently)
179
        self.assertLogStartsWith(log, 'foo a string\nCalled from:\n')
2725.1.1 by Robert Collins
Add -Devil flag to highlight the use of problematic API calls.
180
        # should show two frame: this frame and the one above
181
        self.assertContainsRe(log,
3195.1.1 by Andrew Bennetts
Always include timestamps in the trace file (i.e. remove -Dtimes in favour of having it switched on permanently)
182
            'test_trace\\.py", line \\d+, in test_mutter_callsite_1\n')
2725.1.1 by Robert Collins
Add -Devil flag to highlight the use of problematic API calls.
183
        # this frame should be the final one
184
        self.assertEndsWith(log, ' "a string")\n')
185
186
    def test_mutter_callsite_2(self):
187
        """mutter_callsite can capture 2 levels of stack frame."""
188
        mutter_callsite(2, "foo %s", "a string")
189
        log = self._get_log(keep_log_file=True)
190
        # begin with the message
3195.1.1 by Andrew Bennetts
Always include timestamps in the trace file (i.e. remove -Dtimes in favour of having it switched on permanently)
191
        self.assertLogStartsWith(log, 'foo a string\nCalled from:\n')
2725.1.1 by Robert Collins
Add -Devil flag to highlight the use of problematic API calls.
192
        # should show two frame: this frame and the one above
193
        self.assertContainsRe(log,
194
            'test_trace.py", line \d+, in test_mutter_callsite_2\n')
195
        # this frame should be the final one
196
        self.assertEndsWith(log, ' "a string")\n')
197
1185.85.5 by John Arbash Meinel
mutter() should not fail because of unicode errors
198
    def test_mutter_never_fails(self):
199
        # Even if the decode/encode stage fails, mutter should not
200
        # raise an exception
201
        mutter(u'Writing a greek mu (\xb5) works in a unicode string')
202
        mutter('But fails in an ascii string \xb5')
1948.1.4 by John Arbash Meinel
Update test_never_fails, to cover one of the failure points
203
        mutter('and in an ascii argument: %s', '\xb5')
1927.3.1 by Carl Friedrich Bolz
Throw away on-disk logfile when possible.
204
        log = self._get_log(keep_log_file=True)
1185.85.5 by John Arbash Meinel
mutter() should not fail because of unicode errors
205
        self.assertContainsRe(log, 'Writing a greek mu')
1948.1.9 by John Arbash Meinel
Change mutter() so that it doesn't try so hard to write out perfect utf8, instead, rather than using a utf8 file, it changes unicode to utf8 manually
206
        self.assertContainsRe(log, "But fails in an ascii string")
207
        self.assertContainsRe(log, u"ascii argument: \xb5")
2725.1.1 by Robert Collins
Add -Devil flag to highlight the use of problematic API calls.
208
3173.1.12 by Martin Pool
Add test_push_log_file
209
    def test_push_log_file(self):
210
        """Can push and pop log file, and this catches mutter messages.
211
3943.8.1 by Marius Kruger
remove all trailing whitespace from bzr source
212
        This is primarily for use in the test framework.
3173.1.12 by Martin Pool
Add test_push_log_file
213
        """
214
        tmp1 = tempfile.NamedTemporaryFile()
215
        tmp2 = tempfile.NamedTemporaryFile()
216
        try:
217
            memento1 = push_log_file(tmp1)
218
            mutter("comment to file1")
219
            try:
220
                memento2 = push_log_file(tmp2)
221
                try:
222
                    mutter("comment to file2")
223
                finally:
224
                    pop_log_file(memento2)
225
                mutter("again to file1")
226
            finally:
227
                pop_log_file(memento1)
228
            # the files were opened in binary mode, so should have exactly
229
            # these bytes.  and removing the file as the log target should
3173.1.15 by Martin Pool
Update test_push_log_file to handle there always being timestamps at the start of the trace messages
230
            # have caused them to be flushed out.  need to match using regexps
231
            # as there's a timestamp at the front.
232
            tmp1.seek(0)
233
            self.assertContainsRe(tmp1.read(),
234
                r"\d+\.\d+  comment to file1\n\d+\.\d+  again to file1\n")
235
            tmp2.seek(0)
236
            self.assertContainsRe(tmp2.read(),
237
                r"\d+\.\d+  comment to file2\n")
3173.1.12 by Martin Pool
Add test_push_log_file
238
        finally:
239
            tmp1.close()
240
            tmp2.close()
241
4634.118.1 by John Arbash Meinel
Fix bug #503886, errors setting up logging go to stderr.
242
    def test__open_bzr_log_uses_stderr_for_failures(self):
243
        # If _open_bzr_log cannot open the file, then we should write the
244
        # warning to stderr. Since this is normally happening before logging is
245
        # set up.
246
        self.addCleanup(setattr, sys, 'stderr', sys.stderr)
247
        self.addCleanup(setattr, trace, '_bzr_log_filename',
248
                        trace._bzr_log_filename)
249
        sys.stderr = StringIO()
250
        # Set the log file to something that cannot exist
251
        os.environ['BZR_LOG'] = os.getcwd() + '/no-dir/bzr.log'
252
        logf = trace._open_bzr_log()
253
        self.assertIs(None, logf)
254
        self.assertContainsRe(sys.stderr.getvalue(),
255
                              'failed to open trace file: .*/no-dir/bzr.log')
2768.1.10 by Ian Clatworthy
Add tests for new methods in trace.py
256
257
class TestVerbosityLevel(TestCase):
258
259
    def test_verbosity_level(self):
260
        set_verbosity_level(1)
261
        self.assertEqual(1, get_verbosity_level())
262
        self.assertTrue(is_verbose())
263
        self.assertFalse(is_quiet())
264
        set_verbosity_level(-1)
265
        self.assertEqual(-1, get_verbosity_level())
266
        self.assertFalse(is_verbose())
267
        self.assertTrue(is_quiet())
268
        set_verbosity_level(0)
269
        self.assertEqual(0, get_verbosity_level())
270
        self.assertFalse(is_verbose())
271
        self.assertFalse(is_quiet())
272
273
    def test_be_quiet(self):
274
        # Confirm the old API still works
275
        be_quiet(True)
276
        self.assertEqual(-1, get_verbosity_level())
277
        be_quiet(False)
278
        self.assertEqual(0, get_verbosity_level())
2851.3.1 by Martin Pool
Add unit test for _rollover_trace_maybe
279
280
281
class TestBzrLog(TestCaseInTempDir):
282
283
    def test_log_rollover(self):
284
        temp_log_name = 'test-log'
285
        trace_file = open(temp_log_name, 'at')
286
        trace_file.write('test_log_rollover padding\n' * 1000000)
287
        trace_file.close()
288
        _rollover_trace_maybe(temp_log_name)
289
        # should have been rolled over
290
        self.assertFalse(os.access(temp_log_name, os.R_OK))