~bzr-pqm/bzr/bzr.dev

3173.1.12 by Martin Pool
Add test_push_log_file
1
# Copyright (C) 2005, 2006, 2007, 2008 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
15
# Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
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,
30
    )
1551.9.3 by Aaron Bentley
Revert buggy apport changes
31
from bzrlib.tests import TestCaseInTempDir, TestCase
2768.1.10 by Ian Clatworthy
Add tests for new methods in trace.py
32
from bzrlib.trace import (
33
    mutter, mutter_callsite, report_exception,
34
    set_verbosity_level, get_verbosity_level, is_quiet, is_verbose, be_quiet,
3173.1.12 by Martin Pool
Add test_push_log_file
35
    pop_log_file,
36
    push_log_file,
2851.3.1 by Martin Pool
Add unit test for _rollover_trace_maybe
37
    _rollover_trace_maybe,
2768.1.10 by Ian Clatworthy
Add tests for new methods in trace.py
38
    )
1185.33.9 by Martin Pool
Add new selftest module.
39
1740.5.2 by Martin Pool
Improved tests for display of exceptions.
40
41
def _format_exception():
42
    """Format an exception as it would normally be displayed to the user"""
43
    buf = StringIO()
1551.9.3 by Aaron Bentley
Revert buggy apport changes
44
    report_exception(sys.exc_info(), buf)
45
    return buf.getvalue()
1740.5.2 by Martin Pool
Improved tests for display of exceptions.
46
47
1185.33.9 by Martin Pool
Add new selftest module.
48
class TestTrace(TestCase):
1740.5.2 by Martin Pool
Improved tests for display of exceptions.
49
1551.9.3 by Aaron Bentley
Revert buggy apport changes
50
    def test_format_sys_exception(self):
1185.33.9 by Martin Pool
Add new selftest module.
51
        try:
52
            raise NotImplementedError, "time travel"
53
        except NotImplementedError:
54
            pass
1551.9.3 by Aaron Bentley
Revert buggy apport changes
55
        err = _format_exception()
1740.5.2 by Martin Pool
Improved tests for display of exceptions.
56
        self.assertEqualDiff(err.splitlines()[0],
1740.5.3 by Martin Pool
Cleanup more exception-formatting code
57
                'bzr: ERROR: exceptions.NotImplementedError: time travel')
1740.5.2 by Martin Pool
Improved tests for display of exceptions.
58
        self.assertContainsRe(err,
59
                r'File.*test_trace.py')
1185.33.9 by Martin Pool
Add new selftest module.
60
1740.5.3 by Martin Pool
Cleanup more exception-formatting code
61
    def test_format_interrupt_exception(self):
62
        try:
63
            raise KeyboardInterrupt()
1740.5.5 by Martin Pool
Show short form for OSError and IOError too
64
        except KeyboardInterrupt:
1740.5.3 by Martin Pool
Cleanup more exception-formatting code
65
            # XXX: Some risk that a *real* keyboard interrupt won't be seen
66
            pass
1551.9.3 by Aaron Bentley
Revert buggy apport changes
67
        msg = _format_exception()
1740.5.3 by Martin Pool
Cleanup more exception-formatting code
68
        self.assertTrue(len(msg) > 0)
69
        self.assertEqualDiff(msg, 'bzr: interrupted\n')
70
1740.5.5 by Martin Pool
Show short form for OSError and IOError too
71
    def test_format_os_error(self):
72
        try:
73
            file('nosuchfile22222')
74
        except (OSError, IOError):
75
            pass
1551.9.3 by Aaron Bentley
Revert buggy apport changes
76
        msg = _format_exception()
1740.5.5 by Martin Pool
Show short form for OSError and IOError too
77
        self.assertContainsRe(msg, r'^bzr: ERROR: \[Errno .*\] No such file.*nosuchfile')
78
1948.1.5 by John Arbash Meinel
Make sure BzrCommandError can handle unicode arguments
79
    def test_format_unicode_error(self):
80
        try:
81
            raise errors.BzrCommandError(u'argument foo\xb5 does not exist')
82
        except errors.BzrCommandError:
83
            pass
1551.9.3 by Aaron Bentley
Revert buggy apport changes
84
        msg = _format_exception()
1740.5.5 by Martin Pool
Show short form for OSError and IOError too
85
1185.33.9 by Martin Pool
Add new selftest module.
86
    def test_format_exception(self):
1740.5.2 by Martin Pool
Improved tests for display of exceptions.
87
        """Short formatting of bzr exceptions"""
1185.33.9 by Martin Pool
Add new selftest module.
88
        try:
2067.3.1 by Martin Pool
Clean up BzrNewError, other exception classes and users.
89
            raise errors.NotBranchError('wibble')
1948.1.5 by John Arbash Meinel
Make sure BzrCommandError can handle unicode arguments
90
        except errors.NotBranchError:
1185.33.9 by Martin Pool
Add new selftest module.
91
            pass
1551.9.3 by Aaron Bentley
Revert buggy apport changes
92
        msg = _format_exception()
1740.5.2 by Martin Pool
Improved tests for display of exceptions.
93
        self.assertTrue(len(msg) > 0)
2745.3.2 by Daniel Watkins
Updated tests to reflect new error text.
94
        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.
95
3497.3.2 by Martin Pool
Show short error for missing libraries
96
    def test_report_external_import_error(self):
97
        """Short friendly message for missing system modules."""
98
        try:
99
            import ImaginaryModule
100
        except ImportError, e:
101
            pass
102
        else:
103
            self.fail("somehow succeeded in importing %r" % ImaginaryModule)
104
        msg = _format_exception()
105
        self.assertEqual(msg,
106
            'bzr: ERROR: No module named ImaginaryModule\n'
107
            'You may need to install this Python library separately.\n')
108
109
    def test_report_import_syntax_error(self):
110
        try:
111
            raise ImportError("syntax error")
112
        except ImportError, e:
113
            pass
114
        msg = _format_exception()
115
        self.assertContainsRe(msg,
116
            r"Traceback \(most recent call last\)")
117
1185.33.51 by Martin Pool
Fix trace of non-ascii messages, and add test.
118
    def test_trace_unicode(self):
119
        """Write Unicode to trace log"""
120
        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.
121
        self.assertContainsRe(self._get_log(keep_log_file=True),
122
                              "the unicode character for benzene is")
1948.1.2 by John Arbash Meinel
Fix the test_trace functions to actually test that things are written to the log
123
    
124
    def test_trace_argument_unicode(self):
125
        """Write a Unicode argument to the trace log"""
126
        mutter(u'the unicode character for benzene is %s', u'\N{BENZENE RING}')
1927.3.4 by Carl Friedrich Bolz
Merge bzr.dev.
127
        self.assertContainsRe(self._get_log(keep_log_file=True),
128
                              'the unicode character')
1185.85.5 by John Arbash Meinel
mutter() should not fail because of unicode errors
129
1948.1.3 by John Arbash Meinel
Fix mutter() so even if args are invalid, it still works
130
    def test_trace_argument_utf8(self):
131
        """Write a Unicode argument to the trace log"""
132
        mutter(u'the unicode character for benzene is %s',
133
               u'\N{BENZENE RING}'.encode('utf-8'))
1927.3.4 by Carl Friedrich Bolz
Merge bzr.dev.
134
        self.assertContainsRe(self._get_log(keep_log_file=True),
135
                              'the unicode character')
1948.1.3 by John Arbash Meinel
Fix mutter() so even if args are invalid, it still works
136
1740.5.7 by Martin Pool
Add test for formatting of EPIPE
137
    def test_report_broken_pipe(self):
138
        try:
139
            raise IOError(errno.EPIPE, 'broken pipe foofofo')
140
        except IOError, e:
1551.9.3 by Aaron Bentley
Revert buggy apport changes
141
            msg = _format_exception()
1740.5.7 by Martin Pool
Add test for formatting of EPIPE
142
            self.assertEquals(msg, "bzr: broken pipe\n")
143
        else:
144
            self.fail("expected error not raised")
1740.5.9 by Martin Pool
[merge] bzr.dev
145
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)
146
    def assertLogStartsWith(self, log, string):
147
        """Like assertStartsWith, but skips the log timestamp."""
148
        self.assertContainsRe(log,
149
            '^\\d+\\.\\d+  ' + re.escape(string))
150
2725.1.1 by Robert Collins
Add -Devil flag to highlight the use of problematic API calls.
151
    def test_mutter_callsite_1(self):
152
        """mutter_callsite can capture 1 level of stack frame."""
153
        mutter_callsite(1, "foo %s", "a string")
154
        log = self._get_log(keep_log_file=True)
155
        # 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)
156
        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.
157
        # should show two frame: this frame and the one above
158
        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)
159
            '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.
160
        # this frame should be the final one
161
        self.assertEndsWith(log, ' "a string")\n')
162
163
    def test_mutter_callsite_2(self):
164
        """mutter_callsite can capture 2 levels of stack frame."""
165
        mutter_callsite(2, "foo %s", "a string")
166
        log = self._get_log(keep_log_file=True)
167
        # 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)
168
        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.
169
        # should show two frame: this frame and the one above
170
        self.assertContainsRe(log,
171
            'test_trace.py", line \d+, in test_mutter_callsite_2\n')
172
        # this frame should be the final one
173
        self.assertEndsWith(log, ' "a string")\n')
174
1185.85.5 by John Arbash Meinel
mutter() should not fail because of unicode errors
175
    def test_mutter_never_fails(self):
176
        # Even if the decode/encode stage fails, mutter should not
177
        # raise an exception
178
        mutter(u'Writing a greek mu (\xb5) works in a unicode string')
179
        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
180
        mutter('and in an ascii argument: %s', '\xb5')
1927.3.1 by Carl Friedrich Bolz
Throw away on-disk logfile when possible.
181
        log = self._get_log(keep_log_file=True)
1185.85.5 by John Arbash Meinel
mutter() should not fail because of unicode errors
182
        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
183
        self.assertContainsRe(log, "But fails in an ascii string")
184
        self.assertContainsRe(log, u"ascii argument: \xb5")
2725.1.1 by Robert Collins
Add -Devil flag to highlight the use of problematic API calls.
185
3173.1.12 by Martin Pool
Add test_push_log_file
186
    def test_push_log_file(self):
187
        """Can push and pop log file, and this catches mutter messages.
188
189
        This is primarily for use in the test framework. 
190
        """
191
        tmp1 = tempfile.NamedTemporaryFile()
192
        tmp2 = tempfile.NamedTemporaryFile()
193
        try:
194
            memento1 = push_log_file(tmp1)
195
            mutter("comment to file1")
196
            try:
197
                memento2 = push_log_file(tmp2)
198
                try:
199
                    mutter("comment to file2")
200
                finally:
201
                    pop_log_file(memento2)
202
                mutter("again to file1")
203
            finally:
204
                pop_log_file(memento1)
205
            # the files were opened in binary mode, so should have exactly
206
            # 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
207
            # have caused them to be flushed out.  need to match using regexps
208
            # as there's a timestamp at the front.
209
            tmp1.seek(0)
210
            self.assertContainsRe(tmp1.read(),
211
                r"\d+\.\d+  comment to file1\n\d+\.\d+  again to file1\n")
212
            tmp2.seek(0)
213
            self.assertContainsRe(tmp2.read(),
214
                r"\d+\.\d+  comment to file2\n")
3173.1.12 by Martin Pool
Add test_push_log_file
215
        finally:
216
            tmp1.close()
217
            tmp2.close()
218
2768.1.10 by Ian Clatworthy
Add tests for new methods in trace.py
219
220
class TestVerbosityLevel(TestCase):
221
222
    def test_verbosity_level(self):
223
        set_verbosity_level(1)
224
        self.assertEqual(1, get_verbosity_level())
225
        self.assertTrue(is_verbose())
226
        self.assertFalse(is_quiet())
227
        set_verbosity_level(-1)
228
        self.assertEqual(-1, get_verbosity_level())
229
        self.assertFalse(is_verbose())
230
        self.assertTrue(is_quiet())
231
        set_verbosity_level(0)
232
        self.assertEqual(0, get_verbosity_level())
233
        self.assertFalse(is_verbose())
234
        self.assertFalse(is_quiet())
235
236
    def test_be_quiet(self):
237
        # Confirm the old API still works
238
        be_quiet(True)
239
        self.assertEqual(-1, get_verbosity_level())
240
        be_quiet(False)
241
        self.assertEqual(0, get_verbosity_level())
2851.3.1 by Martin Pool
Add unit test for _rollover_trace_maybe
242
243
244
class TestBzrLog(TestCaseInTempDir):
245
246
    def test_log_rollover(self):
247
        temp_log_name = 'test-log'
248
        trace_file = open(temp_log_name, 'at')
249
        trace_file.write('test_log_rollover padding\n' * 1000000)
250
        trace_file.close()
251
        _rollover_trace_maybe(temp_log_name)
252
        # should have been rolled over
253
        self.assertFalse(os.access(temp_log_name, os.R_OK))