1
# Copyright (C) 2005, 2008, 2009 Canonical Ltd
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.
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.
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
15
# Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
17
"""Tests for the bzrlib ui
21
from StringIO import StringIO
31
from bzrlib.symbol_versioning import (
34
from bzrlib.tests import (
39
from bzrlib.tests.test_progress import (
43
from bzrlib.ui import (
50
from bzrlib.ui.text import (
57
class UITests(tests.TestCase):
59
def test_text_factory_ascii_password(self):
60
ui = tests.TestUIFactory(stdin='secret\n',
61
stdout=tests.StringIOWrapper(),
62
stderr=tests.StringIOWrapper())
63
pb = ui.nested_progress_bar()
65
self.assertEqual('secret',
66
self.apply_redirected(ui.stdin, ui.stdout,
69
# ': ' is appended to prompt
70
self.assertEqual(': ', ui.stderr.getvalue())
71
self.assertEqual('', ui.stdout.readline())
72
# stdin should be empty
73
self.assertEqual('', ui.stdin.readline())
77
def test_text_factory_utf8_password(self):
78
"""Test an utf8 password.
80
We can't predict what encoding users will have for stdin, so we force
81
it to utf8 to test that we transport the password correctly.
83
ui = tests.TestUIFactory(stdin=u'baz\u1234'.encode('utf8'),
84
stdout=tests.StringIOWrapper(),
85
stderr=tests.StringIOWrapper())
86
ui.stderr.encoding = ui.stdout.encoding = ui.stdin.encoding = 'utf8'
87
pb = ui.nested_progress_bar()
89
password = self.apply_redirected(ui.stdin, ui.stdout, ui.stderr,
91
u'Hello \u1234 %(user)s',
93
# We use StringIO objects, we need to decode them
94
self.assertEqual(u'baz\u1234', password.decode('utf8'))
95
self.assertEqual(u'Hello \u1234 some\u1234: ',
96
ui.stderr.getvalue().decode('utf8'))
97
# stdin and stdout should be empty
98
self.assertEqual('', ui.stdin.readline())
99
self.assertEqual('', ui.stdout.readline())
103
def test_progress_construction(self):
104
"""TextUIFactory constructs the right progress view.
106
for (file_class, term, pb, expected_pb_class) in (
107
# on an xterm, either use them or not as the user requests,
108
# otherwise default on
109
(_TTYStringIO, 'xterm', 'none', NullProgressView),
110
(_TTYStringIO, 'xterm', 'text', TextProgressView),
111
(_TTYStringIO, 'xterm', None, TextProgressView),
112
# on a dumb terminal, again if there's explicit configuration do
113
# it, otherwise default off
114
(_TTYStringIO, 'dumb', 'none', NullProgressView),
115
(_TTYStringIO, 'dumb', 'text', TextProgressView),
116
(_TTYStringIO, 'dumb', None, NullProgressView),
117
# on a non-tty terminal, it's null regardless of $TERM
118
(StringIO, 'xterm', None, NullProgressView),
119
(StringIO, 'dumb', None, NullProgressView),
120
# however, it can still be forced on
121
(StringIO, 'dumb', 'text', TextProgressView),
123
os.environ['TERM'] = term
125
if 'BZR_PROGRESS_BAR' in os.environ:
126
del os.environ['BZR_PROGRESS_BAR']
128
os.environ['BZR_PROGRESS_BAR'] = pb
129
stdin = file_class('')
130
stderr = file_class()
131
stdout = file_class()
132
uif = make_ui_for_terminal(stdin, stdout, stderr)
133
self.assertIsInstance(uif, TextUIFactory,
134
"TERM=%s BZR_PROGRESS_BAR=%s uif=%r" % (term, pb, uif,))
135
self.assertIsInstance(uif.make_progress_view(),
137
"TERM=%s BZR_PROGRESS_BAR=%s uif=%r" % (term, pb, uif,))
139
def test_text_ui_non_terminal(self):
140
"""Even on non-ttys, make_ui_for_terminal gives a text ui."""
141
stdin = _NonTTYStringIO('')
142
stderr = _NonTTYStringIO()
143
stdout = _NonTTYStringIO()
144
for term_type in ['dumb', None, 'xterm']:
145
if term_type is None:
146
del os.environ['TERM']
148
os.environ['TERM'] = term_type
149
uif = make_ui_for_terminal(stdin, stdout, stderr)
150
self.assertIsInstance(uif, TextUIFactory,
151
'TERM=%r' % (term_type,))
153
def test_progress_note(self):
156
ui_factory = TextUIFactory(stdin=StringIO(''),
159
pb = ui_factory.nested_progress_bar()
161
result = self.applyDeprecated(deprecated_in((2, 1, 0)),
164
self.assertEqual(None, result)
165
self.assertEqual("t\n", stdout.getvalue())
166
# Since there was no update() call, there should be no clear() call
167
self.failIf(re.search(r'^\r {10,}\r$',
168
stderr.getvalue()) is not None,
169
'We cleared the stderr without anything to put there')
173
def test_progress_note_clears(self):
174
stderr = _TTYStringIO()
175
stdout = _TTYStringIO()
176
# so that we get a TextProgressBar
177
os.environ['TERM'] = 'xterm'
178
ui_factory = TextUIFactory(
180
stdout=stdout, stderr=stderr)
181
self.assertIsInstance(ui_factory._progress_view,
183
pb = ui_factory.nested_progress_bar()
185
# Create a progress update that isn't throttled
187
result = self.applyDeprecated(deprecated_in((2, 1, 0)),
189
self.assertEqual(None, result)
190
self.assertEqual("t\n", stdout.getvalue())
191
# the exact contents will depend on the terminal width and we don't
192
# care about that right now - but you're probably running it on at
193
# least a 10-character wide terminal :)
194
self.assertContainsRe(stderr.getvalue(), r'\r {10,}\r$')
198
def test_progress_nested(self):
199
# test factory based nested and popping.
200
ui = TextUIFactory(None, None, None)
201
pb1 = ui.nested_progress_bar()
202
pb2 = ui.nested_progress_bar()
203
# You do get a warning if the outermost progress bar wasn't finished
204
# first - it's not clear if this is really useful or if it should just
205
# become orphaned -- mbp 20090120
206
warnings, _ = self.callCatchWarnings(pb1.finished)
207
if len(warnings) != 1:
208
self.fail("unexpected warnings: %r" % (warnings,))
212
def test_text_ui_get_boolean(self):
213
stdin = StringIO("y\n" # True
215
"yes with garbage\nY\n" # True
216
"not an answer\nno\n" # False
217
"I'm sure!\nyes\n" # True
222
factory = TextUIFactory(stdin, stdout, stderr)
223
self.assertEqual(True, factory.get_boolean(""))
224
self.assertEqual(False, factory.get_boolean(""))
225
self.assertEqual(True, factory.get_boolean(""))
226
self.assertEqual(False, factory.get_boolean(""))
227
self.assertEqual(True, factory.get_boolean(""))
228
self.assertEqual(False, factory.get_boolean(""))
229
self.assertEqual("foo\n", factory.stdin.read())
230
# stdin should be empty
231
self.assertEqual('', factory.stdin.readline())
233
def test_text_factory_prompt(self):
234
# see <https://launchpad.net/bugs/365891>
235
factory = TextUIFactory(StringIO(), StringIO(), StringIO())
236
factory.prompt('foo %2e')
237
self.assertEqual('', factory.stdout.getvalue())
238
self.assertEqual('foo %2e', factory.stderr.getvalue())
240
def test_text_factory_prompts_and_clears(self):
241
# a get_boolean call should clear the pb before prompting
243
os.environ['TERM'] = 'xterm'
244
factory = TextUIFactory(stdin=StringIO("yada\ny\n"), stdout=out, stderr=out)
245
pb = factory.nested_progress_bar()
247
pb.show_spinner = False
248
pb.show_count = False
249
pb.update("foo", 0, 1)
250
self.assertEqual(True,
251
self.apply_redirected(None, factory.stdout,
255
output = out.getvalue()
256
self.assertContainsRe(factory.stdout.getvalue(),
258
self.assertContainsRe(factory.stdout.getvalue(),
259
r"what do you want\? \[y/n\]: what do you want\? \[y/n\]: ")
260
# stdin should have been totally consumed
261
self.assertEqual('', factory.stdin.readline())
263
def test_text_tick_after_update(self):
264
ui_factory = TextUIFactory(stdout=StringIO(), stderr=StringIO())
265
pb = ui_factory.nested_progress_bar()
267
pb.update('task', 0, 3)
268
# Reset the clock, so that it actually tries to repaint itself
269
ui_factory._progress_view._last_repaint = time.time() - 1.0
274
def test_text_ui_getusername(self):
275
factory = TextUIFactory(None, None, None)
276
factory.stdin = StringIO("someuser\n\n")
277
factory.stdout = StringIO()
278
factory.stderr = StringIO()
279
factory.stdout.encoding = "utf8"
280
# there is no output from the base factory
281
self.assertEqual("someuser",
282
factory.get_username('Hello %(host)s', host='some'))
283
self.assertEquals("Hello some: ", factory.stderr.getvalue())
284
self.assertEquals('', factory.stdout.getvalue())
285
self.assertEqual("", factory.get_username("Gebruiker"))
286
# stdin should be empty
287
self.assertEqual('', factory.stdin.readline())
289
def test_text_ui_getusername_utf8(self):
290
ui = tests.TestUIFactory(stdin=u'someuser\u1234'.encode('utf8'),
291
stdout=tests.StringIOWrapper(),
292
stderr=tests.StringIOWrapper())
293
ui.stderr.encoding = ui.stdout.encoding = ui.stdin.encoding = "utf8"
294
pb = ui.nested_progress_bar()
296
# there is no output from the base factory
297
username = self.apply_redirected(ui.stdin, ui.stdout, ui.stderr,
298
ui.get_username, u'Hello\u1234 %(host)s', host=u'some\u1234')
299
self.assertEquals(u"someuser\u1234", username.decode('utf8'))
300
self.assertEquals(u"Hello\u1234 some\u1234: ",
301
ui.stderr.getvalue().decode("utf8"))
302
self.assertEquals('', ui.stdout.getvalue())
307
class CLIUITests(TestCase):
309
def test_cli_factory_deprecated(self):
310
uif = self.applyDeprecated(deprecated_in((1, 18, 0)),
312
StringIO(), StringIO(), StringIO())
313
self.assertIsInstance(uif, UIFactory)
316
class SilentUITests(TestCase):
318
def test_silent_factory_get_password(self):
319
# A silent factory that can't do user interaction can't get a
320
# password. Possibly it should raise a more specific error but it
322
ui = SilentUIFactory()
326
self.apply_redirected,
327
None, stdout, stdout, ui.get_password)
328
# and it didn't write anything out either
329
self.assertEqual('', stdout.getvalue())
331
def test_silent_ui_getbool(self):
332
factory = SilentUIFactory()
336
self.apply_redirected,
337
None, stdout, stdout, factory.get_boolean, "foo")
340
class TestUIFactoryTests(TestCase):
342
def test_test_ui_factory_progress(self):
343
# there's no output; we just want to make sure this doesn't crash -
344
# see https://bugs.edge.launchpad.net/bzr/+bug/408201
346
pb = ui.nested_progress_bar()
352
class CannedInputUIFactoryTests(TestCase):
354
def test_canned_input_get_input(self):
355
uif = CannedInputUIFactory([True, 'mbp', 'password'])
356
self.assertEqual(uif.get_boolean('Extra cheese?'), True)
357
self.assertEqual(uif.get_username('Enter your user name'), 'mbp')
358
self.assertEqual(uif.get_password('Password for %(host)s', host='example.com'),
362
class TestBoolFromString(tests.TestCase):
364
def assertIsTrue(self, s, accepted_values=None):
365
res = _mod_ui.bool_from_string(s, accepted_values=accepted_values)
366
self.assertEquals(True, res)
368
def assertIsFalse(self, s, accepted_values=None):
369
res = _mod_ui.bool_from_string(s, accepted_values=accepted_values)
370
self.assertEquals(False, res)
372
def assertIsNone(self, s, accepted_values=None):
373
res = _mod_ui.bool_from_string(s, accepted_values=accepted_values)
374
self.assertIs(None, res)
376
def test_know_valid_values(self):
377
self.assertIsTrue('true')
378
self.assertIsFalse('false')
379
self.assertIsTrue('1')
380
self.assertIsFalse('0')
381
self.assertIsTrue('on')
382
self.assertIsFalse('off')
383
self.assertIsTrue('yes')
384
self.assertIsFalse('no')
385
self.assertIsTrue('y')
386
self.assertIsFalse('n')
387
# Also try some case variations
388
self.assertIsTrue('True')
389
self.assertIsFalse('False')
390
self.assertIsTrue('On')
391
self.assertIsFalse('Off')
392
self.assertIsTrue('ON')
393
self.assertIsFalse('OFF')
394
self.assertIsTrue('oN')
395
self.assertIsFalse('oFf')
397
def test_invalid_values(self):
398
self.assertIsNone(None)
399
self.assertIsNone('doubt')
400
self.assertIsNone('frue')
401
self.assertIsNone('talse')
402
self.assertIsNone('42')
404
def test_provided_values(self):
405
av = dict(y=True, n=False, yes=True, no=False)
406
self.assertIsTrue('y', av)
407
self.assertIsTrue('Y', av)
408
self.assertIsTrue('Yes', av)
409
self.assertIsFalse('n', av)
410
self.assertIsFalse('N', av)
411
self.assertIsFalse('No', av)
412
self.assertIsNone('1', av)
413
self.assertIsNone('0', av)
414
self.assertIsNone('on', av)
415
self.assertIsNone('off', av)