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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
15
# Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
17
17
# "weren't nothing promised to you. do i look like i got a promise face?"
28
28
from bzrlib import (
31
from bzrlib.tests import TestCaseInTempDir, TestCase
33
from bzrlib.tests import features, TestCaseInTempDir, TestCase
32
34
from bzrlib.trace import (
33
35
mutter, mutter_callsite, report_exception,
34
36
set_verbosity_level, get_verbosity_level, is_quiet, is_verbose, be_quiet,
37
39
_rollover_trace_maybe,
48
51
class TestTrace(TestCase):
50
53
def test_format_sys_exception(self):
54
# Test handling of an internal/unexpected error that probably
55
# indicates a bug in bzr. The details of the message may vary
56
# depending on whether apport is available or not. See test_crash for
52
59
raise NotImplementedError, "time travel"
53
60
except NotImplementedError:
68
75
self.assertTrue(len(msg) > 0)
69
76
self.assertEqualDiff(msg, 'bzr: interrupted\n')
78
def test_format_memory_error(self):
83
msg = _format_exception()
84
self.assertEquals(msg,
85
"bzr: out of memory\nUse -Dmem_dump to dump memory to a file.\n")
87
def test_format_mem_dump(self):
88
self.requireFeature(features.meliae)
89
debug.debug_flags.add('mem_dump')
94
msg = _format_exception()
95
self.assertStartsWith(msg,
96
"bzr: out of memory\nMemory dumped to ")
71
98
def test_format_os_error(self):
100
os.rmdir('nosuchfile22222')
103
msg = _format_exception()
104
# Linux seems to give "No such file" but Windows gives "The system
105
# cannot find the file specified".
106
self.assertEqual('bzr: ERROR: %s\n' % (e_str,), msg)
108
def test_format_io_error(self):
73
110
file('nosuchfile22222')
74
except (OSError, IOError):
76
msg = _format_exception()
77
self.assertContainsRe(msg, r'^bzr: ERROR: \[Errno .*\] No such file.*nosuchfile')
113
msg = _format_exception()
114
# Even though Windows and Linux differ for 'os.rmdir', they both give
115
# 'No such file' for open()
116
self.assertContainsRe(msg,
117
r'^bzr: ERROR: \[Errno .*\] No such file.*nosuchfile')
119
def test_format_pywintypes_error(self):
120
self.requireFeature(features.pywintypes)
121
import pywintypes, win32file
123
win32file.RemoveDirectory('nosuchfile22222')
124
except pywintypes.error:
126
msg = _format_exception()
127
# GZ 2010-05-03: Formatting for pywintypes.error is basic, a 3-tuple
128
# with errno, function name, and locale error message
129
self.assertContainsRe(msg,
130
r"^bzr: ERROR: \(2, 'RemoveDirectory[AW]?', .*\)")
132
def test_format_sockets_error(self):
135
sock = socket.socket()
136
sock.send("This should fail.")
139
msg = _format_exception()
141
self.assertNotContainsRe(msg,
142
r"Traceback (most recent call last):")
79
144
def test_format_unicode_error(self):
114
179
msg = _format_exception()
115
180
self.assertContainsRe(msg,
116
r"Traceback \(most recent call last\)")
181
r'Bazaar has encountered an internal error')
118
183
def test_trace_unicode(self):
119
184
"""Write Unicode to trace log"""
120
185
self.log(u'the unicode character for benzene is \N{BENZENE RING}')
121
self.assertContainsRe(self._get_log(keep_log_file=True),
122
"the unicode character for benzene is")
187
self.assertContainsRe(log, "the unicode character for benzene is")
124
189
def test_trace_argument_unicode(self):
125
190
"""Write a Unicode argument to the trace log"""
126
191
mutter(u'the unicode character for benzene is %s', u'\N{BENZENE RING}')
127
self.assertContainsRe(self._get_log(keep_log_file=True),
128
'the unicode character')
193
self.assertContainsRe(log, 'the unicode character')
130
195
def test_trace_argument_utf8(self):
131
196
"""Write a Unicode argument to the trace log"""
132
197
mutter(u'the unicode character for benzene is %s',
133
198
u'\N{BENZENE RING}'.encode('utf-8'))
134
self.assertContainsRe(self._get_log(keep_log_file=True),
135
'the unicode character')
200
self.assertContainsRe(log, 'the unicode character')
137
202
def test_report_broken_pipe(self):
151
216
def test_mutter_callsite_1(self):
152
217
"""mutter_callsite can capture 1 level of stack frame."""
153
218
mutter_callsite(1, "foo %s", "a string")
154
log = self._get_log(keep_log_file=True)
155
220
# begin with the message
156
221
self.assertLogStartsWith(log, 'foo a string\nCalled from:\n')
157
222
# should show two frame: this frame and the one above
163
228
def test_mutter_callsite_2(self):
164
229
"""mutter_callsite can capture 2 levels of stack frame."""
165
230
mutter_callsite(2, "foo %s", "a string")
166
log = self._get_log(keep_log_file=True)
167
232
# begin with the message
168
233
self.assertLogStartsWith(log, 'foo a string\nCalled from:\n')
169
234
# should show two frame: this frame and the one above
175
240
def test_mutter_never_fails(self):
176
241
# Even if the decode/encode stage fails, mutter should not
177
242
# raise an exception
243
# This test checks that mutter doesn't fail; the current behaviour
244
# is that it doesn't fail *and writes non-utf8*.
178
245
mutter(u'Writing a greek mu (\xb5) works in a unicode string')
179
246
mutter('But fails in an ascii string \xb5')
180
247
mutter('and in an ascii argument: %s', '\xb5')
181
log = self._get_log(keep_log_file=True)
182
249
self.assertContainsRe(log, 'Writing a greek mu')
183
250
self.assertContainsRe(log, "But fails in an ascii string")
184
self.assertContainsRe(log, u"ascii argument: \xb5")
251
# However, the log content object does unicode replacement on reading
252
# to let it get unicode back where good data has been written. So we
253
# have to do a replaceent here as well.
254
self.assertContainsRe(log, "ascii argument: \xb5".decode('utf8',
257
def test_show_error(self):
259
show_error(u'error2 \xb5 blah')
260
show_error('arg: %s', 'blah')
261
show_error('arg2: %(key)s', {'key':'stuff'})
263
raise Exception("oops")
265
show_error('kwarg', exc_info=True)
267
self.assertContainsRe(log, 'error1')
268
self.assertContainsRe(log, u'error2 \xb5 blah')
269
self.assertContainsRe(log, 'arg: blah')
270
self.assertContainsRe(log, 'arg2: stuff')
271
self.assertContainsRe(log, 'kwarg')
272
self.assertContainsRe(log, 'Traceback \\(most recent call last\\):')
273
self.assertContainsRe(log, 'File ".*test_trace.py", line .*, in test_show_error')
274
self.assertContainsRe(log, 'raise Exception\\("oops"\\)')
275
self.assertContainsRe(log, 'Exception: oops')
186
277
def test_push_log_file(self):
187
278
"""Can push and pop log file, and this catches mutter messages.
189
This is primarily for use in the test framework.
280
This is primarily for use in the test framework.
191
282
tmp1 = tempfile.NamedTemporaryFile()
192
283
tmp2 = tempfile.NamedTemporaryFile()
310
def test__open_bzr_log_uses_stderr_for_failures(self):
311
# If _open_bzr_log cannot open the file, then we should write the
312
# warning to stderr. Since this is normally happening before logging is
314
self.overrideAttr(sys, 'stderr', StringIO())
315
# Set the log file to something that cannot exist
316
self.overrideEnv('BZR_LOG', os.getcwd() + '/no-dir/bzr.log')
317
self.overrideAttr(trace, '_bzr_log_filename')
318
logf = trace._open_bzr_log()
319
self.assertIs(None, logf)
320
self.assertContainsRe(sys.stderr.getvalue(),
321
'failed to open trace file: .*/no-dir/bzr.log')
220
324
class TestVerbosityLevel(TestCase):
246
350
def test_log_rollover(self):
247
351
temp_log_name = 'test-log'
248
352
trace_file = open(temp_log_name, 'at')
249
trace_file.write('test_log_rollover padding\n' * 1000000)
353
trace_file.writelines(['test_log_rollover padding\n'] * 200000)
250
354
trace_file.close()
251
355
_rollover_trace_maybe(temp_log_name)
252
356
# should have been rolled over
253
357
self.assertFalse(os.access(temp_log_name, os.R_OK))
360
class TestTraceConfiguration(TestCaseInTempDir):
362
def test_default_config(self):
363
config = trace.DefaultConfig()
364
self.overrideAttr(trace, "_bzr_log_filename", None)
365
trace._bzr_log_filename = None
366
expected_filename = trace._get_bzr_log_filename()
367
self.assertEqual(None, trace._bzr_log_filename)
370
# Should have entered and setup a default filename.
371
self.assertEqual(expected_filename, trace._bzr_log_filename)
373
config.__exit__(None, None, None)
374
# Should have exited and cleaned up.
375
self.assertEqual(None, trace._bzr_log_filename)