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(""))
 
 
150
    def test_progress_nested(self):
 
 
151
        # test factory based nested and popping.
 
 
152
        ui = TextUIFactory(None, None, None)
 
 
153
        pb1 = ui.nested_progress_bar()
 
 
154
        pb2 = ui.nested_progress_bar()
 
 
155
        # You do get a warning if the outermost progress bar wasn't finished
 
 
156
        # first - it's not clear if this is really useful or if it should just
 
 
157
        # become orphaned -- mbp 20090120
 
 
158
        warnings, _ = self.callCatchWarnings(pb1.finished)
 
 
159
        if len(warnings) != 1:
 
 
160
            self.fail("unexpected warnings: %r" % (warnings,))
 
 
164
    def test_progress_stack(self):
 
 
165
        # test the progress bar stack which the default text factory
 
 
169
        # make a stack, which accepts parameters like a pb.
 
 
170
        stack = self.applyDeprecated(
 
 
171
            deprecated_in((1, 12, 0)),
 
 
173
            to_file=stderr, to_messages_file=stdout)
 
 
175
        self.assertFalse(getattr(stack, 'note', False))
 
 
176
        pb1 = stack.get_nested()
 
 
177
        pb2 = stack.get_nested()
 
 
178
        warnings, _ = self.callCatchWarnings(pb1.finished)
 
 
179
        self.assertEqual(len(warnings), 1)
 
 
182
        # the text ui factory never actually removes the stack once its setup.
 
 
183
        # we need to be able to nest again correctly from here.
 
 
184
        pb1 = stack.get_nested()
 
 
185
        pb2 = stack.get_nested()
 
 
186
        warnings, _ = self.callCatchWarnings(pb1.finished)
 
 
187
        self.assertEqual(len(warnings), 1)
 
 
191
    def assert_get_bool_acceptance_of_user_input(self, factory):
 
 
192
        factory.stdin = StringIO("y\nyes with garbage\n"
 
 
193
                                 "yes\nn\nnot an answer\n"
 
 
195
        factory.stdout = StringIO()
 
 
196
        # there is no output from the base factory
 
 
197
        self.assertEqual(True, factory.get_boolean(""))
 
 
198
        self.assertEqual(True, factory.get_boolean(""))
 
 
199
        self.assertEqual(False, factory.get_boolean(""))
 
147
200
        self.assertEqual(False, factory.get_boolean(""))
 
148
201
        self.assertEqual("foo\n", factory.stdin.read())
 
149
202
        # stdin should be empty
 
150
203
        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(""))
 
164
 
    def test_text_factory_prompt(self):
 
165
 
        # see <https://launchpad.net/bugs/365891>
 
166
 
        StringIO = tests.StringIOWrapper
 
167
 
        factory = _mod_ui_text.TextUIFactory(StringIO(), StringIO(), StringIO())
 
168
 
        factory.prompt('foo %2e')
 
169
 
        self.assertEqual('', factory.stdout.getvalue())
 
170
 
        self.assertEqual('foo %2e', factory.stderr.getvalue())
 
 
205
    def test_silent_ui_getbool(self):
 
 
206
        factory = SilentUIFactory()
 
 
207
        self.assert_get_bool_acceptance_of_user_input(factory)
 
 
209
    def test_silent_factory_prompts_silently(self):
 
 
210
        factory = SilentUIFactory()
 
 
212
        factory.stdin = StringIO("y\n")
 
 
213
        self.assertEqual(True,
 
 
214
                         self.apply_redirected(None, stdout, stdout,
 
 
215
                                               factory.get_boolean, "foo"))
 
 
216
        self.assertEqual("", stdout.getvalue())
 
 
217
        # stdin should be empty
 
 
218
        self.assertEqual('', factory.stdin.readline())
 
 
220
    def test_text_ui_getbool(self):
 
 
221
        factory = TextUIFactory(None, None, None)
 
 
222
        self.assert_get_bool_acceptance_of_user_input(factory)
 
172
224
    def test_text_factory_prompts_and_clears(self):
 
173
225
        # 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)
 
 
227
        factory = TextUIFactory(stdin=StringIO("yada\ny\n"), stdout=out, stderr=out)
 
179
228
        pb = factory.nested_progress_bar()
 
180
229
        pb.show_bar = False
 
181
230
        pb.show_spinner = False
 
 
209
 
    def test_text_ui_getusername(self):
 
210
 
        factory = _mod_ui_text.TextUIFactory(None, None, None)
 
211
 
        factory.stdin = tests.StringIOWrapper("someuser\n\n")
 
212
 
        factory.stdout = tests.StringIOWrapper()
 
213
 
        factory.stderr = tests.StringIOWrapper()
 
214
 
        factory.stdout.encoding = "utf8"
 
215
 
        # there is no output from the base factory
 
216
 
        self.assertEqual("someuser",
 
217
 
                         factory.get_username('Hello %(host)s', host='some'))
 
218
 
        self.assertEquals("Hello some: ", factory.stderr.getvalue())
 
219
 
        self.assertEquals('', factory.stdout.getvalue())
 
220
 
        self.assertEqual("", factory.get_username("Gebruiker"))
 
221
 
        # stdin should be empty
 
222
 
        self.assertEqual('', factory.stdin.readline())
 
224
 
    def test_text_ui_getusername_utf8(self):
 
225
 
        ui = tests.TestUIFactory(stdin=u'someuser\u1234'.encode('utf8'),
 
226
 
                                 stdout=tests.StringIOWrapper(),
 
227
 
                                 stderr=tests.StringIOWrapper())
 
228
 
        ui.stderr.encoding = ui.stdout.encoding = ui.stdin.encoding = "utf8"
 
229
 
        pb = ui.nested_progress_bar()
 
231
 
            # there is no output from the base factory
 
232
 
            username = self.apply_redirected(ui.stdin, ui.stdout, ui.stderr,
 
233
 
                ui.get_username, u'Hello\u1234 %(host)s', host=u'some\u1234')
 
234
 
            self.assertEquals(u"someuser\u1234", username.decode('utf8'))
 
235
 
            self.assertEquals(u"Hello\u1234 some\u1234: ",
 
236
 
                              ui.stderr.getvalue().decode("utf8"))
 
237
 
            self.assertEquals('', ui.stdout.getvalue())
 
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)
 
 
258
class TestTextProgressView(TestCase):
 
 
259
    """Tests for text display of progress bars.
 
 
261
    # XXX: These might be a bit easier to write if the rendering and
 
 
262
    # state-maintaining parts of TextProgressView were more separate, and if
 
 
263
    # the progress task called back directly to its own view not to the ui
 
 
264
    # factory. -- mbp 20090312
 
 
266
    def _make_factory(self):
 
 
268
        uif = TextUIFactory(stderr=out)
 
 
269
        uif._progress_view._width = 80
 
 
272
    def test_render_progress_easy(self):
 
 
273
        """Just one task and one quarter done"""
 
 
274
        out, uif = self._make_factory()
 
 
275
        task = uif.nested_progress_bar()
 
 
276
        task.update('reticulating splines', 5, 20)
 
 
278
'\r[####/               ] reticulating splines 5/20                               \r'
 
 
281
    def test_render_progress_nested(self):
 
 
282
        """Tasks proportionally contribute to overall progress"""
 
 
283
        out, uif = self._make_factory()
 
 
284
        task = uif.nested_progress_bar()
 
 
285
        task.update('reticulating splines', 0, 2)
 
 
286
        task2 = uif.nested_progress_bar()
 
 
287
        task2.update('stage2', 1, 2)
 
 
288
        # so we're in the first half of the main task, and half way through
 
 
291
r'[####\               ] reticulating splines:stage2 1/2'
 
 
292
            , uif._progress_view._render_line())
 
 
293
        # if the nested task is complete, then we're all the way through the
 
 
294
        # first half of the overall work
 
 
295
        task2.update('stage2', 2, 2)
 
 
297
r'[#########|          ] reticulating splines:stage2 2/2'
 
 
298
            , uif._progress_view._render_line())
 
 
300
    def test_render_progress_sub_nested(self):
 
 
301
        """Intermediate tasks don't mess up calculation."""
 
 
302
        out, uif = self._make_factory()
 
 
303
        task_a = uif.nested_progress_bar()
 
 
304
        task_a.update('a', 0, 2)
 
 
305
        task_b = uif.nested_progress_bar()
 
 
307
        task_c = uif.nested_progress_bar()
 
 
308
        task_c.update('c', 1, 2)
 
 
309
        # the top-level task is in its first half; the middle one has no
 
 
310
        # progress indication, just a label; and the bottom one is half done,
 
 
311
        # so the overall fraction is 1/4
 
 
313
            r'[####|               ] a:b:c 1/2'
 
 
314
            , uif._progress_view._render_line())