~bzr-pqm/bzr/bzr.dev

5557.1.7 by John Arbash Meinel
Merge in the bzr.dev 5582
1
# Copyright (C) 2005-2011 Canonical Ltd
1185.49.22 by John Arbash Meinel
Added get_password to the UIFactory, using it inside of sftp.py
2
#
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.
7
#
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.
12
#
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
4183.7.1 by Sabin Iacob
update FSF mailing address
15
# Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
1185.49.22 by John Arbash Meinel
Added get_password to the UIFactory, using it inside of sftp.py
16
17
"""Tests for the bzrlib ui
18
"""
19
4017.1.1 by John Arbash Meinel
Get a pb.tick() to work after calling pb.update()
20
import time
1185.49.22 by John Arbash Meinel
Added get_password to the UIFactory, using it inside of sftp.py
21
4797.40.2 by Martin Pool
Add missing import
22
from StringIO import StringIO
23
5416.1.11 by Martin Pool
Add ConfirmationUserInterfacePolicy that lets specific confirmations be configured off.
24
from testtools.matchers import *
25
4488.1.1 by Vincent Ladeuil
(vila) Cleanup imports in some test files
26
from bzrlib import (
5230.1.3 by Martin Pool
Use configured output encoding for make_output_stream
27
    config,
4634.144.5 by Martin Pool
Cleaner presentation and tests for warn_cross_format_fetch
28
    remote,
4488.1.1 by Vincent Ladeuil
(vila) Cleanup imports in some test files
29
    tests,
30
    ui as _mod_ui,
4449.3.4 by Martin Pool
ProgressTask now talks to ProgressView; easier to test
31
    )
5230.1.3 by Martin Pool
Use configured output encoding for make_output_stream
32
from bzrlib.tests import (
33
    fixtures,
34
    test_progress,
35
    )
4597.3.36 by Vincent Ladeuil
Fix imports and various cleanups in test_ui.
36
from bzrlib.ui import text as _mod_ui_text
5422.1.5 by Martin Pool
Move ProgressRecordingUIFactory to bzrlib.tests.testui
37
from bzrlib.tests.testui import (
5422.1.4 by Martin Pool
Rename CapturingUIFactory to ProgressRecordingUIFactory
38
    ProgressRecordingUIFactory,
5422.1.1 by Martin Pool
Move CapturingUIFactory out of per_workingtree tests into somewhere reusable
39
    )
1185.49.22 by John Arbash Meinel
Added get_password to the UIFactory, using it inside of sftp.py
40
1681.1.2 by Robert Collins
* bzrlib.ui.text.TextUIFactory now accepts a bar_type parameter which
41
5230.1.3 by Martin Pool
Use configured output encoding for make_output_stream
42
class TestUIConfiguration(tests.TestCaseWithTransport):
43
44
    def test_output_encoding_configuration(self):
5230.1.5 by Martin Pool
Merge updated test fixtures
45
        enc = fixtures.generate_unicode_encodings().next()
5230.1.3 by Martin Pool
Use configured output encoding for make_output_stream
46
        config.GlobalConfig().set_user_option('output_encoding',
47
            enc)
48
        ui = tests.TestUIFactory(stdin=None,
5230.1.5 by Martin Pool
Merge updated test fixtures
49
            stdout=tests.StringIOWrapper(),
50
            stderr=tests.StringIOWrapper())
5416.1.11 by Martin Pool
Add ConfirmationUserInterfacePolicy that lets specific confirmations be configured off.
51
        output = ui.make_output_stream()
52
        self.assertEquals(output.encoding, enc)
5230.1.3 by Martin Pool
Use configured output encoding for make_output_stream
53
54
4711.1.6 by Martin Pool
Separate TextUIFactory tests
55
class TestTextUIFactory(tests.TestCase):
2294.4.1 by Vincent Ladeuil
Add a UIFactory.get_login method, fix tests.
56
5416.1.1 by Martin Pool
Use a structured ui_factory.confirm_action rather than just get_boolean
57
    def make_test_ui_factory(self, stdin_contents):
58
        ui = tests.TestUIFactory(stdin=stdin_contents,
59
                                 stdout=tests.StringIOWrapper(),
60
                                 stderr=tests.StringIOWrapper())
61
        return ui
62
63
    def test_text_factory_confirm(self):
64
        # turns into reading a regular boolean
65
        ui = self.make_test_ui_factory('n\n')
5863.6.1 by Jelmer Vernooij
Require a unicode prompt to be passed into all methods that prompt.
66
        self.assertEquals(ui.confirm_action(u'Should %(thing)s pass?',
5416.1.1 by Martin Pool
Use a structured ui_factory.confirm_action rather than just get_boolean
67
            'bzrlib.tests.test_ui.confirmation',
68
            {'thing': 'this'},),
69
            False)
70
2294.4.1 by Vincent Ladeuil
Add a UIFactory.get_login method, fix tests.
71
    def test_text_factory_ascii_password(self):
5416.1.1 by Martin Pool
Use a structured ui_factory.confirm_action rather than just get_boolean
72
        ui = self.make_test_ui_factory('secret\n')
2294.4.1 by Vincent Ladeuil
Add a UIFactory.get_login method, fix tests.
73
        pb = ui.nested_progress_bar()
74
        try:
75
            self.assertEqual('secret',
76
                             self.apply_redirected(ui.stdin, ui.stdout,
4368.3.1 by Vincent Ladeuil
Use stderr for UI prompt to address bug #376582.
77
                                                   ui.stderr,
2294.4.1 by Vincent Ladeuil
Add a UIFactory.get_login method, fix tests.
78
                                                   ui.get_password))
79
            # ': ' is appended to prompt
4368.3.1 by Vincent Ladeuil
Use stderr for UI prompt to address bug #376582.
80
            self.assertEqual(': ', ui.stderr.getvalue())
81
            self.assertEqual('', ui.stdout.readline())
2363.4.3 by Vincent Ladeuil
Tidy-up tests.
82
            # stdin should be empty
2363.4.6 by Vincent Ladeuil
Fix tests around stdin emptyness.
83
            self.assertEqual('', ui.stdin.readline())
2294.4.1 by Vincent Ladeuil
Add a UIFactory.get_login method, fix tests.
84
        finally:
85
            pb.finished()
86
87
    def test_text_factory_utf8_password(self):
88
        """Test an utf8 password.
89
90
        We can't predict what encoding users will have for stdin, so we force
91
        it to utf8 to test that we transport the password correctly.
92
        """
5416.1.1 by Martin Pool
Use a structured ui_factory.confirm_action rather than just get_boolean
93
        ui = self.make_test_ui_factory(u'baz\u1234'.encode('utf8'))
4368.3.1 by Vincent Ladeuil
Use stderr for UI prompt to address bug #376582.
94
        ui.stderr.encoding = ui.stdout.encoding = ui.stdin.encoding = 'utf8'
2294.4.1 by Vincent Ladeuil
Add a UIFactory.get_login method, fix tests.
95
        pb = ui.nested_progress_bar()
96
        try:
4368.3.1 by Vincent Ladeuil
Use stderr for UI prompt to address bug #376582.
97
            password = self.apply_redirected(ui.stdin, ui.stdout, ui.stderr,
2294.4.1 by Vincent Ladeuil
Add a UIFactory.get_login method, fix tests.
98
                                             ui.get_password,
99
                                             u'Hello \u1234 %(user)s',
100
                                             user=u'some\u1234')
101
            # We use StringIO objects, we need to decode them
102
            self.assertEqual(u'baz\u1234', password.decode('utf8'))
103
            self.assertEqual(u'Hello \u1234 some\u1234: ',
4368.3.1 by Vincent Ladeuil
Use stderr for UI prompt to address bug #376582.
104
                             ui.stderr.getvalue().decode('utf8'))
105
            # stdin and stdout should be empty
2363.4.6 by Vincent Ladeuil
Fix tests around stdin emptyness.
106
            self.assertEqual('', ui.stdin.readline())
4368.3.1 by Vincent Ladeuil
Use stderr for UI prompt to address bug #376582.
107
            self.assertEqual('', ui.stdout.readline())
2294.4.1 by Vincent Ladeuil
Add a UIFactory.get_login method, fix tests.
108
        finally:
109
            pb.finished()
1534.5.6 by Robert Collins
split out converter logic into per-format objects.
110
4449.3.38 by Martin Pool
Cleanup get_boolean tests
111
    def test_text_ui_get_boolean(self):
4773.1.2 by Vincent Ladeuil
Cleanup imports in test_ui
112
        stdin = tests.StringIOWrapper("y\n" # True
113
                                      "n\n" # False
6182.2.4 by Benoît Pierre
Fix ui tests.
114
                                      " \n y \n" # True
115
                                      " no \n" # False
4773.1.2 by Vincent Ladeuil
Cleanup imports in test_ui
116
                                      "yes with garbage\nY\n" # True
117
                                      "not an answer\nno\n" # False
118
                                      "I'm sure!\nyes\n" # True
119
                                      "NO\n" # False
120
                                      "foo\n")
4597.3.36 by Vincent Ladeuil
Fix imports and various cleanups in test_ui.
121
        stdout = tests.StringIOWrapper()
122
        stderr = tests.StringIOWrapper()
123
        factory = _mod_ui_text.TextUIFactory(stdin, stdout, stderr)
5863.6.1 by Jelmer Vernooij
Require a unicode prompt to be passed into all methods that prompt.
124
        self.assertEqual(True, factory.get_boolean(u""))
125
        self.assertEqual(False, factory.get_boolean(u""))
126
        self.assertEqual(True, factory.get_boolean(u""))
127
        self.assertEqual(False, factory.get_boolean(u""))
128
        self.assertEqual(True, factory.get_boolean(u""))
129
        self.assertEqual(False, factory.get_boolean(u""))
6182.2.4 by Benoît Pierre
Fix ui tests.
130
        self.assertEqual(True, factory.get_boolean(u""))
131
        self.assertEqual(False, factory.get_boolean(u""))
1687.1.4 by Robert Collins
Add bzrlib.ui.ui_factory.get_boolean().
132
        self.assertEqual("foo\n", factory.stdin.read())
2363.4.4 by Vincent Ladeuil
More tidying-up.
133
        # stdin should be empty
2363.4.6 by Vincent Ladeuil
Fix tests around stdin emptyness.
134
        self.assertEqual('', factory.stdin.readline())
6182.2.4 by Benoît Pierre
Fix ui tests.
135
        # return false on EOF
136
        self.assertEqual(False, factory.get_boolean(u""))
1687.1.4 by Robert Collins
Add bzrlib.ui.ui_factory.get_boolean().
137
6182.2.20 by Benoît Pierre
Add some more tests for TextUIFactory.choose.
138
    def test_text_ui_choose_bad_parameters(self):
139
        stdin = tests.StringIOWrapper()
140
        stdout = tests.StringIOWrapper()
141
        stderr = tests.StringIOWrapper()
142
        factory = _mod_ui_text.TextUIFactory(stdin, stdout, stderr)
143
        # invalid default index
144
        self.assertRaises(ValueError, factory.choose, u"", u"&Yes\n&No", 3)
145
        # duplicated choice
146
        self.assertRaises(ValueError, factory.choose, u"", u"&choice\n&ChOiCe")
147
        # duplicated shortcut
148
        self.assertRaises(ValueError, factory.choose, u"", u"&choice1\nchoi&ce2")
149
6182.2.22 by Benoît Pierre
More TextUIFactory.choose tests: check prompts.
150
    def test_text_ui_choose_prompt(self):
151
        stdin = tests.StringIOWrapper()
152
        stdout = tests.StringIOWrapper()
153
        stderr = tests.StringIOWrapper()
154
        factory = _mod_ui_text.TextUIFactory(stdin, stdout, stderr)
155
        # choices with explicit shortcuts
156
        factory.choose(u"prompt", u"&yes\n&No\nmore &info")
157
        self.assertEqual("prompt ([y]es, [N]o, more [i]nfo): \n", factory.stderr.getvalue())
158
        # automatic shortcuts
159
        factory.stderr.truncate(0)
160
        factory.choose(u"prompt", u"yes\nNo\nmore info")
161
        self.assertEqual("prompt ([y]es, [N]o, [m]ore info): \n", factory.stderr.getvalue())
162
6182.2.17 by Benoît Pierre
Add some tests for TextUIFactory.choose.
163
    def test_text_ui_choose_return_values(self):
164
        choose = lambda: factory.choose(u"", u"&Yes\n&No\nMaybe\nmore &info", 3)
165
        stdin = tests.StringIOWrapper("y\n" # 0
166
                                      "n\n" # 1
167
                                      " \n" # default: 3
168
                                      " no \n" # 1
6182.2.25 by Benoît Pierre
Tweak test_text_ui_choose_return_values a little.
169
                                      "b\na\nd \n" # bad shortcuts, all ignored
6182.2.17 by Benoît Pierre
Add some tests for TextUIFactory.choose.
170
                                      "yes with garbage\nY\n" # 0
171
                                      "not an answer\nno\n" # 1
172
                                      "info\nmore info\n" # 3
173
                                      "Maybe\n" # 2
174
                                      "foo\n")
175
        stdout = tests.StringIOWrapper()
176
        stderr = tests.StringIOWrapper()
177
        factory = _mod_ui_text.TextUIFactory(stdin, stdout, stderr)
178
        self.assertEqual(0, choose())
179
        self.assertEqual(1, choose())
180
        self.assertEqual(3, choose())
181
        self.assertEqual(1, choose())
182
        self.assertEqual(0, choose())
183
        self.assertEqual(1, choose())
184
        self.assertEqual(3, choose())
185
        self.assertEqual(2, choose())
186
        self.assertEqual("foo\n", factory.stdin.read())
187
        # stdin should be empty
188
        self.assertEqual('', factory.stdin.readline())
189
        # return None on EOF
190
        self.assertEqual(None, choose())
191
6182.2.19 by Benoît Pierre
Add more tests for TextUIFactory.choose.
192
    def test_text_ui_choose_no_default(self):
193
        stdin = tests.StringIOWrapper(" \n" # no default, invalid!
194
                                      " yes \n" # 0
195
                                      "foo\n")
196
        stdout = tests.StringIOWrapper()
197
        stderr = tests.StringIOWrapper()
198
        factory = _mod_ui_text.TextUIFactory(stdin, stdout, stderr)
199
        self.assertEqual(0, factory.choose(u"", u"&Yes\n&No"))
200
        self.assertEqual("foo\n", factory.stdin.read())
201
4597.3.37 by Vincent Ladeuil
Allows ui factories to query users for an integer.
202
    def test_text_ui_get_integer(self):
203
        stdin = tests.StringIOWrapper(
204
            "1\n"
205
            "  -2  \n"
206
            "hmmm\nwhat else ?\nCome on\nok 42\n4.24\n42\n")
207
        stdout = tests.StringIOWrapper()
208
        stderr = tests.StringIOWrapper()
209
        factory = _mod_ui_text.TextUIFactory(stdin, stdout, stderr)
5863.6.1 by Jelmer Vernooij
Require a unicode prompt to be passed into all methods that prompt.
210
        self.assertEqual(1, factory.get_integer(u""))
211
        self.assertEqual(-2, factory.get_integer(u""))
212
        self.assertEqual(42, factory.get_integer(u""))
4597.3.37 by Vincent Ladeuil
Allows ui factories to query users for an integer.
213
4300.3.1 by Martin Pool
Fix string expansion in TextUIFactory.prompt
214
    def test_text_factory_prompt(self):
215
        # see <https://launchpad.net/bugs/365891>
4597.3.36 by Vincent Ladeuil
Fix imports and various cleanups in test_ui.
216
        StringIO = tests.StringIOWrapper
217
        factory = _mod_ui_text.TextUIFactory(StringIO(), StringIO(), StringIO())
5863.6.1 by Jelmer Vernooij
Require a unicode prompt to be passed into all methods that prompt.
218
        factory.prompt(u'foo %2e')
4368.3.1 by Vincent Ladeuil
Use stderr for UI prompt to address bug #376582.
219
        self.assertEqual('', factory.stdout.getvalue())
220
        self.assertEqual('foo %2e', factory.stderr.getvalue())
4300.3.1 by Martin Pool
Fix string expansion in TextUIFactory.prompt
221
1687.1.4 by Robert Collins
Add bzrlib.ui.ui_factory.get_boolean().
222
    def test_text_factory_prompts_and_clears(self):
223
        # a get_boolean call should clear the pb before prompting
4597.3.36 by Vincent Ladeuil
Fix imports and various cleanups in test_ui.
224
        out = test_progress._TTYStringIO()
5570.3.9 by Vincent Ladeuil
More use cases for overrideEnv, _cleanEnvironment *may* contain too much variables now.
225
        self.overrideEnv('TERM', 'xterm')
4597.3.36 by Vincent Ladeuil
Fix imports and various cleanups in test_ui.
226
        factory = _mod_ui_text.TextUIFactory(
227
            stdin=tests.StringIOWrapper("yada\ny\n"),
228
            stdout=out, stderr=out)
5339.2.6 by Martin Pool
One more UI test needs updates for spinner being at the front
229
        factory._avail_width = lambda: 79
3882.8.10 by Martin Pool
Fix up test_ui for new progress bars
230
        pb = factory.nested_progress_bar()
231
        pb.show_bar = False
232
        pb.show_spinner = False
233
        pb.show_count = False
234
        pb.update("foo", 0, 1)
2363.4.4 by Vincent Ladeuil
More tidying-up.
235
        self.assertEqual(True,
236
                         self.apply_redirected(None, factory.stdout,
237
                                               factory.stdout,
238
                                               factory.get_boolean,
5863.6.1 by Jelmer Vernooij
Require a unicode prompt to be passed into all methods that prompt.
239
                                               u"what do you want"))
3882.8.10 by Martin Pool
Fix up test_ui for new progress bars
240
        output = out.getvalue()
5339.2.6 by Martin Pool
One more UI test needs updates for spinner being at the front
241
        self.assertContainsRe(output,
242
            "| foo *\r\r  *\r*")
6182.2.4 by Benoît Pierre
Fix ui tests.
243
        self.assertContainsString(output,
244
            r"what do you want? ([y]es, [n]o): what do you want? ([y]es, [n]o): ")
3882.8.10 by Martin Pool
Fix up test_ui for new progress bars
245
        # stdin should have been totally consumed
2363.4.6 by Vincent Ladeuil
Fix tests around stdin emptyness.
246
        self.assertEqual('', factory.stdin.readline())
4017.1.1 by John Arbash Meinel
Get a pb.tick() to work after calling pb.update()
247
248
    def test_text_tick_after_update(self):
4597.3.36 by Vincent Ladeuil
Fix imports and various cleanups in test_ui.
249
        ui_factory = _mod_ui_text.TextUIFactory(stdout=tests.StringIOWrapper(),
250
                                                stderr=tests.StringIOWrapper())
4017.1.1 by John Arbash Meinel
Get a pb.tick() to work after calling pb.update()
251
        pb = ui_factory.nested_progress_bar()
252
        try:
253
            pb.update('task', 0, 3)
254
            # Reset the clock, so that it actually tries to repaint itself
255
            ui_factory._progress_view._last_repaint = time.time() - 1.0
256
            pb.tick()
257
        finally:
258
            pb.finished()
4110.2.15 by Martin Pool
Fix bug in showing task progress and add a test
259
4222.2.1 by Jelmer Vernooij
Add get_username() call to the UIFactory.
260
    def test_text_ui_getusername(self):
4597.3.36 by Vincent Ladeuil
Fix imports and various cleanups in test_ui.
261
        factory = _mod_ui_text.TextUIFactory(None, None, None)
262
        factory.stdin = tests.StringIOWrapper("someuser\n\n")
263
        factory.stdout = tests.StringIOWrapper()
264
        factory.stderr = tests.StringIOWrapper()
4222.2.6 by Jelmer Vernooij
Remove use of NotATerminal.
265
        factory.stdout.encoding = "utf8"
4222.2.1 by Jelmer Vernooij
Add get_username() call to the UIFactory.
266
        # there is no output from the base factory
4368.3.1 by Vincent Ladeuil
Use stderr for UI prompt to address bug #376582.
267
        self.assertEqual("someuser",
5863.6.1 by Jelmer Vernooij
Require a unicode prompt to be passed into all methods that prompt.
268
                         factory.get_username(u'Hello %(host)s', host='some'))
4368.3.1 by Vincent Ladeuil
Use stderr for UI prompt to address bug #376582.
269
        self.assertEquals("Hello some: ", factory.stderr.getvalue())
270
        self.assertEquals('', factory.stdout.getvalue())
5863.6.1 by Jelmer Vernooij
Require a unicode prompt to be passed into all methods that prompt.
271
        self.assertEqual("", factory.get_username(u"Gebruiker"))
4222.2.1 by Jelmer Vernooij
Add get_username() call to the UIFactory.
272
        # stdin should be empty
273
        self.assertEqual('', factory.stdin.readline())
274
4222.2.2 by Jelmer Vernooij
Review from vila: Deal with UTF8 strings in prompts, fix typo.
275
    def test_text_ui_getusername_utf8(self):
4488.1.1 by Vincent Ladeuil
(vila) Cleanup imports in some test files
276
        ui = tests.TestUIFactory(stdin=u'someuser\u1234'.encode('utf8'),
277
                                 stdout=tests.StringIOWrapper(),
278
                                 stderr=tests.StringIOWrapper())
4368.3.1 by Vincent Ladeuil
Use stderr for UI prompt to address bug #376582.
279
        ui.stderr.encoding = ui.stdout.encoding = ui.stdin.encoding = "utf8"
4222.2.3 by Jelmer Vernooij
Also check for unicode usernames.
280
        pb = ui.nested_progress_bar()
281
        try:
282
            # there is no output from the base factory
4368.3.1 by Vincent Ladeuil
Use stderr for UI prompt to address bug #376582.
283
            username = self.apply_redirected(ui.stdin, ui.stdout, ui.stderr,
4222.2.12 by Jelmer Vernooij
Redirect to fix utf8 test with LC_ALL=C.
284
                ui.get_username, u'Hello\u1234 %(host)s', host=u'some\u1234')
4222.2.8 by Jelmer Vernooij
Fix copy-n-paste error.
285
            self.assertEquals(u"someuser\u1234", username.decode('utf8'))
4368.3.1 by Vincent Ladeuil
Use stderr for UI prompt to address bug #376582.
286
            self.assertEquals(u"Hello\u1234 some\u1234: ",
287
                              ui.stderr.getvalue().decode("utf8"))
288
            self.assertEquals('', ui.stdout.getvalue())
4222.2.3 by Jelmer Vernooij
Also check for unicode usernames.
289
        finally:
290
            pb.finished()
4222.2.2 by Jelmer Vernooij
Review from vila: Deal with UTF8 strings in prompts, fix typo.
291
4961.1.3 by Martin Pool
trace quietness now controls whether the progress bar appears
292
    def test_quietness(self):
5570.3.9 by Vincent Ladeuil
More use cases for overrideEnv, _cleanEnvironment *may* contain too much variables now.
293
        self.overrideEnv('BZR_PROGRESS_BAR', 'text')
4961.1.3 by Martin Pool
trace quietness now controls whether the progress bar appears
294
        ui_factory = _mod_ui_text.TextUIFactory(None,
295
            test_progress._TTYStringIO(),
296
            test_progress._TTYStringIO())
297
        self.assertIsInstance(ui_factory._progress_view,
298
            _mod_ui_text.TextProgressView)
299
        ui_factory.be_quiet(True)
300
        self.assertIsInstance(ui_factory._progress_view,
301
            _mod_ui_text.NullProgressView)
302
4634.144.10 by Martin Pool
Update test_ui for warning suppression
303
    def test_text_ui_show_user_warning(self):
4634.144.5 by Martin Pool
Cleaner presentation and tests for warn_cross_format_fetch
304
        from bzrlib.repofmt.groupcompress_repo import RepositoryFormat2a
5757.1.7 by Jelmer Vernooij
Fix more imports.
305
        from bzrlib.repofmt.knitpack_repo import RepositoryFormatKnitPack5
4634.144.5 by Martin Pool
Cleaner presentation and tests for warn_cross_format_fetch
306
        err = StringIO()
307
        out = StringIO()
308
        ui = tests.TextUIFactory(stdin=None, stdout=out, stderr=err)
309
        remote_fmt = remote.RemoteRepositoryFormat()
310
        remote_fmt._network_name = RepositoryFormatKnitPack5().network_name()
4634.144.10 by Martin Pool
Update test_ui for warning suppression
311
        ui.show_user_warning('cross_format_fetch', from_format=RepositoryFormat2a(),
312
            to_format=remote_fmt)
4634.144.5 by Martin Pool
Cleaner presentation and tests for warn_cross_format_fetch
313
        self.assertEquals('', out.getvalue())
314
        self.assertEquals("Doing on-the-fly conversion from RepositoryFormat2a() to "
315
            "RemoteRepositoryFormat(_network_name='Bazaar RepositoryFormatKnitPack5 "
316
            "(bzr 1.6)\\n').\nThis may take some time. Upgrade the repositories to "
317
            "the same format for better performance.\n",
318
            err.getvalue())
4634.144.10 by Martin Pool
Update test_ui for warning suppression
319
        # and now with it suppressed please
320
        err = StringIO()
321
        out = StringIO()
322
        ui = tests.TextUIFactory(stdin=None, stdout=out, stderr=err)
4634.144.11 by Martin Pool
Rename squelched to suppressed
323
        ui.suppressed_warnings.add('cross_format_fetch')
4634.144.10 by Martin Pool
Update test_ui for warning suppression
324
        ui.show_user_warning('cross_format_fetch', from_format=RepositoryFormat2a(),
325
            to_format=remote_fmt)
326
        self.assertEquals('', out.getvalue())
327
        self.assertEquals('', err.getvalue())
4634.144.5 by Martin Pool
Cleaner presentation and tests for warn_cross_format_fetch
328
4110.2.15 by Martin Pool
Fix bug in showing task progress and add a test
329
4884.1.1 by Vincent Ladeuil
Cleanup some test imports
330
class TestTextUIOutputStream(tests.TestCase):
4792.8.1 by Martin Pool
Add TextUIOutputStream coordinated with progress view
331
    """Tests for output stream that synchronizes with progress bar."""
332
333
    def test_output_clears_terminal(self):
4884.1.1 by Vincent Ladeuil
Cleanup some test imports
334
        stdout = tests.StringIOWrapper()
335
        stderr = tests.StringIOWrapper()
4792.8.1 by Martin Pool
Add TextUIOutputStream coordinated with progress view
336
        clear_calls = []
337
4884.1.1 by Vincent Ladeuil
Cleanup some test imports
338
        uif =  _mod_ui_text.TextUIFactory(None, stdout, stderr)
4792.8.1 by Martin Pool
Add TextUIOutputStream coordinated with progress view
339
        uif.clear_term = lambda: clear_calls.append('clear')
340
4884.1.1 by Vincent Ladeuil
Cleanup some test imports
341
        stream = _mod_ui_text.TextUIOutputStream(uif, uif.stdout)
4792.8.1 by Martin Pool
Add TextUIOutputStream coordinated with progress view
342
        stream.write("Hello world!\n")
343
        stream.write("there's more...\n")
4792.8.3 by Martin Pool
Add TextUIOutputStream.writelines
344
        stream.writelines(["1\n", "2\n", "3\n"])
4884.1.1 by Vincent Ladeuil
Cleanup some test imports
345
4792.8.1 by Martin Pool
Add TextUIOutputStream coordinated with progress view
346
        self.assertEqual(stdout.getvalue(),
347
            "Hello world!\n"
4792.8.3 by Martin Pool
Add TextUIOutputStream.writelines
348
            "there's more...\n"
349
            "1\n2\n3\n")
350
        self.assertEqual(['clear', 'clear', 'clear'],
4792.8.1 by Martin Pool
Add TextUIOutputStream coordinated with progress view
351
            clear_calls)
352
4792.8.8 by Martin Pool
Add TextUIOutputStream.flush
353
        stream.flush()
4792.8.1 by Martin Pool
Add TextUIOutputStream coordinated with progress view
354
355
4711.1.6 by Martin Pool
Separate TextUIFactory tests
356
class UITests(tests.TestCase):
357
358
    def test_progress_construction(self):
359
        """TextUIFactory constructs the right progress view.
360
        """
4597.3.36 by Vincent Ladeuil
Fix imports and various cleanups in test_ui.
361
        TTYStringIO = test_progress._TTYStringIO
362
        FileStringIO = tests.StringIOWrapper
4711.1.6 by Martin Pool
Separate TextUIFactory tests
363
        for (file_class, term, pb, expected_pb_class) in (
364
            # on an xterm, either use them or not as the user requests,
365
            # otherwise default on
4597.3.36 by Vincent Ladeuil
Fix imports and various cleanups in test_ui.
366
            (TTYStringIO, 'xterm', 'none', _mod_ui_text.NullProgressView),
367
            (TTYStringIO, 'xterm', 'text', _mod_ui_text.TextProgressView),
368
            (TTYStringIO, 'xterm', None, _mod_ui_text.TextProgressView),
4711.1.6 by Martin Pool
Separate TextUIFactory tests
369
            # on a dumb terminal, again if there's explicit configuration do
370
            # it, otherwise default off
4597.3.36 by Vincent Ladeuil
Fix imports and various cleanups in test_ui.
371
            (TTYStringIO, 'dumb', 'none', _mod_ui_text.NullProgressView),
372
            (TTYStringIO, 'dumb', 'text', _mod_ui_text.TextProgressView),
373
            (TTYStringIO, 'dumb', None, _mod_ui_text.NullProgressView),
4711.1.6 by Martin Pool
Separate TextUIFactory tests
374
            # on a non-tty terminal, it's null regardless of $TERM
4597.3.36 by Vincent Ladeuil
Fix imports and various cleanups in test_ui.
375
            (FileStringIO, 'xterm', None, _mod_ui_text.NullProgressView),
376
            (FileStringIO, 'dumb', None, _mod_ui_text.NullProgressView),
4711.1.6 by Martin Pool
Separate TextUIFactory tests
377
            # however, it can still be forced on
4597.3.36 by Vincent Ladeuil
Fix imports and various cleanups in test_ui.
378
            (FileStringIO, 'dumb', 'text', _mod_ui_text.TextProgressView),
4711.1.6 by Martin Pool
Separate TextUIFactory tests
379
            ):
5570.3.9 by Vincent Ladeuil
More use cases for overrideEnv, _cleanEnvironment *may* contain too much variables now.
380
            self.overrideEnv('TERM', term)
381
            self.overrideEnv('BZR_PROGRESS_BAR', pb)
4711.1.6 by Martin Pool
Separate TextUIFactory tests
382
            stdin = file_class('')
383
            stderr = file_class()
384
            stdout = file_class()
4597.3.36 by Vincent Ladeuil
Fix imports and various cleanups in test_ui.
385
            uif = _mod_ui.make_ui_for_terminal(stdin, stdout, stderr)
386
            self.assertIsInstance(uif, _mod_ui_text.TextUIFactory,
4711.1.6 by Martin Pool
Separate TextUIFactory tests
387
                "TERM=%s BZR_PROGRESS_BAR=%s uif=%r" % (term, pb, uif,))
388
            self.assertIsInstance(uif.make_progress_view(),
389
                expected_pb_class,
390
                "TERM=%s BZR_PROGRESS_BAR=%s uif=%r" % (term, pb, uif,))
391
392
    def test_text_ui_non_terminal(self):
393
        """Even on non-ttys, make_ui_for_terminal gives a text ui."""
4597.3.36 by Vincent Ladeuil
Fix imports and various cleanups in test_ui.
394
        stdin = test_progress._NonTTYStringIO('')
395
        stderr = test_progress._NonTTYStringIO()
396
        stdout = test_progress._NonTTYStringIO()
4711.1.6 by Martin Pool
Separate TextUIFactory tests
397
        for term_type in ['dumb', None, 'xterm']:
5570.3.9 by Vincent Ladeuil
More use cases for overrideEnv, _cleanEnvironment *may* contain too much variables now.
398
            self.overrideEnv('TERM', term_type)
4597.3.36 by Vincent Ladeuil
Fix imports and various cleanups in test_ui.
399
            uif = _mod_ui.make_ui_for_terminal(stdin, stdout, stderr)
400
            self.assertIsInstance(uif, _mod_ui_text.TextUIFactory,
4711.1.6 by Martin Pool
Separate TextUIFactory tests
401
                'TERM=%r' % (term_type,))
402
403
4597.3.36 by Vincent Ladeuil
Fix imports and various cleanups in test_ui.
404
class SilentUITests(tests.TestCase):
4449.3.19 by Martin Pool
SilentUIFactory now always errors when asked for input
405
4449.3.36 by Martin Pool
Update tests: SilentUIFactory no longer does get_boolean or get_password
406
    def test_silent_factory_get_password(self):
407
        # A silent factory that can't do user interaction can't get a
408
        # password.  Possibly it should raise a more specific error but it
409
        # can't succeed.
4597.3.36 by Vincent Ladeuil
Fix imports and various cleanups in test_ui.
410
        ui = _mod_ui.SilentUIFactory()
411
        stdout = tests.StringIOWrapper()
4449.3.19 by Martin Pool
SilentUIFactory now always errors when asked for input
412
        self.assertRaises(
413
            NotImplementedError,
414
            self.apply_redirected,
415
            None, stdout, stdout, ui.get_password)
416
        # and it didn't write anything out either
417
        self.assertEqual('', stdout.getvalue())
418
419
    def test_silent_ui_getbool(self):
4597.3.36 by Vincent Ladeuil
Fix imports and various cleanups in test_ui.
420
        factory = _mod_ui.SilentUIFactory()
421
        stdout = tests.StringIOWrapper()
4449.3.19 by Martin Pool
SilentUIFactory now always errors when asked for input
422
        self.assertRaises(
423
            NotImplementedError,
424
            self.apply_redirected,
5863.6.1 by Jelmer Vernooij
Require a unicode prompt to be passed into all methods that prompt.
425
            None, stdout, stdout, factory.get_boolean, u"foo")
4449.3.42 by Martin Pool
Add basic test for CannedInputUIFactory
426
427
4597.3.36 by Vincent Ladeuil
Fix imports and various cleanups in test_ui.
428
class TestUIFactoryTests(tests.TestCase):
4580.2.2 by Martin Pool
Add test for bug 408201
429
430
    def test_test_ui_factory_progress(self):
431
        # there's no output; we just want to make sure this doesn't crash -
5243.1.2 by Martin
Point launchpad links in comments at production server rather than edge
432
        # see https://bugs.launchpad.net/bzr/+bug/408201
4597.3.36 by Vincent Ladeuil
Fix imports and various cleanups in test_ui.
433
        ui = tests.TestUIFactory()
4580.2.2 by Martin Pool
Add test for bug 408201
434
        pb = ui.nested_progress_bar()
435
        pb.update('hello')
436
        pb.tick()
437
        pb.finished()
438
439
4597.3.36 by Vincent Ladeuil
Fix imports and various cleanups in test_ui.
440
class CannedInputUIFactoryTests(tests.TestCase):
441
4449.3.42 by Martin Pool
Add basic test for CannedInputUIFactory
442
    def test_canned_input_get_input(self):
4597.3.37 by Vincent Ladeuil
Allows ui factories to query users for an integer.
443
        uif = _mod_ui.CannedInputUIFactory([True, 'mbp', 'password', 42])
5863.6.1 by Jelmer Vernooij
Require a unicode prompt to be passed into all methods that prompt.
444
        self.assertEqual(True, uif.get_boolean(u'Extra cheese?'))
445
        self.assertEqual('mbp', uif.get_username(u'Enter your user name'))
4597.3.37 by Vincent Ladeuil
Allows ui factories to query users for an integer.
446
        self.assertEqual('password',
5863.6.1 by Jelmer Vernooij
Require a unicode prompt to be passed into all methods that prompt.
447
                         uif.get_password(u'Password for %(host)s',
4597.3.37 by Vincent Ladeuil
Allows ui factories to query users for an integer.
448
                                          host='example.com'))
5863.6.1 by Jelmer Vernooij
Require a unicode prompt to be passed into all methods that prompt.
449
        self.assertEqual(42, uif.get_integer(u'And all that jazz ?'))
4110.2.17 by Martin Pool
If one ProgressTask has no count, it passes through that of its child
450
4503.2.1 by Vincent Ladeuil
Get a bool from a string.
451
452
class TestBoolFromString(tests.TestCase):
453
454
    def assertIsTrue(self, s, accepted_values=None):
455
        res = _mod_ui.bool_from_string(s, accepted_values=accepted_values)
456
        self.assertEquals(True, res)
457
458
    def assertIsFalse(self, s, accepted_values=None):
459
        res = _mod_ui.bool_from_string(s, accepted_values=accepted_values)
460
        self.assertEquals(False, res)
461
462
    def assertIsNone(self, s, accepted_values=None):
463
        res = _mod_ui.bool_from_string(s, accepted_values=accepted_values)
464
        self.assertIs(None, res)
465
466
    def test_know_valid_values(self):
467
        self.assertIsTrue('true')
468
        self.assertIsFalse('false')
469
        self.assertIsTrue('1')
470
        self.assertIsFalse('0')
471
        self.assertIsTrue('on')
472
        self.assertIsFalse('off')
473
        self.assertIsTrue('yes')
474
        self.assertIsFalse('no')
475
        self.assertIsTrue('y')
476
        self.assertIsFalse('n')
477
        # Also try some case variations
478
        self.assertIsTrue('True')
479
        self.assertIsFalse('False')
480
        self.assertIsTrue('On')
481
        self.assertIsFalse('Off')
482
        self.assertIsTrue('ON')
483
        self.assertIsFalse('OFF')
484
        self.assertIsTrue('oN')
485
        self.assertIsFalse('oFf')
486
487
    def test_invalid_values(self):
488
        self.assertIsNone(None)
489
        self.assertIsNone('doubt')
490
        self.assertIsNone('frue')
491
        self.assertIsNone('talse')
492
        self.assertIsNone('42')
493
494
    def test_provided_values(self):
495
        av = dict(y=True, n=False, yes=True, no=False)
496
        self.assertIsTrue('y', av)
497
        self.assertIsTrue('Y', av)
498
        self.assertIsTrue('Yes', av)
499
        self.assertIsFalse('n', av)
500
        self.assertIsFalse('N', av)
501
        self.assertIsFalse('No', av)
502
        self.assertIsNone('1', av)
503
        self.assertIsNone('0', av)
504
        self.assertIsNone('on', av)
505
        self.assertIsNone('off', av)
5422.1.1 by Martin Pool
Move CapturingUIFactory out of per_workingtree tests into somewhere reusable
506
507
5416.1.11 by Martin Pool
Add ConfirmationUserInterfacePolicy that lets specific confirmations be configured off.
508
class TestConfirmationUserInterfacePolicy(tests.TestCase):
509
510
    def test_confirm_action_default(self):
511
        base_ui = _mod_ui.NoninteractiveUIFactory()
512
        for answer in [True, False]:
513
            self.assertEquals(
514
                _mod_ui.ConfirmationUserInterfacePolicy(base_ui, answer, {})
515
                .confirm_action("Do something?",
516
                    "bzrlib.tests.do_something", {}),
517
                answer)
518
519
    def test_confirm_action_specific(self):
520
        base_ui = _mod_ui.NoninteractiveUIFactory()
521
        for default_answer in [True, False]:
522
            for specific_answer in [True, False]:
523
                for conf_id in ['given_id', 'other_id']:
524
                    wrapper = _mod_ui.ConfirmationUserInterfacePolicy(
525
                        base_ui, default_answer, dict(given_id=specific_answer))
526
                    result = wrapper.confirm_action("Do something?", conf_id, {})
527
                    if conf_id == 'given_id':
528
                        self.assertEquals(result, specific_answer)
529
                    else:
530
                        self.assertEquals(result, default_answer)
531
532
    def test_repr(self):
533
        base_ui = _mod_ui.NoninteractiveUIFactory()
534
        wrapper = _mod_ui.ConfirmationUserInterfacePolicy(
535
            base_ui, True, dict(a=2))
536
        self.assertThat(repr(wrapper),
537
            Equals("ConfirmationUserInterfacePolicy("
538
                "NoninteractiveUIFactory(), True, {'a': 2})"))
5416.2.4 by Martin Pool
resolve against trunk
539
540
5422.1.4 by Martin Pool
Rename CapturingUIFactory to ProgressRecordingUIFactory
541
class TestProgressRecordingUI(tests.TestCase):
5422.1.1 by Martin Pool
Move CapturingUIFactory out of per_workingtree tests into somewhere reusable
542
    """Test test-oriented UIFactory that records progress updates"""
543
544
    def test_nested_ignore_depth_beyond_one(self):
545
        # we only want to capture the first level out progress, not
546
        # want sub-components might do. So we have nested bars ignored.
5422.1.4 by Martin Pool
Rename CapturingUIFactory to ProgressRecordingUIFactory
547
        factory = ProgressRecordingUIFactory()
5422.1.1 by Martin Pool
Move CapturingUIFactory out of per_workingtree tests into somewhere reusable
548
        pb1 = factory.nested_progress_bar()
549
        pb1.update('foo', 0, 1)
550
        pb2 = factory.nested_progress_bar()
551
        pb2.update('foo', 0, 1)
552
        pb2.finished()
553
        pb1.finished()
554
        self.assertEqual([("update", 0, 1, 'foo')], factory._calls)