/brz/remove-bazaar

To get this branch, use:
bzr branch http://gegoxaren.bato24.eu/bzr/brz/remove-bazaar
6614.1.3 by Vincent Ladeuil
Fix assertEquals being deprecated by using assertEqual.
1
# Copyright (C) 2005-2011, 2016 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
1740.5.7 by Martin Pool
Add test for formatting of EPIPE
21
import errno
6015.52.1 by Martin Packman
Add tests for logging various problematic values
22
import logging
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
6624 by Jelmer Vernooij
Merge Python3 porting work ('py3 pokes')
28
from .. import (
5448.5.8 by Karl Bielefeldt
Add test for message format with -Dmem_dump
29
    debug,
1948.1.5 by John Arbash Meinel
Make sure BzrCommandError can handle unicode arguments
30
    errors,
4634.118.1 by John Arbash Meinel
Fix bug #503886, errors setting up logging go to stderr.
31
    trace,
1948.1.5 by John Arbash Meinel
Make sure BzrCommandError can handle unicode arguments
32
    )
6624 by Jelmer Vernooij
Merge Python3 porting work ('py3 pokes')
33
from ..sixish import (
6677.1.5 by Martin
Make crash debug and trace modules pass on Python 3
34
    PY3,
35
    StringIO,
6621.22.2 by Martin
Use BytesIO or StringIO from bzrlib.sixish
36
    )
7427.1.1 by Jelmer Vernooij
Skip test when directory creation always succeeds.
37
from . import features, TestCaseInTempDir, TestCase, TestSkipped
6624 by Jelmer Vernooij
Merge Python3 porting work ('py3 pokes')
38
from ..trace import (
2768.1.10 by Ian Clatworthy
Add tests for new methods in trace.py
39
    mutter, mutter_callsite, report_exception,
40
    set_verbosity_level, get_verbosity_level, is_quiet, is_verbose, be_quiet,
3173.1.12 by Martin Pool
Add test_push_log_file
41
    pop_log_file,
42
    push_log_file,
2851.3.1 by Martin Pool
Add unit test for _rollover_trace_maybe
43
    _rollover_trace_maybe,
5055.4.1 by Gordon Tyler
Fixed show_error args and added test for show_error.
44
    show_error,
2768.1.10 by Ian Clatworthy
Add tests for new methods in trace.py
45
    )
1185.33.9 by Martin Pool
Add new selftest module.
46
1740.5.2 by Martin Pool
Improved tests for display of exceptions.
47
48
def _format_exception():
49
    """Format an exception as it would normally be displayed to the user"""
6677.1.5 by Martin
Make crash debug and trace modules pass on Python 3
50
    buf = StringIO()
5765.1.1 by John Arbash Meinel
Merge Federico Culloca's PointlessCommit message, and revert accidental changes.
51
    report_exception(sys.exc_info(), buf)
1551.9.3 by Aaron Bentley
Revert buggy apport changes
52
    return buf.getvalue()
1740.5.2 by Martin Pool
Improved tests for display of exceptions.
53
54
1185.33.9 by Martin Pool
Add new selftest module.
55
class TestTrace(TestCase):
1740.5.2 by Martin Pool
Improved tests for display of exceptions.
56
1551.9.3 by Aaron Bentley
Revert buggy apport changes
57
    def test_format_sys_exception(self):
4584.3.20 by Martin Pool
Tweak trace tests to cope without the traceback being printed
58
        # Test handling of an internal/unexpected error that probably
6622.1.33 by Jelmer Vernooij
Fix more tests (all?)
59
        # indicates a bug in brz.  The details of the message may vary
4584.3.20 by Martin Pool
Tweak trace tests to cope without the traceback being printed
60
        # depending on whether apport is available or not.  See test_crash for
61
        # more.
1185.33.9 by Martin Pool
Add new selftest module.
62
        try:
6619.3.11 by Jelmer Vernooij
Use modern exceptions.
63
            raise NotImplementedError("time travel")
1185.33.9 by Martin Pool
Add new selftest module.
64
        except NotImplementedError:
6677.1.5 by Martin
Make crash debug and trace modules pass on Python 3
65
            err = _format_exception()
66
        self.assertContainsRe(err,
7143.15.2 by Jelmer Vernooij
Run autopep8.
67
                              '^brz: ERROR: NotImplementedError: time travel')
1740.5.2 by Martin Pool
Improved tests for display of exceptions.
68
        self.assertContainsRe(err,
7143.15.2 by Jelmer Vernooij
Run autopep8.
69
                              'Bazaar has encountered an internal error.')
1185.33.9 by Martin Pool
Add new selftest module.
70
1740.5.3 by Martin Pool
Cleanup more exception-formatting code
71
    def test_format_interrupt_exception(self):
72
        try:
73
            raise KeyboardInterrupt()
1740.5.5 by Martin Pool
Show short form for OSError and IOError too
74
        except KeyboardInterrupt:
1740.5.3 by Martin Pool
Cleanup more exception-formatting code
75
            # XXX: Some risk that a *real* keyboard interrupt won't be seen
6677.1.5 by Martin
Make crash debug and trace modules pass on Python 3
76
            msg = _format_exception()
77
        self.assertEqual(msg, 'brz: interrupted\n')
1740.5.3 by Martin Pool
Cleanup more exception-formatting code
78
4634.26.1 by Martin Pool
Cleaner message when out of memory
79
    def test_format_memory_error(self):
80
        try:
81
            raise MemoryError()
82
        except MemoryError:
6677.1.5 by Martin
Make crash debug and trace modules pass on Python 3
83
            msg = _format_exception()
7143.15.5 by Jelmer Vernooij
More PEP8 fixes.
84
        self.assertEqual(
85
            msg,
86
            "brz: out of memory\nUse -Dmem_dump to dump memory to a file.\n")
4634.26.1 by Martin Pool
Cleaner message when out of memory
87
5448.5.8 by Karl Bielefeldt
Add test for message format with -Dmem_dump
88
    def test_format_mem_dump(self):
6091.2.2 by Max Bowsher
Per jam's review comments, get rid of features.meliae_feature, which is new in
89
        self.requireFeature(features.meliae)
5448.5.8 by Karl Bielefeldt
Add test for message format with -Dmem_dump
90
        debug.debug_flags.add('mem_dump')
91
        try:
92
            raise MemoryError()
93
        except MemoryError:
6677.1.5 by Martin
Make crash debug and trace modules pass on Python 3
94
            msg = _format_exception()
5448.5.8 by Karl Bielefeldt
Add test for message format with -Dmem_dump
95
        self.assertStartsWith(msg,
7143.15.2 by Jelmer Vernooij
Run autopep8.
96
                              "brz: out of memory\nMemory dumped to ")
5448.5.8 by Karl Bielefeldt
Add test for message format with -Dmem_dump
97
1740.5.5 by Martin Pool
Show short form for OSError and IOError too
98
    def test_format_os_error(self):
99
        try:
4095.1.1 by Martin Pool
Add more distinct tests for IOError and OSError
100
            os.rmdir('nosuchfile22222')
6619.3.2 by Jelmer Vernooij
Apply 2to3 except fix.
101
        except OSError as e:
4789.22.1 by John Arbash Meinel
Fix a test_trace failure on Windows.
102
            e_str = str(e)
6677.1.5 by Martin
Make crash debug and trace modules pass on Python 3
103
            msg = _format_exception()
4789.22.1 by John Arbash Meinel
Fix a test_trace failure on Windows.
104
        # Linux seems to give "No such file" but Windows gives "The system
105
        # cannot find the file specified".
6622.1.27 by Jelmer Vernooij
s/bzr/brz/
106
        self.assertEqual('brz: ERROR: %s\n' % (e_str,), msg)
4095.1.1 by Martin Pool
Add more distinct tests for IOError and OSError
107
108
    def test_format_io_error(self):
109
        try:
6677.1.5 by Martin
Make crash debug and trace modules pass on Python 3
110
            open('nosuchfile22222')
4095.1.1 by Martin Pool
Add more distinct tests for IOError and OSError
111
        except IOError:
6677.1.5 by Martin
Make crash debug and trace modules pass on Python 3
112
            msg = _format_exception()
4789.22.1 by John Arbash Meinel
Fix a test_trace failure on Windows.
113
        # Even though Windows and Linux differ for 'os.rmdir', they both give
114
        # 'No such file' for open()
7143.15.5 by Jelmer Vernooij
More PEP8 fixes.
115
        # However it now gets translated so we can not test for a specific
116
        # message
4789.22.1 by John Arbash Meinel
Fix a test_trace failure on Windows.
117
        self.assertContainsRe(msg,
7143.15.2 by Jelmer Vernooij
Run autopep8.
118
                              '^brz: ERROR: \\[Errno .*\\] .*nosuchfile')
1740.5.5 by Martin Pool
Show short form for OSError and IOError too
119
5200.4.1 by Martin
Test how pywintypes.error is displayed to the user
120
    def test_format_pywintypes_error(self):
5200.4.5 by Martin
Move pywintypes ModuleAvailableFeature to bzrlib.tests.features
121
        self.requireFeature(features.pywintypes)
7143.15.2 by Jelmer Vernooij
Run autopep8.
122
        import pywintypes
123
        import win32file
5200.4.1 by Martin
Test how pywintypes.error is displayed to the user
124
        try:
125
            win32file.RemoveDirectory('nosuchfile22222')
126
        except pywintypes.error:
6677.1.5 by Martin
Make crash debug and trace modules pass on Python 3
127
            msg = _format_exception()
5200.4.1 by Martin
Test how pywintypes.error is displayed to the user
128
        # GZ 2010-05-03: Formatting for pywintypes.error is basic, a 3-tuple
129
        #                with errno, function name, and locale error message
7143.15.5 by Jelmer Vernooij
More PEP8 fixes.
130
        self.assertContainsRe(
131
            msg, "^brz: ERROR: \\(2, 'RemoveDirectory[AW]?', .*\\)")
6677.1.5 by Martin
Make crash debug and trace modules pass on Python 3
132
5920.2.1 by Toon Nolten
Added a test for a sockets.error that wasn't caught in versions prior to python 2.6
133
    def test_format_sockets_error(self):
134
        try:
5920.2.2 by Toon Nolten
Fixed typo, and import
135
            import socket
5920.2.1 by Toon Nolten
Added a test for a sockets.error that wasn't caught in versions prior to python 2.6
136
            sock = socket.socket()
6677.1.5 by Martin
Make crash debug and trace modules pass on Python 3
137
            sock.send(b"This should fail.")
5920.2.2 by Toon Nolten
Fixed typo, and import
138
        except socket.error:
6677.1.5 by Martin
Make crash debug and trace modules pass on Python 3
139
            msg = _format_exception()
140
5920.2.3 by Toon Nolten
More general test.
141
        self.assertNotContainsRe(msg,
7143.15.2 by Jelmer Vernooij
Run autopep8.
142
                                 "Traceback \\(most recent call last\\):")
5200.4.1 by Martin
Test how pywintypes.error is displayed to the user
143
1948.1.5 by John Arbash Meinel
Make sure BzrCommandError can handle unicode arguments
144
    def test_format_unicode_error(self):
145
        try:
146
            raise errors.BzrCommandError(u'argument foo\xb5 does not exist')
147
        except errors.BzrCommandError:
6677.1.5 by Martin
Make crash debug and trace modules pass on Python 3
148
            msg = _format_exception()
149
        if PY3:
150
            expected = 'brz: ERROR: argument foo\xb5 does not exist\n'
151
        else:
152
            # GZ 2017-06-10: Pretty bogus, should encode per the output stream
153
            expected = 'brz: ERROR: argument foo\xc2\xb5 does not exist\n'
154
        self.assertEqual(msg, expected)
1740.5.5 by Martin Pool
Show short form for OSError and IOError too
155
1185.33.9 by Martin Pool
Add new selftest module.
156
    def test_format_exception(self):
6622.1.33 by Jelmer Vernooij
Fix more tests (all?)
157
        """Short formatting of brz exceptions"""
1185.33.9 by Martin Pool
Add new selftest module.
158
        try:
2067.3.1 by Martin Pool
Clean up BzrNewError, other exception classes and users.
159
            raise errors.NotBranchError('wibble')
1948.1.5 by John Arbash Meinel
Make sure BzrCommandError can handle unicode arguments
160
        except errors.NotBranchError:
6677.1.5 by Martin
Make crash debug and trace modules pass on Python 3
161
            msg = _format_exception()
162
        self.assertEqual(msg, 'brz: ERROR: Not a branch: \"wibble\".\n')
1185.33.63 by Martin Pool
Better display of BzrError classes that are not BzrNewErrors.
163
3497.3.2 by Martin Pool
Show short error for missing libraries
164
    def test_report_external_import_error(self):
165
        """Short friendly message for missing system modules."""
166
        try:
167
            import ImaginaryModule
7143.15.5 by Jelmer Vernooij
More PEP8 fixes.
168
        except ImportError:
6677.1.5 by Martin
Make crash debug and trace modules pass on Python 3
169
            msg = _format_exception()
3497.3.2 by Martin Pool
Show short error for missing libraries
170
        else:
171
            self.fail("somehow succeeded in importing %r" % ImaginaryModule)
7143.15.5 by Jelmer Vernooij
More PEP8 fixes.
172
        self.assertContainsRe(
173
            msg,
174
            "^brz: ERROR: No module named '?ImaginaryModule'?\n"
175
            "You may need to install this Python library separately.\n$")
3497.3.2 by Martin Pool
Show short error for missing libraries
176
177
    def test_report_import_syntax_error(self):
178
        try:
179
            raise ImportError("syntax error")
7143.15.5 by Jelmer Vernooij
More PEP8 fixes.
180
        except ImportError:
6677.1.5 by Martin
Make crash debug and trace modules pass on Python 3
181
            msg = _format_exception()
3497.3.2 by Martin Pool
Show short error for missing libraries
182
        self.assertContainsRe(msg,
7143.15.2 by Jelmer Vernooij
Run autopep8.
183
                              'Bazaar has encountered an internal error')
3497.3.2 by Martin Pool
Show short error for missing libraries
184
1185.33.51 by Martin Pool
Fix trace of non-ascii messages, and add test.
185
    def test_trace_unicode(self):
186
        """Write Unicode to trace log"""
187
        self.log(u'the unicode character for benzene is \N{BENZENE RING}')
4794.1.15 by Robert Collins
Review feedback.
188
        log = self.get_log()
4794.1.8 by Robert Collins
Move the passing of test logs to the result to be via the getDetails API and remove all public use of TestCase._get_log.
189
        self.assertContainsRe(log, "the unicode character for benzene is")
3943.8.1 by Marius Kruger
remove all trailing whitespace from bzr source
190
1948.1.2 by John Arbash Meinel
Fix the test_trace functions to actually test that things are written to the log
191
    def test_trace_argument_unicode(self):
192
        """Write a Unicode argument to the trace log"""
193
        mutter(u'the unicode character for benzene is %s', u'\N{BENZENE RING}')
4794.1.15 by Robert Collins
Review feedback.
194
        log = self.get_log()
4794.1.8 by Robert Collins
Move the passing of test logs to the result to be via the getDetails API and remove all public use of TestCase._get_log.
195
        self.assertContainsRe(log, 'the unicode character')
1185.85.5 by John Arbash Meinel
mutter() should not fail because of unicode errors
196
1948.1.3 by John Arbash Meinel
Fix mutter() so even if args are invalid, it still works
197
    def test_trace_argument_utf8(self):
198
        """Write a Unicode argument to the trace log"""
199
        mutter(u'the unicode character for benzene is %s',
200
               u'\N{BENZENE RING}'.encode('utf-8'))
4794.1.15 by Robert Collins
Review feedback.
201
        log = self.get_log()
4794.1.8 by Robert Collins
Move the passing of test logs to the result to be via the getDetails API and remove all public use of TestCase._get_log.
202
        self.assertContainsRe(log, 'the unicode character')
1948.1.3 by John Arbash Meinel
Fix mutter() so even if args are invalid, it still works
203
7008.2.1 by Martin
Make mutter saner with string interpolation
204
    def test_trace_argument_exception(self):
205
        err = Exception('an error')
206
        mutter(u'can format stringable classes %s', err)
207
        log = self.get_log()
208
        self.assertContainsRe(log, 'can format stringable classes an error')
209
1740.5.7 by Martin Pool
Add test for formatting of EPIPE
210
    def test_report_broken_pipe(self):
211
        try:
212
            raise IOError(errno.EPIPE, 'broken pipe foofofo')
7143.15.5 by Jelmer Vernooij
More PEP8 fixes.
213
        except IOError:
1551.9.3 by Aaron Bentley
Revert buggy apport changes
214
            msg = _format_exception()
6622.1.27 by Jelmer Vernooij
s/bzr/brz/
215
            self.assertEqual(msg, "brz: broken pipe\n")
1740.5.7 by Martin Pool
Add test for formatting of EPIPE
216
        else:
217
            self.fail("expected error not raised")
1740.5.9 by Martin Pool
[merge] bzr.dev
218
6677.1.5 by Martin
Make crash debug and trace modules pass on Python 3
219
    def assertLogContainsLine(self, log, string):
220
        """Assert log contains a line including log timestamp."""
221
        # Does not check absolute position in log as there may be kipple.
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)
222
        self.assertContainsRe(log,
7143.15.2 by Jelmer Vernooij
Run autopep8.
223
                              '(?m)^\\d+\\.\\d+  ' + re.escape(string))
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)
224
2725.1.1 by Robert Collins
Add -Devil flag to highlight the use of problematic API calls.
225
    def test_mutter_callsite_1(self):
226
        """mutter_callsite can capture 1 level of stack frame."""
227
        mutter_callsite(1, "foo %s", "a string")
4794.1.15 by Robert Collins
Review feedback.
228
        log = self.get_log()
2725.1.1 by Robert Collins
Add -Devil flag to highlight the use of problematic API calls.
229
        # begin with the message
6677.1.5 by Martin
Make crash debug and trace modules pass on Python 3
230
        self.assertLogContainsLine(log, 'foo a string\nCalled from:\n')
2725.1.1 by Robert Collins
Add -Devil flag to highlight the use of problematic API calls.
231
        # should show two frame: this frame and the one above
7143.15.5 by Jelmer Vernooij
More PEP8 fixes.
232
        self.assertContainsRe(
233
            log, '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.
234
        # this frame should be the final one
235
        self.assertEndsWith(log, ' "a string")\n')
236
237
    def test_mutter_callsite_2(self):
238
        """mutter_callsite can capture 2 levels of stack frame."""
239
        mutter_callsite(2, "foo %s", "a string")
4794.1.15 by Robert Collins
Review feedback.
240
        log = self.get_log()
2725.1.1 by Robert Collins
Add -Devil flag to highlight the use of problematic API calls.
241
        # begin with the message
6677.1.5 by Martin
Make crash debug and trace modules pass on Python 3
242
        self.assertLogContainsLine(log, 'foo a string\nCalled from:\n')
2725.1.1 by Robert Collins
Add -Devil flag to highlight the use of problematic API calls.
243
        # should show two frame: this frame and the one above
7143.15.5 by Jelmer Vernooij
More PEP8 fixes.
244
        self.assertContainsRe(
245
            log, 'test_trace.py", line \\d+, in test_mutter_callsite_2\n')
2725.1.1 by Robert Collins
Add -Devil flag to highlight the use of problematic API calls.
246
        # this frame should be the final one
247
        self.assertEndsWith(log, ' "a string")\n')
248
1185.85.5 by John Arbash Meinel
mutter() should not fail because of unicode errors
249
    def test_mutter_never_fails(self):
7008.2.1 by Martin
Make mutter saner with string interpolation
250
        """Even with unencodable input mutter should not raise errors."""
251
        mutter(u'can write unicode \xa7')
252
        mutter('can interpolate unicode %s', u'\xa7')
253
        mutter(b'can write bytes \xa7')
254
        mutter('can repr bytes %r', b'\xa7')
255
        mutter('can interpolate bytes %s', b'\xa7')
256
        # Log will always be written as utf-8
4794.1.15 by Robert Collins
Review feedback.
257
        log = self.get_log()
7008.2.1 by Martin
Make mutter saner with string interpolation
258
        self.assertContainsRe(
259
            log,
260
            u'.* +can write unicode \xa7\n'
261
            u'.* +can interpolate unicode \xa7\n'
262
            u'.* +can write bytes \ufffd\n'
263
            u'.* +can repr bytes b\'\\\\xa7\'\n'
264
            u'.* +can interpolate bytes (?:\ufffd|b\'\\\\xa7\')\n')
6325.3.1 by Vincent Ladeuil
Give meaningful deprecation warnings for deprecated test features
265
5055.4.1 by Gordon Tyler
Fixed show_error args and added test for show_error.
266
    def test_show_error(self):
267
        show_error('error1')
268
        show_error(u'error2 \xb5 blah')
5055.4.2 by Gordon Tyler
Improved test_show_error to test kwargs passing.
269
        show_error('arg: %s', 'blah')
7143.15.2 by Jelmer Vernooij
Run autopep8.
270
        show_error('arg2: %(key)s', {'key': 'stuff'})
5055.4.3 by Gordon Tyler
Improved test_show_error to check kwargs, dict substitution and exception handling.
271
        try:
272
            raise Exception("oops")
7143.15.5 by Jelmer Vernooij
More PEP8 fixes.
273
        except BaseException:
5055.4.3 by Gordon Tyler
Improved test_show_error to check kwargs, dict substitution and exception handling.
274
            show_error('kwarg', exc_info=True)
5055.4.1 by Gordon Tyler
Fixed show_error args and added test for show_error.
275
        log = self.get_log()
276
        self.assertContainsRe(log, 'error1')
277
        self.assertContainsRe(log, u'error2 \xb5 blah')
5055.4.2 by Gordon Tyler
Improved test_show_error to test kwargs passing.
278
        self.assertContainsRe(log, 'arg: blah')
5055.4.3 by Gordon Tyler
Improved test_show_error to check kwargs, dict substitution and exception handling.
279
        self.assertContainsRe(log, 'arg2: stuff')
280
        self.assertContainsRe(log, 'kwarg')
281
        self.assertContainsRe(log, 'Traceback \\(most recent call last\\):')
7143.15.2 by Jelmer Vernooij
Run autopep8.
282
        self.assertContainsRe(
283
            log, 'File ".*test_trace.py", line .*, in test_show_error')
5055.4.3 by Gordon Tyler
Improved test_show_error to check kwargs, dict substitution and exception handling.
284
        self.assertContainsRe(log, 'raise Exception\\("oops"\\)')
285
        self.assertContainsRe(log, 'Exception: oops')
2725.1.1 by Robert Collins
Add -Devil flag to highlight the use of problematic API calls.
286
3173.1.12 by Martin Pool
Add test_push_log_file
287
    def test_push_log_file(self):
288
        """Can push and pop log file, and this catches mutter messages.
289
3943.8.1 by Marius Kruger
remove all trailing whitespace from bzr source
290
        This is primarily for use in the test framework.
3173.1.12 by Martin Pool
Add test_push_log_file
291
        """
292
        tmp1 = tempfile.NamedTemporaryFile()
293
        tmp2 = tempfile.NamedTemporaryFile()
294
        try:
295
            memento1 = push_log_file(tmp1)
296
            mutter("comment to file1")
297
            try:
298
                memento2 = push_log_file(tmp2)
299
                try:
300
                    mutter("comment to file2")
301
                finally:
302
                    pop_log_file(memento2)
303
                mutter("again to file1")
304
            finally:
305
                pop_log_file(memento1)
306
            # the files were opened in binary mode, so should have exactly
307
            # 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
308
            # have caused them to be flushed out.  need to match using regexps
309
            # as there's a timestamp at the front.
310
            tmp1.seek(0)
311
            self.assertContainsRe(tmp1.read(),
7143.15.2 by Jelmer Vernooij
Run autopep8.
312
                                  b"\\d+\\.\\d+  comment to file1\n"
313
                                  b"\\d+\\.\\d+  again to file1\n")
3173.1.15 by Martin Pool
Update test_push_log_file to handle there always being timestamps at the start of the trace messages
314
            tmp2.seek(0)
315
            self.assertContainsRe(tmp2.read(),
7143.15.2 by Jelmer Vernooij
Run autopep8.
316
                                  b"\\d+\\.\\d+  comment to file2\n")
3173.1.12 by Martin Pool
Add test_push_log_file
317
        finally:
318
            tmp1.close()
319
            tmp2.close()
320
6622.1.33 by Jelmer Vernooij
Fix more tests (all?)
321
    def test__open_brz_log_uses_stderr_for_failures(self):
322
        # If _open_brz_log cannot open the file, then we should write the
4634.118.1 by John Arbash Meinel
Fix bug #503886, errors setting up logging go to stderr.
323
        # warning to stderr. Since this is normally happening before logging is
324
        # set up.
6677.1.5 by Martin
Make crash debug and trace modules pass on Python 3
325
        self.overrideAttr(sys, 'stderr', StringIO())
4634.118.1 by John Arbash Meinel
Fix bug #503886, errors setting up logging go to stderr.
326
        # Set the log file to something that cannot exist
6624 by Jelmer Vernooij
Merge Python3 porting work ('py3 pokes')
327
        self.overrideEnv('BRZ_LOG', '/no-such-dir/brz.log')
6622.1.33 by Jelmer Vernooij
Fix more tests (all?)
328
        self.overrideAttr(trace, '_brz_log_filename')
329
        logf = trace._open_brz_log()
7427.1.2 by Jelmer Vernooij
One more.
330
        if os.path.isdir('/no-such-dir'):
331
            raise TestSkipped('directory creation succeeded')
4634.118.1 by John Arbash Meinel
Fix bug #503886, errors setting up logging go to stderr.
332
        self.assertIs(None, logf)
6619.3.26 by Martin
Fix fallout from 2to3 getcwdu transformation and other test uses
333
        self.assertContainsRe(
334
            sys.stderr.getvalue(),
6624 by Jelmer Vernooij
Merge Python3 porting work ('py3 pokes')
335
            "failed to open trace file: .* '/no-such-dir/brz.log'$")
2768.1.10 by Ian Clatworthy
Add tests for new methods in trace.py
336
7371.1.1 by Jelmer Vernooij
Don't crash when we are unable to create the cache directory.
337
    def test__open_brz_log_ignores_cache_dir_error(self):
338
        # If the cache directory can not be created and _open_brz_log can thus
339
        # not open the file, then we should write the warning to stderr. Since
340
        # this is normally happening before logging is set up.
341
        self.overrideAttr(sys, 'stderr', StringIO())
342
        # Set the cache directory to something that cannot exist
343
        self.overrideEnv('BRZ_LOG', None)
344
        self.overrideEnv('BRZ_HOME', '/no-such-dir')
345
        self.overrideEnv('XDG_CACHE_HOME', '/no-such-dir')
346
        self.overrideAttr(trace, '_brz_log_filename')
347
        logf = trace._open_brz_log()
7427.1.1 by Jelmer Vernooij
Skip test when directory creation always succeeds.
348
        if os.path.isdir('/no-such-dir'):
349
            raise TestSkipped('directory creation succeeded')
7371.1.1 by Jelmer Vernooij
Don't crash when we are unable to create the cache directory.
350
        self.assertIs(None, logf)
351
        self.assertContainsRe(
352
            sys.stderr.getvalue(),
353
            "failed to open trace file: .* '/no-such-dir'$")
354
4985.1.5 by Vincent Ladeuil
Deploying the new overrideAttr facility further reduces the complexity
355
2768.1.10 by Ian Clatworthy
Add tests for new methods in trace.py
356
class TestVerbosityLevel(TestCase):
357
358
    def test_verbosity_level(self):
359
        set_verbosity_level(1)
360
        self.assertEqual(1, get_verbosity_level())
361
        self.assertTrue(is_verbose())
362
        self.assertFalse(is_quiet())
363
        set_verbosity_level(-1)
364
        self.assertEqual(-1, get_verbosity_level())
365
        self.assertFalse(is_verbose())
366
        self.assertTrue(is_quiet())
367
        set_verbosity_level(0)
368
        self.assertEqual(0, get_verbosity_level())
369
        self.assertFalse(is_verbose())
370
        self.assertFalse(is_quiet())
371
372
    def test_be_quiet(self):
373
        # Confirm the old API still works
374
        be_quiet(True)
375
        self.assertEqual(-1, get_verbosity_level())
376
        be_quiet(False)
377
        self.assertEqual(0, get_verbosity_level())
2851.3.1 by Martin Pool
Add unit test for _rollover_trace_maybe
378
379
6015.52.1 by Martin Packman
Add tests for logging various problematic values
380
class TestLogging(TestCase):
381
    """Check logging functionality robustly records information"""
382
383
    def test_note(self):
384
        trace.note("Noted")
385
        self.assertEqual("    INFO  Noted\n", self.get_log())
386
387
    def test_warning(self):
388
        trace.warning("Warned")
389
        self.assertEqual(" WARNING  Warned\n", self.get_log())
390
391
    def test_log(self):
6622.1.33 by Jelmer Vernooij
Fix more tests (all?)
392
        logging.getLogger("brz").error("Errored")
6015.52.1 by Martin Packman
Add tests for logging various problematic values
393
        self.assertEqual("   ERROR  Errored\n", self.get_log())
394
395
    def test_log_sub(self):
6622.1.33 by Jelmer Vernooij
Fix more tests (all?)
396
        logging.getLogger("brz.test_log_sub").debug("Whispered")
6015.52.1 by Martin Packman
Add tests for logging various problematic values
397
        self.assertEqual("   DEBUG  Whispered\n", self.get_log())
398
399
    def test_log_unicode_msg(self):
6622.1.33 by Jelmer Vernooij
Fix more tests (all?)
400
        logging.getLogger("brz").debug(u"\xa7")
6015.52.1 by Martin Packman
Add tests for logging various problematic values
401
        self.assertEqual(u"   DEBUG  \xa7\n", self.get_log())
402
403
    def test_log_unicode_arg(self):
6622.1.33 by Jelmer Vernooij
Fix more tests (all?)
404
        logging.getLogger("brz").debug("%s", u"\xa7")
6015.52.1 by Martin Packman
Add tests for logging various problematic values
405
        self.assertEqual(u"   DEBUG  \xa7\n", self.get_log())
406
407
    def test_log_utf8_msg(self):
6677.1.5 by Martin
Make crash debug and trace modules pass on Python 3
408
        logging.getLogger("brz").debug(b"\xc2\xa7")
6015.52.1 by Martin Packman
Add tests for logging various problematic values
409
        self.assertEqual(u"   DEBUG  \xa7\n", self.get_log())
410
411
    def test_log_utf8_arg(self):
6677.1.5 by Martin
Make crash debug and trace modules pass on Python 3
412
        logging.getLogger("brz").debug(b"%s", b"\xc2\xa7")
413
        if PY3:
414
            expected = u"   DEBUG  b'\\xc2\\xa7'\n"
415
        else:
416
            expected = u"   DEBUG  \xa7\n"
417
        self.assertEqual(expected, self.get_log())
6015.52.1 by Martin Packman
Add tests for logging various problematic values
418
419
    def test_log_bytes_msg(self):
6677.1.5 by Martin
Make crash debug and trace modules pass on Python 3
420
        logging.getLogger("brz").debug(b"\xa7")
6015.52.1 by Martin Packman
Add tests for logging various problematic values
421
        log = self.get_log()
422
        self.assertContainsString(log, "UnicodeDecodeError: ")
7143.15.5 by Jelmer Vernooij
More PEP8 fixes.
423
        self.assertContainsRe(
424
            log, "Logging record unformattable: b?'\\\\xa7' % \\(\\)\n")
6015.52.1 by Martin Packman
Add tests for logging various problematic values
425
426
    def test_log_bytes_arg(self):
6677.1.5 by Martin
Make crash debug and trace modules pass on Python 3
427
        logging.getLogger("brz").debug(b"%s", b"\xa7")
6015.52.1 by Martin Packman
Add tests for logging various problematic values
428
        log = self.get_log()
6677.1.5 by Martin
Make crash debug and trace modules pass on Python 3
429
        if PY3:
430
            self.assertEqual(u"   DEBUG  b'\\xa7'\n", self.get_log())
431
        else:
432
            self.assertContainsString(log, "UnicodeDecodeError: ")
7143.15.5 by Jelmer Vernooij
More PEP8 fixes.
433
            self.assertContainsRe(
434
                log,
435
                "Logging record unformattable: ?'%s' % \\(b?'\\\\xa7',\\)\n")
6015.52.1 by Martin Packman
Add tests for logging various problematic values
436
437
    def test_log_mixed_strings(self):
6677.1.5 by Martin
Make crash debug and trace modules pass on Python 3
438
        logging.getLogger("brz").debug(u"%s", b"\xa7")
6015.52.1 by Martin Packman
Add tests for logging various problematic values
439
        log = self.get_log()
6677.1.5 by Martin
Make crash debug and trace modules pass on Python 3
440
        if PY3:
441
            self.assertEqual(u"   DEBUG  b'\\xa7'\n", self.get_log())
442
        else:
443
            self.assertContainsString(log, "UnicodeDecodeError: ")
7143.15.5 by Jelmer Vernooij
More PEP8 fixes.
444
            self.assertContainsRe(
445
                log,
446
                "Logging record unformattable: u'%s' % \\('\\\\xa7',\\)\n")
6015.52.1 by Martin Packman
Add tests for logging various problematic values
447
448
    def test_log_repr_broken(self):
449
        class BadRepr(object):
450
            def __repr__(self):
451
                raise ValueError("Broken object")
6622.1.33 by Jelmer Vernooij
Fix more tests (all?)
452
        logging.getLogger("brz").debug("%s", BadRepr())
6015.52.1 by Martin Packman
Add tests for logging various problematic values
453
        log = self.get_log()
454
        self.assertContainsRe(log, "ValueError: Broken object\n")
455
        self.assertContainsRe(log, "Logging record unformattable: '%s' % .*\n")
456
457
2851.3.1 by Martin Pool
Add unit test for _rollover_trace_maybe
458
class TestBzrLog(TestCaseInTempDir):
459
460
    def test_log_rollover(self):
461
        temp_log_name = 'test-log'
462
        trace_file = open(temp_log_name, 'at')
4789.22.2 by John Arbash Meinel
Speed up a slow memory-hungry test that doesn't need to be.
463
        trace_file.writelines(['test_log_rollover padding\n'] * 200000)
2851.3.1 by Martin Pool
Add unit test for _rollover_trace_maybe
464
        trace_file.close()
465
        _rollover_trace_maybe(temp_log_name)
466
        # should have been rolled over
467
        self.assertFalse(os.access(temp_log_name, os.R_OK))
5320.2.5 by Robert Collins
Make bzrlib startup use a trace context manager.
468
469
470
class TestTraceConfiguration(TestCaseInTempDir):
471
472
    def test_default_config(self):
473
        config = trace.DefaultConfig()
6622.1.33 by Jelmer Vernooij
Fix more tests (all?)
474
        self.overrideAttr(trace, "_brz_log_filename", None)
475
        trace._brz_log_filename = None
476
        expected_filename = trace._get_brz_log_filename()
477
        self.assertEqual(None, trace._brz_log_filename)
5320.2.5 by Robert Collins
Make bzrlib startup use a trace context manager.
478
        config.__enter__()
479
        try:
480
            # Should have entered and setup a default filename.
6622.1.33 by Jelmer Vernooij
Fix more tests (all?)
481
            self.assertEqual(expected_filename, trace._brz_log_filename)
5320.2.5 by Robert Collins
Make bzrlib startup use a trace context manager.
482
        finally:
483
            config.__exit__(None, None, None)
484
            # Should have exited and cleaned up.
6622.1.33 by Jelmer Vernooij
Fix more tests (all?)
485
            self.assertEqual(None, trace._brz_log_filename)