131
def test_text_ui_get_boolean(self):
132
stdin = tests.StringIOWrapper("y\n" # True
134
"yes with garbage\nY\n" # True
135
"not an answer\nno\n" # False
136
"I'm sure!\nyes\n" # True
139
stdout = tests.StringIOWrapper()
140
stderr = tests.StringIOWrapper()
141
factory = _mod_ui_text.TextUIFactory(stdin, stdout, stderr)
142
self.assertEqual(True, factory.get_boolean(""))
143
self.assertEqual(False, factory.get_boolean(""))
144
self.assertEqual(True, factory.get_boolean(""))
145
self.assertEqual(False, factory.get_boolean(""))
146
self.assertEqual(True, factory.get_boolean(""))
153
def test_progress_nested(self):
154
# test factory based nested and popping.
155
ui = TextUIFactory(None, None, None)
156
pb1 = ui.nested_progress_bar()
157
pb2 = ui.nested_progress_bar()
158
# You do get a warning if the outermost progress bar wasn't finished
159
# first - it's not clear if this is really useful or if it should just
160
# become orphaned -- mbp 20090120
161
warnings, _ = self.callCatchWarnings(pb1.finished)
162
if len(warnings) != 1:
163
self.fail("unexpected warnings: %r" % (warnings,))
167
def test_progress_stack(self):
168
# test the progress bar stack which the default text factory
172
# make a stack, which accepts parameters like a pb.
173
stack = self.applyDeprecated(
174
deprecated_in((1, 12, 0)),
176
to_file=stderr, to_messages_file=stdout)
178
self.assertFalse(getattr(stack, 'note', False))
179
pb1 = stack.get_nested()
180
pb2 = stack.get_nested()
181
warnings, _ = self.callCatchWarnings(pb1.finished)
182
self.assertEqual(len(warnings), 1)
185
# the text ui factory never actually removes the stack once its setup.
186
# we need to be able to nest again correctly from here.
187
pb1 = stack.get_nested()
188
pb2 = stack.get_nested()
189
warnings, _ = self.callCatchWarnings(pb1.finished)
190
self.assertEqual(len(warnings), 1)
194
def assert_get_bool_acceptance_of_user_input(self, factory):
195
factory.stdin = StringIO("y\nyes with garbage\n"
196
"yes\nn\nnot an answer\n"
198
factory.stdout = StringIO()
199
factory.stderr = StringIO()
200
# there is no output from the base factory
201
self.assertEqual(True, factory.get_boolean(""))
202
self.assertEqual(True, factory.get_boolean(""))
203
self.assertEqual(False, factory.get_boolean(""))
147
204
self.assertEqual(False, factory.get_boolean(""))
148
205
self.assertEqual("foo\n", factory.stdin.read())
149
206
# stdin should be empty
150
207
self.assertEqual('', factory.stdin.readline())
152
def test_text_ui_get_integer(self):
153
stdin = tests.StringIOWrapper(
156
"hmmm\nwhat else ?\nCome on\nok 42\n4.24\n42\n")
157
stdout = tests.StringIOWrapper()
158
stderr = tests.StringIOWrapper()
159
factory = _mod_ui_text.TextUIFactory(stdin, stdout, stderr)
160
self.assertEqual(1, factory.get_integer(""))
161
self.assertEqual(-2, factory.get_integer(""))
162
self.assertEqual(42, factory.get_integer(""))
209
def test_silent_ui_getbool(self):
210
factory = SilentUIFactory()
211
self.assert_get_bool_acceptance_of_user_input(factory)
213
def test_silent_factory_prompts_silently(self):
214
factory = SilentUIFactory()
216
factory.stdin = StringIO("y\n")
217
self.assertEqual(True,
218
self.apply_redirected(None, stdout, stdout,
219
factory.get_boolean, "foo"))
220
self.assertEqual("", stdout.getvalue())
221
# stdin should be empty
222
self.assertEqual('', factory.stdin.readline())
224
def test_text_ui_getbool(self):
225
factory = TextUIFactory(None, None, None)
226
self.assert_get_bool_acceptance_of_user_input(factory)
164
228
def test_text_factory_prompt(self):
165
229
# see <https://launchpad.net/bugs/365891>
166
StringIO = tests.StringIOWrapper
167
factory = _mod_ui_text.TextUIFactory(StringIO(), StringIO(), StringIO())
230
factory = TextUIFactory(None, StringIO(), StringIO(), StringIO())
168
231
factory.prompt('foo %2e')
169
232
self.assertEqual('', factory.stdout.getvalue())
170
233
self.assertEqual('foo %2e', factory.stderr.getvalue())
172
235
def test_text_factory_prompts_and_clears(self):
173
236
# a get_boolean call should clear the pb before prompting
174
out = test_progress._TTYStringIO()
175
os.environ['TERM'] = 'xterm'
176
factory = _mod_ui_text.TextUIFactory(
177
stdin=tests.StringIOWrapper("yada\ny\n"),
178
stdout=out, stderr=out)
238
factory = TextUIFactory(stdin=StringIO("yada\ny\n"), stdout=out, stderr=out)
179
239
pb = factory.nested_progress_bar()
180
240
pb.show_bar = False
181
241
pb.show_spinner = False
241
def test_quietness(self):
242
os.environ['BZR_PROGRESS_BAR'] = 'text'
243
ui_factory = _mod_ui_text.TextUIFactory(None,
244
test_progress._TTYStringIO(),
245
test_progress._TTYStringIO())
246
self.assertIsInstance(ui_factory._progress_view,
247
_mod_ui_text.TextProgressView)
248
ui_factory.be_quiet(True)
249
self.assertIsInstance(ui_factory._progress_view,
250
_mod_ui_text.NullProgressView)
252
def test_text_ui_show_user_warning(self):
253
from bzrlib.repofmt.groupcompress_repo import RepositoryFormat2a
254
from bzrlib.repofmt.pack_repo import RepositoryFormatKnitPack5
257
ui = tests.TextUIFactory(stdin=None, stdout=out, stderr=err)
258
remote_fmt = remote.RemoteRepositoryFormat()
259
remote_fmt._network_name = RepositoryFormatKnitPack5().network_name()
260
ui.show_user_warning('cross_format_fetch', from_format=RepositoryFormat2a(),
261
to_format=remote_fmt)
262
self.assertEquals('', out.getvalue())
263
self.assertEquals("Doing on-the-fly conversion from RepositoryFormat2a() to "
264
"RemoteRepositoryFormat(_network_name='Bazaar RepositoryFormatKnitPack5 "
265
"(bzr 1.6)\\n').\nThis may take some time. Upgrade the repositories to "
266
"the same format for better performance.\n",
268
# and now with it suppressed please
271
ui = tests.TextUIFactory(stdin=None, stdout=out, stderr=err)
272
ui.suppressed_warnings.add('cross_format_fetch')
273
ui.show_user_warning('cross_format_fetch', from_format=RepositoryFormat2a(),
274
to_format=remote_fmt)
275
self.assertEquals('', out.getvalue())
276
self.assertEquals('', err.getvalue())
279
class TestTextUIOutputStream(tests.TestCase):
280
"""Tests for output stream that synchronizes with progress bar."""
282
def test_output_clears_terminal(self):
283
stdout = tests.StringIOWrapper()
284
stderr = tests.StringIOWrapper()
287
uif = _mod_ui_text.TextUIFactory(None, stdout, stderr)
288
uif.clear_term = lambda: clear_calls.append('clear')
290
stream = _mod_ui_text.TextUIOutputStream(uif, uif.stdout)
291
stream.write("Hello world!\n")
292
stream.write("there's more...\n")
293
stream.writelines(["1\n", "2\n", "3\n"])
295
self.assertEqual(stdout.getvalue(),
299
self.assertEqual(['clear', 'clear', 'clear'],
305
class UITests(tests.TestCase):
307
def test_progress_construction(self):
308
"""TextUIFactory constructs the right progress view.
310
TTYStringIO = test_progress._TTYStringIO
311
FileStringIO = tests.StringIOWrapper
312
for (file_class, term, pb, expected_pb_class) in (
313
# on an xterm, either use them or not as the user requests,
314
# otherwise default on
315
(TTYStringIO, 'xterm', 'none', _mod_ui_text.NullProgressView),
316
(TTYStringIO, 'xterm', 'text', _mod_ui_text.TextProgressView),
317
(TTYStringIO, 'xterm', None, _mod_ui_text.TextProgressView),
318
# on a dumb terminal, again if there's explicit configuration do
319
# it, otherwise default off
320
(TTYStringIO, 'dumb', 'none', _mod_ui_text.NullProgressView),
321
(TTYStringIO, 'dumb', 'text', _mod_ui_text.TextProgressView),
322
(TTYStringIO, 'dumb', None, _mod_ui_text.NullProgressView),
323
# on a non-tty terminal, it's null regardless of $TERM
324
(FileStringIO, 'xterm', None, _mod_ui_text.NullProgressView),
325
(FileStringIO, 'dumb', None, _mod_ui_text.NullProgressView),
326
# however, it can still be forced on
327
(FileStringIO, 'dumb', 'text', _mod_ui_text.TextProgressView),
329
os.environ['TERM'] = term
331
if 'BZR_PROGRESS_BAR' in os.environ:
332
del os.environ['BZR_PROGRESS_BAR']
334
os.environ['BZR_PROGRESS_BAR'] = pb
335
stdin = file_class('')
336
stderr = file_class()
337
stdout = file_class()
338
uif = _mod_ui.make_ui_for_terminal(stdin, stdout, stderr)
339
self.assertIsInstance(uif, _mod_ui_text.TextUIFactory,
340
"TERM=%s BZR_PROGRESS_BAR=%s uif=%r" % (term, pb, uif,))
341
self.assertIsInstance(uif.make_progress_view(),
343
"TERM=%s BZR_PROGRESS_BAR=%s uif=%r" % (term, pb, uif,))
345
def test_text_ui_non_terminal(self):
346
"""Even on non-ttys, make_ui_for_terminal gives a text ui."""
347
stdin = test_progress._NonTTYStringIO('')
348
stderr = test_progress._NonTTYStringIO()
349
stdout = test_progress._NonTTYStringIO()
350
for term_type in ['dumb', None, 'xterm']:
351
if term_type is None:
352
del os.environ['TERM']
354
os.environ['TERM'] = term_type
355
uif = _mod_ui.make_ui_for_terminal(stdin, stdout, stderr)
356
self.assertIsInstance(uif, _mod_ui_text.TextUIFactory,
357
'TERM=%r' % (term_type,))
360
class SilentUITests(tests.TestCase):
362
def test_silent_factory_get_password(self):
363
# A silent factory that can't do user interaction can't get a
364
# password. Possibly it should raise a more specific error but it
366
ui = _mod_ui.SilentUIFactory()
367
stdout = tests.StringIOWrapper()
370
self.apply_redirected,
371
None, stdout, stdout, ui.get_password)
372
# and it didn't write anything out either
373
self.assertEqual('', stdout.getvalue())
375
def test_silent_ui_getbool(self):
376
factory = _mod_ui.SilentUIFactory()
377
stdout = tests.StringIOWrapper()
380
self.apply_redirected,
381
None, stdout, stdout, factory.get_boolean, "foo")
384
class TestUIFactoryTests(tests.TestCase):
386
def test_test_ui_factory_progress(self):
387
# there's no output; we just want to make sure this doesn't crash -
388
# see https://bugs.edge.launchpad.net/bzr/+bug/408201
389
ui = tests.TestUIFactory()
390
pb = ui.nested_progress_bar()
396
class CannedInputUIFactoryTests(tests.TestCase):
398
def test_canned_input_get_input(self):
399
uif = _mod_ui.CannedInputUIFactory([True, 'mbp', 'password', 42])
400
self.assertEqual(True, uif.get_boolean('Extra cheese?'))
401
self.assertEqual('mbp', uif.get_username('Enter your user name'))
402
self.assertEqual('password',
403
uif.get_password('Password for %(host)s',
405
self.assertEqual(42, uif.get_integer('And all that jazz ?'))
408
class TestBoolFromString(tests.TestCase):
410
def assertIsTrue(self, s, accepted_values=None):
411
res = _mod_ui.bool_from_string(s, accepted_values=accepted_values)
412
self.assertEquals(True, res)
414
def assertIsFalse(self, s, accepted_values=None):
415
res = _mod_ui.bool_from_string(s, accepted_values=accepted_values)
416
self.assertEquals(False, res)
418
def assertIsNone(self, s, accepted_values=None):
419
res = _mod_ui.bool_from_string(s, accepted_values=accepted_values)
420
self.assertIs(None, res)
422
def test_know_valid_values(self):
423
self.assertIsTrue('true')
424
self.assertIsFalse('false')
425
self.assertIsTrue('1')
426
self.assertIsFalse('0')
427
self.assertIsTrue('on')
428
self.assertIsFalse('off')
429
self.assertIsTrue('yes')
430
self.assertIsFalse('no')
431
self.assertIsTrue('y')
432
self.assertIsFalse('n')
433
# Also try some case variations
434
self.assertIsTrue('True')
435
self.assertIsFalse('False')
436
self.assertIsTrue('On')
437
self.assertIsFalse('Off')
438
self.assertIsTrue('ON')
439
self.assertIsFalse('OFF')
440
self.assertIsTrue('oN')
441
self.assertIsFalse('oFf')
443
def test_invalid_values(self):
444
self.assertIsNone(None)
445
self.assertIsNone('doubt')
446
self.assertIsNone('frue')
447
self.assertIsNone('talse')
448
self.assertIsNone('42')
450
def test_provided_values(self):
451
av = dict(y=True, n=False, yes=True, no=False)
452
self.assertIsTrue('y', av)
453
self.assertIsTrue('Y', av)
454
self.assertIsTrue('Yes', av)
455
self.assertIsFalse('n', av)
456
self.assertIsFalse('N', av)
457
self.assertIsFalse('No', av)
458
self.assertIsNone('1', av)
459
self.assertIsNone('0', av)
460
self.assertIsNone('on', av)
461
self.assertIsNone('off', av)
311
class TestTextProgressView(TestCase):
312
"""Tests for text display of progress bars.
314
# XXX: These might be a bit easier to write if the rendering and
315
# state-maintaining parts of TextProgressView were more separate, and if
316
# the progress task called back directly to its own view not to the ui
317
# factory. -- mbp 20090312
319
def _make_factory(self):
321
uif = TextUIFactory(stderr=out)
322
uif._progress_view._width = 80
325
def test_render_progress_easy(self):
326
"""Just one task and one quarter done"""
327
out, uif = self._make_factory()
328
task = uif.nested_progress_bar()
329
task.update('reticulating splines', 5, 20)
331
'\r[####/ ] reticulating splines 5/20 \r'
334
def test_render_progress_nested(self):
335
"""Tasks proportionally contribute to overall progress"""
336
out, uif = self._make_factory()
337
task = uif.nested_progress_bar()
338
task.update('reticulating splines', 0, 2)
339
task2 = uif.nested_progress_bar()
340
task2.update('stage2', 1, 2)
341
# so we're in the first half of the main task, and half way through
344
r'[####\ ] reticulating splines:stage2 1/2'
345
, uif._progress_view._render_line())
346
# if the nested task is complete, then we're all the way through the
347
# first half of the overall work
348
task2.update('stage2', 2, 2)
350
r'[#########| ] reticulating splines:stage2 2/2'
351
, uif._progress_view._render_line())
353
def test_render_progress_sub_nested(self):
354
"""Intermediate tasks don't mess up calculation."""
355
out, uif = self._make_factory()
356
task_a = uif.nested_progress_bar()
357
task_a.update('a', 0, 2)
358
task_b = uif.nested_progress_bar()
360
task_c = uif.nested_progress_bar()
361
task_c.update('c', 1, 2)
362
# the top-level task is in its first half; the middle one has no
363
# progress indication, just a label; and the bottom one is half done,
364
# so the overall fraction is 1/4
366
r'[####| ] a:b:c 1/2'
367
, uif._progress_view._render_line())