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
17
17
# "weren't nothing promised to you. do i look like i got a promise face?"
50
48
class TestTrace(TestCase):
52
50
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
58
52
raise NotImplementedError, "time travel"
59
53
except NotImplementedError:
62
56
self.assertEqualDiff(err.splitlines()[0],
63
57
'bzr: ERROR: exceptions.NotImplementedError: time travel')
64
58
self.assertContainsRe(err,
65
'Bazaar has encountered an internal error.')
59
r'File.*test_trace.py')
67
61
def test_format_interrupt_exception(self):
74
68
self.assertTrue(len(msg) > 0)
75
69
self.assertEqualDiff(msg, 'bzr: interrupted\n')
77
def test_format_memory_error(self):
82
msg = _format_exception()
83
self.assertEquals(msg,
84
"bzr: out of memory\n")
86
71
def test_format_os_error(self):
88
os.rmdir('nosuchfile22222')
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)
96
def test_format_io_error(self):
98
73
file('nosuchfile22222')
74
except (OSError, IOError):
101
76
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')
77
self.assertContainsRe(msg, r'^bzr: ERROR: \[Errno .*\] No such file.*nosuchfile')
107
79
def test_format_unicode_error(self):
121
93
self.assertTrue(len(msg) > 0)
122
94
self.assertEqualDiff(msg, 'bzr: ERROR: Not a branch: \"wibble\".\n')
124
def test_report_external_import_error(self):
125
"""Short friendly message for missing system modules."""
127
import ImaginaryModule
128
except ImportError, e:
131
self.fail("somehow succeeded in importing %r" % ImaginaryModule)
132
msg = _format_exception()
133
self.assertEqual(msg,
134
'bzr: ERROR: No module named ImaginaryModule\n'
135
'You may need to install this Python library separately.\n')
137
def test_report_import_syntax_error(self):
139
raise ImportError("syntax error")
140
except ImportError, e:
142
msg = _format_exception()
143
self.assertContainsRe(msg,
144
r'Bazaar has encountered an internal error')
146
96
def test_trace_unicode(self):
147
97
"""Write Unicode to trace log"""
148
98
self.log(u'the unicode character for benzene is \N{BENZENE RING}')
150
self.assertContainsRe(log, "the unicode character for benzene is")
99
self.assertContainsRe(self._get_log(keep_log_file=True),
100
"the unicode character for benzene is")
152
102
def test_trace_argument_unicode(self):
153
103
"""Write a Unicode argument to the trace log"""
154
104
mutter(u'the unicode character for benzene is %s', u'\N{BENZENE RING}')
156
self.assertContainsRe(log, 'the unicode character')
105
self.assertContainsRe(self._get_log(keep_log_file=True),
106
'the unicode character')
158
108
def test_trace_argument_utf8(self):
159
109
"""Write a Unicode argument to the trace log"""
160
110
mutter(u'the unicode character for benzene is %s',
161
111
u'\N{BENZENE RING}'.encode('utf-8'))
163
self.assertContainsRe(log, 'the unicode character')
112
self.assertContainsRe(self._get_log(keep_log_file=True),
113
'the unicode character')
165
115
def test_report_broken_pipe(self):
179
129
def test_mutter_callsite_1(self):
180
130
"""mutter_callsite can capture 1 level of stack frame."""
181
131
mutter_callsite(1, "foo %s", "a string")
132
log = self._get_log(keep_log_file=True)
183
133
# begin with the message
184
134
self.assertLogStartsWith(log, 'foo a string\nCalled from:\n')
185
135
# should show two frame: this frame and the one above
191
141
def test_mutter_callsite_2(self):
192
142
"""mutter_callsite can capture 2 levels of stack frame."""
193
143
mutter_callsite(2, "foo %s", "a string")
144
log = self._get_log(keep_log_file=True)
195
145
# begin with the message
196
146
self.assertLogStartsWith(log, 'foo a string\nCalled from:\n')
197
147
# should show two frame: this frame and the one above
203
153
def test_mutter_never_fails(self):
204
154
# Even if the decode/encode stage fails, mutter should not
205
155
# raise an exception
206
# This test checks that mutter doesn't fail; the current behaviour
207
# is that it doesn't fail *and writes non-utf8*.
208
156
mutter(u'Writing a greek mu (\xb5) works in a unicode string')
209
157
mutter('But fails in an ascii string \xb5')
210
158
mutter('and in an ascii argument: %s', '\xb5')
159
log = self._get_log(keep_log_file=True)
212
160
self.assertContainsRe(log, 'Writing a greek mu')
213
161
self.assertContainsRe(log, "But fails in an ascii string")
214
# However, the log content object does unicode replacement on reading
215
# to let it get unicode back where good data has been written. So we
216
# have to do a replaceent here as well.
217
self.assertContainsRe(log, "ascii argument: \xb5".decode('utf8',
220
def test_show_error(self):
222
show_error(u'error2 \xb5 blah')
223
show_error('arg: %s', 'blah')
224
show_error('arg2: %(key)s', {'key':'stuff'})
226
raise Exception("oops")
228
show_error('kwarg', exc_info=True)
230
self.assertContainsRe(log, 'error1')
231
self.assertContainsRe(log, u'error2 \xb5 blah')
232
self.assertContainsRe(log, 'arg: blah')
233
self.assertContainsRe(log, 'arg2: stuff')
234
self.assertContainsRe(log, 'kwarg')
235
self.assertContainsRe(log, 'Traceback \\(most recent call last\\):')
236
self.assertContainsRe(log, 'File ".*test_trace.py", line .*, in test_show_error')
237
self.assertContainsRe(log, 'raise Exception\\("oops"\\)')
238
self.assertContainsRe(log, 'Exception: oops')
162
self.assertContainsRe(log, u"ascii argument: \xb5")
240
164
def test_push_log_file(self):
241
165
"""Can push and pop log file, and this catches mutter messages.
243
This is primarily for use in the test framework.
167
This is primarily for use in the test framework.
245
169
tmp1 = tempfile.NamedTemporaryFile()
246
170
tmp2 = tempfile.NamedTemporaryFile()
273
def test__open_bzr_log_uses_stderr_for_failures(self):
274
# If _open_bzr_log cannot open the file, then we should write the
275
# warning to stderr. Since this is normally happening before logging is
277
self.overrideAttr(sys, 'stderr', StringIO())
278
# Set the log file to something that cannot exist
279
# FIXME: A bit dangerous: we are not in an isolated dir here -- vilajam
281
os.environ['BZR_LOG'] = os.getcwd() + '/no-dir/bzr.log'
282
self.overrideAttr(trace, '_bzr_log_filename')
283
logf = trace._open_bzr_log()
284
self.assertIs(None, logf)
285
self.assertContainsRe(sys.stderr.getvalue(),
286
'failed to open trace file: .*/no-dir/bzr.log')
289
198
class TestVerbosityLevel(TestCase):
315
224
def test_log_rollover(self):
316
225
temp_log_name = 'test-log'
317
226
trace_file = open(temp_log_name, 'at')
318
trace_file.writelines(['test_log_rollover padding\n'] * 200000)
227
trace_file.write('test_log_rollover padding\n' * 1000000)
319
228
trace_file.close()
320
229
_rollover_trace_maybe(temp_log_name)
321
230
# should have been rolled over