73
71
self.assertEqual('', ui.stdout.readline())
74
72
# stdin should be empty
75
73
self.assertEqual('', ui.stdin.readline())
77
def test_text_factory_unicode_password(self):
78
"""Test a unicode password."""
79
ui = ui_testing.TextUIFactory(u'baz\u1234')
80
password = ui.get_password(
81
u'Hello \u1234 %(user)s', user=u'some\u1234')
82
self.assertEqual(u'baz\u1234', password)
83
self.assertEqual(u'Hello \u1234 some\u1234: ', ui.stderr.getvalue())
84
# stdin and stdout should be empty
85
self.assertEqual('', ui.stdin.readline())
86
self.assertEqual('', ui.stdout.getvalue())
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_note(self):
104
stderr = tests.StringIOWrapper()
105
stdout = tests.StringIOWrapper()
106
ui_factory = _mod_ui_text.TextUIFactory(stdin=tests.StringIOWrapper(''),
109
pb = ui_factory.nested_progress_bar()
111
result = self.applyDeprecated(deprecated_in((2, 1, 0)),
114
self.assertEqual(None, result)
115
self.assertEqual("t\n", stdout.getvalue())
116
# Since there was no update() call, there should be no clear() call
117
self.failIf(re.search(r'^\r {10,}\r$',
118
stderr.getvalue()) is not None,
119
'We cleared the stderr without anything to put there')
123
def test_progress_note_clears(self):
124
stderr = test_progress._TTYStringIO()
125
stdout = test_progress._TTYStringIO()
126
# so that we get a TextProgressBar
127
os.environ['TERM'] = 'xterm'
128
ui_factory = _mod_ui_text.TextUIFactory(
129
stdin=tests.StringIOWrapper(''),
130
stdout=stdout, stderr=stderr)
131
self.assertIsInstance(ui_factory._progress_view,
132
_mod_ui_text.TextProgressView)
133
pb = ui_factory.nested_progress_bar()
135
# Create a progress update that isn't throttled
137
result = self.applyDeprecated(deprecated_in((2, 1, 0)),
139
self.assertEqual(None, result)
140
self.assertEqual("t\n", stdout.getvalue())
141
# the exact contents will depend on the terminal width and we don't
142
# care about that right now - but you're probably running it on at
143
# least a 10-character wide terminal :)
144
self.assertContainsRe(stderr.getvalue(), r'\r {10,}\r$')
88
148
def test_text_ui_get_boolean(self):
94
"yes with garbage\nY\n" # True
95
"not an answer\nno\n" # False
96
"I'm sure!\nyes\n" # True
99
with ui_testing.TextUIFactory(stdin_text) as factory:
100
self.assertEqual(True, factory.get_boolean(u""))
101
self.assertEqual(False, factory.get_boolean(u""))
102
self.assertEqual(True, factory.get_boolean(u""))
103
self.assertEqual(False, factory.get_boolean(u""))
104
self.assertEqual(True, factory.get_boolean(u""))
105
self.assertEqual(False, factory.get_boolean(u""))
106
self.assertEqual(True, factory.get_boolean(u""))
107
self.assertEqual(False, factory.get_boolean(u""))
108
self.assertEqual("foo\n", factory.stdin.read())
109
# stdin should be empty
110
self.assertEqual('', factory.stdin.readline())
111
# return false on EOF
112
self.assertEqual(False, factory.get_boolean(u""))
114
def test_text_ui_choose_bad_parameters(self):
115
with ui_testing.TextUIFactory(u"") as factory:
116
# invalid default index
117
self.assertRaises(ValueError, factory.choose, u"", u"&Yes\n&No", 3)
120
ValueError, factory.choose, u"", u"&choice\n&ChOiCe")
121
# duplicated shortcut
123
ValueError, factory.choose, u"", u"&choice1\nchoi&ce2")
125
def test_text_ui_choose_prompt_explicit(self):
126
# choices with explicit shortcuts
127
with ui_testing.TextUIFactory(u"") as factory:
128
factory.choose(u"prompt", u"&yes\n&No\nmore &info")
130
"prompt ([y]es, [N]o, more [i]nfo): \n",
131
factory.stderr.getvalue())
133
def test_text_ui_choose_prompt_automatic(self):
134
# automatic shortcuts
135
with ui_testing.TextUIFactory(u"") as factory:
136
factory.choose(u"prompt", u"yes\nNo\nmore info")
138
"prompt ([y]es, [N]o, [m]ore info): \n",
139
factory.stderr.getvalue())
141
def test_text_ui_choose_return_values(self):
143
return factory.choose(u"", u"&Yes\n&No\nMaybe\nmore &info", 3)
149
"b\na\nd \n" # bad shortcuts, all ignored
150
"yes with garbage\nY\n" # 0
151
"not an answer\nno\n" # 1
152
"info\nmore info\n" # 3
155
with ui_testing.TextUIFactory(stdin_text) as factory:
156
self.assertEqual(0, choose())
157
self.assertEqual(1, choose())
158
self.assertEqual(3, choose())
159
self.assertEqual(1, choose())
160
self.assertEqual(0, choose())
161
self.assertEqual(1, choose())
162
self.assertEqual(3, choose())
163
self.assertEqual(2, choose())
164
self.assertEqual("foo\n", factory.stdin.read())
165
# stdin should be empty
166
self.assertEqual('', factory.stdin.readline())
168
self.assertEqual(None, choose())
170
def test_text_ui_choose_no_default(self):
172
" \n" # no default, invalid!
175
with ui_testing.TextUIFactory(stdin_text) as factory:
176
self.assertEqual(0, factory.choose(u"", u"&Yes\n&No"))
177
self.assertEqual("foo\n", factory.stdin.read())
149
stdin = tests.StringIOWrapper("y\n" # True
151
"yes with garbage\nY\n" # True
152
"not an answer\nno\n" # False
153
"I'm sure!\nyes\n" # True
156
stdout = tests.StringIOWrapper()
157
stderr = tests.StringIOWrapper()
158
factory = _mod_ui_text.TextUIFactory(stdin, stdout, stderr)
159
self.assertEqual(True, factory.get_boolean(""))
160
self.assertEqual(False, factory.get_boolean(""))
161
self.assertEqual(True, factory.get_boolean(""))
162
self.assertEqual(False, factory.get_boolean(""))
163
self.assertEqual(True, factory.get_boolean(""))
164
self.assertEqual(False, factory.get_boolean(""))
165
self.assertEqual("foo\n", factory.stdin.read())
166
# stdin should be empty
167
self.assertEqual('', factory.stdin.readline())
179
169
def test_text_ui_get_integer(self):
170
stdin = tests.StringIOWrapper(
183
173
"hmmm\nwhat else ?\nCome on\nok 42\n4.24\n42\n")
184
with ui_testing.TextUIFactory(stdin_text) as factory:
185
self.assertEqual(1, factory.get_integer(u""))
186
self.assertEqual(-2, factory.get_integer(u""))
187
self.assertEqual(42, factory.get_integer(u""))
174
stdout = tests.StringIOWrapper()
175
stderr = tests.StringIOWrapper()
176
factory = _mod_ui_text.TextUIFactory(stdin, stdout, stderr)
177
self.assertEqual(1, factory.get_integer(""))
178
self.assertEqual(-2, factory.get_integer(""))
179
self.assertEqual(42, factory.get_integer(""))
189
181
def test_text_factory_prompt(self):
190
182
# see <https://launchpad.net/bugs/365891>
191
with ui_testing.TextUIFactory() as factory:
192
factory.prompt(u'foo %2e')
193
self.assertEqual('', factory.stdout.getvalue())
194
self.assertEqual('foo %2e', factory.stderr.getvalue())
183
StringIO = tests.StringIOWrapper
184
factory = _mod_ui_text.TextUIFactory(StringIO(), StringIO(), StringIO())
185
factory.prompt('foo %2e')
186
self.assertEqual('', factory.stdout.getvalue())
187
self.assertEqual('foo %2e', factory.stderr.getvalue())
196
189
def test_text_factory_prompts_and_clears(self):
197
190
# a get_boolean call should clear the pb before prompting
198
out = ui_testing.StringIOAsTTY()
199
self.overrideEnv('TERM', 'xterm')
200
factory = ui_testing.TextUIFactory("yada\ny\n", stdout=out, stderr=out)
202
pb = factory.nested_progress_bar()
203
pb._avail_width = lambda: 79
205
pb.show_spinner = False
206
pb.show_count = False
207
pb.update("foo", 0, 1)
210
self.apply_redirected(
211
None, factory.stdout, factory.stdout, factory.get_boolean,
212
u"what do you want"))
213
output = out.getvalue()
214
self.assertContainsRe(output,
216
self.assertContainsString(
218
r"what do you want? ([y]es, [n]o): what do you want? "
220
# stdin should have been totally consumed
221
self.assertEqual('', factory.stdin.readline())
191
out = test_progress._TTYStringIO()
192
os.environ['TERM'] = 'xterm'
193
factory = _mod_ui_text.TextUIFactory(
194
stdin=tests.StringIOWrapper("yada\ny\n"),
195
stdout=out, stderr=out)
196
pb = factory.nested_progress_bar()
198
pb.show_spinner = False
199
pb.show_count = False
200
pb.update("foo", 0, 1)
201
self.assertEqual(True,
202
self.apply_redirected(None, factory.stdout,
206
output = out.getvalue()
207
self.assertContainsRe(factory.stdout.getvalue(),
209
self.assertContainsRe(factory.stdout.getvalue(),
210
r"what do you want\? \[y/n\]: what do you want\? \[y/n\]: ")
211
# stdin should have been totally consumed
212
self.assertEqual('', factory.stdin.readline())
223
214
def test_text_tick_after_update(self):
224
ui_factory = ui_testing.TextUIFactory()
225
with ui_factory.nested_progress_bar() as pb:
215
ui_factory = _mod_ui_text.TextUIFactory(stdout=tests.StringIOWrapper(),
216
stderr=tests.StringIOWrapper())
217
pb = ui_factory.nested_progress_bar()
226
219
pb.update('task', 0, 3)
227
220
# Reset the clock, so that it actually tries to repaint itself
228
221
ui_factory._progress_view._last_repaint = time.time() - 1.0
231
226
def test_text_ui_getusername(self):
232
ui = ui_testing.TextUIFactory('someuser\n\n')
233
self.assertEqual('someuser',
234
ui.get_username(u'Hello %(host)s', host='some'))
235
self.assertEqual('Hello some: ', ui.stderr.getvalue())
236
self.assertEqual('', ui.stdout.getvalue())
237
self.assertEqual('', ui.get_username(u"Gebruiker"))
227
factory = _mod_ui_text.TextUIFactory(None, None, None)
228
factory.stdin = tests.StringIOWrapper("someuser\n\n")
229
factory.stdout = tests.StringIOWrapper()
230
factory.stderr = tests.StringIOWrapper()
231
factory.stdout.encoding = "utf8"
232
# there is no output from the base factory
233
self.assertEqual("someuser",
234
factory.get_username('Hello %(host)s', host='some'))
235
self.assertEquals("Hello some: ", factory.stderr.getvalue())
236
self.assertEquals('', factory.stdout.getvalue())
237
self.assertEqual("", factory.get_username("Gebruiker"))
238
238
# stdin should be empty
239
self.assertEqual('', ui.stdin.readline())
239
self.assertEqual('', factory.stdin.readline())
241
def test_text_ui_getusername_unicode(self):
242
ui = ui_testing.TextUIFactory(u'someuser\u1234')
243
username = ui.get_username(u'Hello %(host)s', host=u'some\u1234')
244
self.assertEqual(u"someuser\u1234", username)
245
self.assertEqual(u"Hello some\u1234: ", ui.stderr.getvalue())
246
self.assertEqual('', ui.stdout.getvalue())
241
def test_text_ui_getusername_utf8(self):
242
ui = tests.TestUIFactory(stdin=u'someuser\u1234'.encode('utf8'),
243
stdout=tests.StringIOWrapper(),
244
stderr=tests.StringIOWrapper())
245
ui.stderr.encoding = ui.stdout.encoding = ui.stdin.encoding = "utf8"
246
pb = ui.nested_progress_bar()
248
# there is no output from the base factory
249
username = self.apply_redirected(ui.stdin, ui.stdout, ui.stderr,
250
ui.get_username, u'Hello\u1234 %(host)s', host=u'some\u1234')
251
self.assertEquals(u"someuser\u1234", username.decode('utf8'))
252
self.assertEquals(u"Hello\u1234 some\u1234: ",
253
ui.stderr.getvalue().decode("utf8"))
254
self.assertEquals('', ui.stdout.getvalue())
248
258
def test_quietness(self):
249
self.overrideEnv('BRZ_PROGRESS_BAR', 'text')
250
ui_factory = ui_testing.TextUIFactory(
251
stderr=ui_testing.StringIOAsTTY())
253
self.assertIsInstance(ui_factory._progress_view,
254
_mod_ui_text.TextProgressView)
255
ui_factory.be_quiet(True)
256
self.assertIsInstance(ui_factory._progress_view,
257
_mod_ui_text.NullProgressView)
259
os.environ['BZR_PROGRESS_BAR'] = 'text'
260
ui_factory = _mod_ui_text.TextUIFactory(None,
261
test_progress._TTYStringIO(),
262
test_progress._TTYStringIO())
263
self.assertIsInstance(ui_factory._progress_view,
264
_mod_ui_text.TextProgressView)
265
ui_factory.be_quiet(True)
266
self.assertIsInstance(ui_factory._progress_view,
267
_mod_ui_text.NullProgressView)
259
269
def test_text_ui_show_user_warning(self):
260
from ..bzr.groupcompress_repo import RepositoryFormat2a
261
from ..bzr.knitpack_repo import RepositoryFormatKnitPack5
262
ui = ui_testing.TextUIFactory()
270
from bzrlib.repofmt.groupcompress_repo import RepositoryFormat2a
271
from bzrlib.repofmt.pack_repo import RepositoryFormatKnitPack5
274
ui = tests.TextUIFactory(stdin=None, stdout=out, stderr=err)
263
275
remote_fmt = remote.RemoteRepositoryFormat()
264
276
remote_fmt._network_name = RepositoryFormatKnitPack5().network_name()
265
ui.show_user_warning(
266
'cross_format_fetch', from_format=RepositoryFormat2a(),
277
ui.show_user_warning('cross_format_fetch', from_format=RepositoryFormat2a(),
267
278
to_format=remote_fmt)
268
self.assertEqual('', ui.stdout.getvalue())
269
self.assertContainsRe(
270
ui.stderr.getvalue(),
271
"^Doing on-the-fly conversion from RepositoryFormat2a\\(\\) to "
272
"RemoteRepositoryFormat\\(_network_name="
273
"b?'Bazaar RepositoryFormatKnitPack5 \\(bzr 1.6\\)\\\\n'\\)\\.\n"
274
"This may take some time. Upgrade the repositories to "
275
"the same format for better performance\\.\n$")
279
self.assertEquals('', out.getvalue())
280
self.assertEquals("Doing on-the-fly conversion from RepositoryFormat2a() to "
281
"RemoteRepositoryFormat(_network_name='Bazaar RepositoryFormatKnitPack5 "
282
"(bzr 1.6)\\n').\nThis may take some time. Upgrade the repositories to "
283
"the same format for better performance.\n",
276
285
# and now with it suppressed please
277
ui = ui_testing.TextUIFactory()
288
ui = tests.TextUIFactory(stdin=None, stdout=out, stderr=err)
278
289
ui.suppressed_warnings.add('cross_format_fetch')
279
ui.show_user_warning(
280
'cross_format_fetch', from_format=RepositoryFormat2a(),
290
ui.show_user_warning('cross_format_fetch', from_format=RepositoryFormat2a(),
281
291
to_format=remote_fmt)
282
self.assertEqual('', ui.stdout.getvalue())
283
self.assertEqual('', ui.stderr.getvalue())
292
self.assertEquals('', out.getvalue())
293
self.assertEquals('', err.getvalue())
286
296
class TestTextUIOutputStream(tests.TestCase):
287
297
"""Tests for output stream that synchronizes with progress bar."""
289
299
def test_output_clears_terminal(self):
300
stdout = tests.StringIOWrapper()
301
stderr = tests.StringIOWrapper()
292
uif = ui_testing.TextUIFactory()
304
uif = _mod_ui_text.TextUIFactory(None, stdout, stderr)
293
305
uif.clear_term = lambda: clear_calls.append('clear')
295
stream = _mod_ui_text.TextUIOutputStream(
296
uif, uif.stdout, 'utf-8', 'strict')
297
stream.write(u"Hello world!\n")
298
stream.write(u"there's more...\n")
299
stream.writelines([u"1\n", u"2\n", u"3\n"])
307
stream = _mod_ui_text.TextUIOutputStream(uif, uif.stdout)
308
stream.write("Hello world!\n")
309
stream.write("there's more...\n")
310
stream.writelines(["1\n", "2\n", "3\n"])
301
self.assertEqual(uif.stdout.getvalue(),
312
self.assertEqual(stdout.getvalue(),
305
316
self.assertEqual(['clear', 'clear', 'clear'],
313
324
def test_progress_construction(self):
314
325
"""TextUIFactory constructs the right progress view.
316
FileStringIO = ui_testing.StringIOWithEncoding
317
TTYStringIO = ui_testing.StringIOAsTTY
327
TTYStringIO = test_progress._TTYStringIO
328
FileStringIO = tests.StringIOWrapper
318
329
for (file_class, term, pb, expected_pb_class) in (
319
# on an xterm, either use them or not as the user requests,
320
# otherwise default on
321
(TTYStringIO, 'xterm', 'none', _mod_ui_text.NullProgressView),
322
(TTYStringIO, 'xterm', 'text', _mod_ui_text.TextProgressView),
323
(TTYStringIO, 'xterm', None, _mod_ui_text.TextProgressView),
324
# on a dumb terminal, again if there's explicit configuration
325
# do it, otherwise default off
326
(TTYStringIO, 'dumb', 'none', _mod_ui_text.NullProgressView),
327
(TTYStringIO, 'dumb', 'text', _mod_ui_text.TextProgressView),
328
(TTYStringIO, 'dumb', None, _mod_ui_text.NullProgressView),
329
# on a non-tty terminal, it's null regardless of $TERM
330
(FileStringIO, 'xterm', None, _mod_ui_text.NullProgressView),
331
(FileStringIO, 'dumb', None, _mod_ui_text.NullProgressView),
332
# however, it can still be forced on
333
(FileStringIO, 'dumb', 'text', _mod_ui_text.TextProgressView),
335
self.overrideEnv('TERM', term)
336
self.overrideEnv('BRZ_PROGRESS_BAR', pb)
337
stdin = file_class(u'')
330
# on an xterm, either use them or not as the user requests,
331
# otherwise default on
332
(TTYStringIO, 'xterm', 'none', _mod_ui_text.NullProgressView),
333
(TTYStringIO, 'xterm', 'text', _mod_ui_text.TextProgressView),
334
(TTYStringIO, 'xterm', None, _mod_ui_text.TextProgressView),
335
# on a dumb terminal, again if there's explicit configuration do
336
# it, otherwise default off
337
(TTYStringIO, 'dumb', 'none', _mod_ui_text.NullProgressView),
338
(TTYStringIO, 'dumb', 'text', _mod_ui_text.TextProgressView),
339
(TTYStringIO, 'dumb', None, _mod_ui_text.NullProgressView),
340
# on a non-tty terminal, it's null regardless of $TERM
341
(FileStringIO, 'xterm', None, _mod_ui_text.NullProgressView),
342
(FileStringIO, 'dumb', None, _mod_ui_text.NullProgressView),
343
# however, it can still be forced on
344
(FileStringIO, 'dumb', 'text', _mod_ui_text.TextProgressView),
346
os.environ['TERM'] = term
348
if 'BZR_PROGRESS_BAR' in os.environ:
349
del os.environ['BZR_PROGRESS_BAR']
351
os.environ['BZR_PROGRESS_BAR'] = pb
352
stdin = file_class('')
338
353
stderr = file_class()
339
354
stdout = file_class()
340
355
uif = _mod_ui.make_ui_for_terminal(stdin, stdout, stderr)
341
self.assertIsInstance(
342
uif, _mod_ui_text.TextUIFactory,
343
"TERM=%s BRZ_PROGRESS_BAR=%s uif=%r" % (term, pb, uif,))
344
self.assertIsInstance(
345
uif.make_progress_view(),
356
self.assertIsInstance(uif, _mod_ui_text.TextUIFactory,
357
"TERM=%s BZR_PROGRESS_BAR=%s uif=%r" % (term, pb, uif,))
358
self.assertIsInstance(uif.make_progress_view(),
346
359
expected_pb_class,
347
"TERM=%s BRZ_PROGRESS_BAR=%s uif=%r" % (term, pb, uif,))
360
"TERM=%s BZR_PROGRESS_BAR=%s uif=%r" % (term, pb, uif,))
349
362
def test_text_ui_non_terminal(self):
350
363
"""Even on non-ttys, make_ui_for_terminal gives a text ui."""
351
stdin = stderr = stdout = ui_testing.StringIOWithEncoding()
364
stdin = test_progress._NonTTYStringIO('')
365
stderr = test_progress._NonTTYStringIO()
366
stdout = test_progress._NonTTYStringIO()
352
367
for term_type in ['dumb', None, 'xterm']:
353
self.overrideEnv('TERM', term_type)
368
if term_type is None:
369
del os.environ['TERM']
371
os.environ['TERM'] = term_type
354
372
uif = _mod_ui.make_ui_for_terminal(stdin, stdout, stderr)
355
373
self.assertIsInstance(uif, _mod_ui_text.TextUIFactory,
356
'TERM=%r' % (term_type,))
374
'TERM=%r' % (term_type,))
359
377
class SilentUITests(tests.TestCase):