/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/tests/test_trace.py

  • Committer: Canonical.com Patch Queue Manager
  • Date: 2007-11-12 21:23:07 UTC
  • mfrom: (2898.4.17 bzr.lp-urls-bug-121200)
  • Revision ID: pqm@pqm.ubuntu.com-20071112212307-eusj64ymto8l9abk
(robertc) Improved lp:/// URL support for the Launchpad plugin (James Henstridge).

Show diffs side-by-side

added added

removed removed

Lines of Context:
1
 
# Copyright (C) 2005-2010 Canonical Ltd
 
1
# Copyright (C) 2005, 2006, 2007 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
12
12
#
13
13
# You should have received a copy of the GNU General Public License
14
14
# along with this program; if not, write to the Free Software
15
 
# Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
 
15
# Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
16
16
 
17
17
# "weren't nothing promised to you.  do i look like i got a promise face?"
18
18
 
21
21
from cStringIO import StringIO
22
22
import errno
23
23
import os
24
 
import re
25
24
import sys
26
 
import tempfile
27
25
 
28
26
from bzrlib import (
29
27
    errors,
30
 
    trace,
31
28
    )
32
 
from bzrlib.tests import features, TestCaseInTempDir, TestCase
 
29
from bzrlib.tests import TestCaseInTempDir, TestCase
33
30
from bzrlib.trace import (
34
31
    mutter, mutter_callsite, report_exception,
35
32
    set_verbosity_level, get_verbosity_level, is_quiet, is_verbose, be_quiet,
36
 
    pop_log_file,
37
 
    push_log_file,
38
33
    _rollover_trace_maybe,
39
 
    show_error,
40
34
    )
41
35
 
42
36
 
50
44
class TestTrace(TestCase):
51
45
 
52
46
    def test_format_sys_exception(self):
53
 
        # Test handling of an internal/unexpected error that probably
54
 
        # indicates a bug in bzr.  The details of the message may vary
55
 
        # depending on whether apport is available or not.  See test_crash for
56
 
        # more.
57
47
        try:
58
48
            raise NotImplementedError, "time travel"
59
49
        except NotImplementedError:
62
52
        self.assertEqualDiff(err.splitlines()[0],
63
53
                'bzr: ERROR: exceptions.NotImplementedError: time travel')
64
54
        self.assertContainsRe(err,
65
 
            'Bazaar has encountered an internal error.')
 
55
                r'File.*test_trace.py')
66
56
 
67
57
    def test_format_interrupt_exception(self):
68
58
        try:
74
64
        self.assertTrue(len(msg) > 0)
75
65
        self.assertEqualDiff(msg, 'bzr: interrupted\n')
76
66
 
77
 
    def test_format_memory_error(self):
78
 
        try:
79
 
            raise MemoryError()
80
 
        except MemoryError:
81
 
            pass
82
 
        msg = _format_exception()
83
 
        self.assertEquals(msg,
84
 
            "bzr: out of memory\n")
85
 
 
86
67
    def test_format_os_error(self):
87
68
        try:
88
 
            os.rmdir('nosuchfile22222')
89
 
        except OSError, e:
90
 
            e_str = str(e)
91
 
        msg = _format_exception()
92
 
        # Linux seems to give "No such file" but Windows gives "The system
93
 
        # cannot find the file specified".
94
 
        self.assertEqual('bzr: ERROR: %s\n' % (e_str,), msg)
95
 
 
96
 
    def test_format_io_error(self):
97
 
        try:
98
69
            file('nosuchfile22222')
99
 
        except IOError:
100
 
            pass
101
 
        msg = _format_exception()
102
 
        # Even though Windows and Linux differ for 'os.rmdir', they both give
103
 
        # 'No such file' for open()
104
 
        self.assertContainsRe(msg,
105
 
            r'^bzr: ERROR: \[Errno .*\] No such file.*nosuchfile')
106
 
 
107
 
    def test_format_pywintypes_error(self):
108
 
        self.requireFeature(features.pywintypes)
109
 
        import pywintypes, win32file
110
 
        try:
111
 
            win32file.RemoveDirectory('nosuchfile22222')
112
 
        except pywintypes.error:
113
 
            pass
114
 
        msg = _format_exception()
115
 
        # GZ 2010-05-03: Formatting for pywintypes.error is basic, a 3-tuple
116
 
        #                with errno, function name, and locale error message
117
 
        self.assertContainsRe(msg,
118
 
            r"^bzr: ERROR: \(2, 'RemoveDirectory[AW]?', .*\)")
 
70
        except (OSError, IOError):
 
71
            pass
 
72
        msg = _format_exception()
 
73
        self.assertContainsRe(msg, r'^bzr: ERROR: \[Errno .*\] No such file.*nosuchfile')
119
74
 
120
75
    def test_format_unicode_error(self):
121
76
        try:
134
89
        self.assertTrue(len(msg) > 0)
135
90
        self.assertEqualDiff(msg, 'bzr: ERROR: Not a branch: \"wibble\".\n')
136
91
 
137
 
    def test_report_external_import_error(self):
138
 
        """Short friendly message for missing system modules."""
139
 
        try:
140
 
            import ImaginaryModule
141
 
        except ImportError, e:
142
 
            pass
143
 
        else:
144
 
            self.fail("somehow succeeded in importing %r" % ImaginaryModule)
145
 
        msg = _format_exception()
146
 
        self.assertEqual(msg,
147
 
            'bzr: ERROR: No module named ImaginaryModule\n'
148
 
            'You may need to install this Python library separately.\n')
149
 
 
150
 
    def test_report_import_syntax_error(self):
151
 
        try:
152
 
            raise ImportError("syntax error")
153
 
        except ImportError, e:
154
 
            pass
155
 
        msg = _format_exception()
156
 
        self.assertContainsRe(msg,
157
 
            r'Bazaar has encountered an internal error')
158
 
 
159
92
    def test_trace_unicode(self):
160
93
        """Write Unicode to trace log"""
161
94
        self.log(u'the unicode character for benzene is \N{BENZENE RING}')
162
 
        log = self.get_log()
163
 
        self.assertContainsRe(log, "the unicode character for benzene is")
164
 
 
 
95
        self.assertContainsRe(self._get_log(keep_log_file=True),
 
96
                              "the unicode character for benzene is")
 
97
    
165
98
    def test_trace_argument_unicode(self):
166
99
        """Write a Unicode argument to the trace log"""
167
100
        mutter(u'the unicode character for benzene is %s', u'\N{BENZENE RING}')
168
 
        log = self.get_log()
169
 
        self.assertContainsRe(log, 'the unicode character')
 
101
        self.assertContainsRe(self._get_log(keep_log_file=True),
 
102
                              'the unicode character')
170
103
 
171
104
    def test_trace_argument_utf8(self):
172
105
        """Write a Unicode argument to the trace log"""
173
106
        mutter(u'the unicode character for benzene is %s',
174
107
               u'\N{BENZENE RING}'.encode('utf-8'))
175
 
        log = self.get_log()
176
 
        self.assertContainsRe(log, 'the unicode character')
 
108
        self.assertContainsRe(self._get_log(keep_log_file=True),
 
109
                              'the unicode character')
177
110
 
178
111
    def test_report_broken_pipe(self):
179
112
        try:
184
117
        else:
185
118
            self.fail("expected error not raised")
186
119
 
187
 
    def assertLogStartsWith(self, log, string):
188
 
        """Like assertStartsWith, but skips the log timestamp."""
189
 
        self.assertContainsRe(log,
190
 
            '^\\d+\\.\\d+  ' + re.escape(string))
191
 
 
192
120
    def test_mutter_callsite_1(self):
193
121
        """mutter_callsite can capture 1 level of stack frame."""
194
122
        mutter_callsite(1, "foo %s", "a string")
195
 
        log = self.get_log()
 
123
        log = self._get_log(keep_log_file=True)
196
124
        # begin with the message
197
 
        self.assertLogStartsWith(log, 'foo a string\nCalled from:\n')
 
125
        self.assertStartsWith(log, 'foo a string\nCalled from:\n')
198
126
        # should show two frame: this frame and the one above
199
127
        self.assertContainsRe(log,
200
 
            'test_trace\\.py", line \\d+, in test_mutter_callsite_1\n')
 
128
            'test_trace\.py", line \d+, in test_mutter_callsite_1\n')
201
129
        # this frame should be the final one
202
130
        self.assertEndsWith(log, ' "a string")\n')
203
131
 
204
132
    def test_mutter_callsite_2(self):
205
133
        """mutter_callsite can capture 2 levels of stack frame."""
206
134
        mutter_callsite(2, "foo %s", "a string")
207
 
        log = self.get_log()
 
135
        log = self._get_log(keep_log_file=True)
208
136
        # begin with the message
209
 
        self.assertLogStartsWith(log, 'foo a string\nCalled from:\n')
 
137
        self.assertStartsWith(log, 'foo a string\nCalled from:\n')
210
138
        # should show two frame: this frame and the one above
211
139
        self.assertContainsRe(log,
212
140
            'test_trace.py", line \d+, in test_mutter_callsite_2\n')
216
144
    def test_mutter_never_fails(self):
217
145
        # Even if the decode/encode stage fails, mutter should not
218
146
        # raise an exception
219
 
        # This test checks that mutter doesn't fail; the current behaviour
220
 
        # is that it doesn't fail *and writes non-utf8*.
221
147
        mutter(u'Writing a greek mu (\xb5) works in a unicode string')
222
148
        mutter('But fails in an ascii string \xb5')
223
149
        mutter('and in an ascii argument: %s', '\xb5')
224
 
        log = self.get_log()
 
150
        log = self._get_log(keep_log_file=True)
225
151
        self.assertContainsRe(log, 'Writing a greek mu')
226
152
        self.assertContainsRe(log, "But fails in an ascii string")
227
 
        # However, the log content object does unicode replacement on reading
228
 
        # to let it get unicode back where good data has been written. So we
229
 
        # have to do a replaceent here as well.
230
 
        self.assertContainsRe(log, "ascii argument: \xb5".decode('utf8',
231
 
            'replace'))
232
 
        
233
 
    def test_show_error(self):
234
 
        show_error('error1')
235
 
        show_error(u'error2 \xb5 blah')
236
 
        show_error('arg: %s', 'blah')
237
 
        show_error('arg2: %(key)s', {'key':'stuff'})
238
 
        try:
239
 
            raise Exception("oops")
240
 
        except:
241
 
            show_error('kwarg', exc_info=True)
242
 
        log = self.get_log()
243
 
        self.assertContainsRe(log, 'error1')
244
 
        self.assertContainsRe(log, u'error2 \xb5 blah')
245
 
        self.assertContainsRe(log, 'arg: blah')
246
 
        self.assertContainsRe(log, 'arg2: stuff')
247
 
        self.assertContainsRe(log, 'kwarg')
248
 
        self.assertContainsRe(log, 'Traceback \\(most recent call last\\):')
249
 
        self.assertContainsRe(log, 'File ".*test_trace.py", line .*, in test_show_error')
250
 
        self.assertContainsRe(log, 'raise Exception\\("oops"\\)')
251
 
        self.assertContainsRe(log, 'Exception: oops')
252
 
 
253
 
    def test_push_log_file(self):
254
 
        """Can push and pop log file, and this catches mutter messages.
255
 
 
256
 
        This is primarily for use in the test framework.
257
 
        """
258
 
        tmp1 = tempfile.NamedTemporaryFile()
259
 
        tmp2 = tempfile.NamedTemporaryFile()
260
 
        try:
261
 
            memento1 = push_log_file(tmp1)
262
 
            mutter("comment to file1")
263
 
            try:
264
 
                memento2 = push_log_file(tmp2)
265
 
                try:
266
 
                    mutter("comment to file2")
267
 
                finally:
268
 
                    pop_log_file(memento2)
269
 
                mutter("again to file1")
270
 
            finally:
271
 
                pop_log_file(memento1)
272
 
            # the files were opened in binary mode, so should have exactly
273
 
            # these bytes.  and removing the file as the log target should
274
 
            # have caused them to be flushed out.  need to match using regexps
275
 
            # as there's a timestamp at the front.
276
 
            tmp1.seek(0)
277
 
            self.assertContainsRe(tmp1.read(),
278
 
                r"\d+\.\d+  comment to file1\n\d+\.\d+  again to file1\n")
279
 
            tmp2.seek(0)
280
 
            self.assertContainsRe(tmp2.read(),
281
 
                r"\d+\.\d+  comment to file2\n")
282
 
        finally:
283
 
            tmp1.close()
284
 
            tmp2.close()
285
 
 
286
 
    def test__open_bzr_log_uses_stderr_for_failures(self):
287
 
        # If _open_bzr_log cannot open the file, then we should write the
288
 
        # warning to stderr. Since this is normally happening before logging is
289
 
        # set up.
290
 
        self.overrideAttr(sys, 'stderr', StringIO())
291
 
        # Set the log file to something that cannot exist
292
 
        # FIXME: A bit dangerous: we are not in an isolated dir here -- vilajam
293
 
        # 20100125
294
 
        os.environ['BZR_LOG'] = os.getcwd() + '/no-dir/bzr.log'
295
 
        self.overrideAttr(trace, '_bzr_log_filename')
296
 
        logf = trace._open_bzr_log()
297
 
        self.assertIs(None, logf)
298
 
        self.assertContainsRe(sys.stderr.getvalue(),
299
 
                              'failed to open trace file: .*/no-dir/bzr.log')
 
153
        self.assertContainsRe(log, u"ascii argument: \xb5")
300
154
 
301
155
 
302
156
class TestVerbosityLevel(TestCase):
328
182
    def test_log_rollover(self):
329
183
        temp_log_name = 'test-log'
330
184
        trace_file = open(temp_log_name, 'at')
331
 
        trace_file.writelines(['test_log_rollover padding\n'] * 200000)
 
185
        trace_file.write('test_log_rollover padding\n' * 1000000)
332
186
        trace_file.close()
333
187
        _rollover_trace_maybe(temp_log_name)
334
188
        # should have been rolled over