~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
114
                                      "yes with garbage\nY\n" # True
115
                                      "not an answer\nno\n" # False
116
                                      "I'm sure!\nyes\n" # True
117
                                      "NO\n" # False
118
                                      "foo\n")
4597.3.36 by Vincent Ladeuil
Fix imports and various cleanups in test_ui.
119
        stdout = tests.StringIOWrapper()
120
        stderr = tests.StringIOWrapper()
121
        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.
122
        self.assertEqual(True, factory.get_boolean(u""))
123
        self.assertEqual(False, factory.get_boolean(u""))
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""))
1687.1.4 by Robert Collins
Add bzrlib.ui.ui_factory.get_boolean().
128
        self.assertEqual("foo\n", factory.stdin.read())
2363.4.4 by Vincent Ladeuil
More tidying-up.
129
        # stdin should be empty
2363.4.6 by Vincent Ladeuil
Fix tests around stdin emptyness.
130
        self.assertEqual('', factory.stdin.readline())
1687.1.4 by Robert Collins
Add bzrlib.ui.ui_factory.get_boolean().
131
4597.3.37 by Vincent Ladeuil
Allows ui factories to query users for an integer.
132
    def test_text_ui_get_integer(self):
133
        stdin = tests.StringIOWrapper(
134
            "1\n"
135
            "  -2  \n"
136
            "hmmm\nwhat else ?\nCome on\nok 42\n4.24\n42\n")
137
        stdout = tests.StringIOWrapper()
138
        stderr = tests.StringIOWrapper()
139
        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.
140
        self.assertEqual(1, factory.get_integer(u""))
141
        self.assertEqual(-2, factory.get_integer(u""))
142
        self.assertEqual(42, factory.get_integer(u""))
4597.3.37 by Vincent Ladeuil
Allows ui factories to query users for an integer.
143
4300.3.1 by Martin Pool
Fix string expansion in TextUIFactory.prompt
144
    def test_text_factory_prompt(self):
145
        # see <https://launchpad.net/bugs/365891>
4597.3.36 by Vincent Ladeuil
Fix imports and various cleanups in test_ui.
146
        StringIO = tests.StringIOWrapper
147
        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.
148
        factory.prompt(u'foo %2e')
4368.3.1 by Vincent Ladeuil
Use stderr for UI prompt to address bug #376582.
149
        self.assertEqual('', factory.stdout.getvalue())
150
        self.assertEqual('foo %2e', factory.stderr.getvalue())
4300.3.1 by Martin Pool
Fix string expansion in TextUIFactory.prompt
151
1687.1.4 by Robert Collins
Add bzrlib.ui.ui_factory.get_boolean().
152
    def test_text_factory_prompts_and_clears(self):
153
        # a get_boolean call should clear the pb before prompting
4597.3.36 by Vincent Ladeuil
Fix imports and various cleanups in test_ui.
154
        out = test_progress._TTYStringIO()
5570.3.9 by Vincent Ladeuil
More use cases for overrideEnv, _cleanEnvironment *may* contain too much variables now.
155
        self.overrideEnv('TERM', 'xterm')
4597.3.36 by Vincent Ladeuil
Fix imports and various cleanups in test_ui.
156
        factory = _mod_ui_text.TextUIFactory(
157
            stdin=tests.StringIOWrapper("yada\ny\n"),
158
            stdout=out, stderr=out)
5339.2.6 by Martin Pool
One more UI test needs updates for spinner being at the front
159
        factory._avail_width = lambda: 79
3882.8.10 by Martin Pool
Fix up test_ui for new progress bars
160
        pb = factory.nested_progress_bar()
161
        pb.show_bar = False
162
        pb.show_spinner = False
163
        pb.show_count = False
164
        pb.update("foo", 0, 1)
2363.4.4 by Vincent Ladeuil
More tidying-up.
165
        self.assertEqual(True,
166
                         self.apply_redirected(None, factory.stdout,
167
                                               factory.stdout,
168
                                               factory.get_boolean,
5863.6.1 by Jelmer Vernooij
Require a unicode prompt to be passed into all methods that prompt.
169
                                               u"what do you want"))
3882.8.10 by Martin Pool
Fix up test_ui for new progress bars
170
        output = out.getvalue()
5339.2.6 by Martin Pool
One more UI test needs updates for spinner being at the front
171
        self.assertContainsRe(output,
172
            "| foo *\r\r  *\r*")
173
        self.assertContainsRe(output,
3882.8.10 by Martin Pool
Fix up test_ui for new progress bars
174
            r"what do you want\? \[y/n\]: what do you want\? \[y/n\]: ")
175
        # stdin should have been totally consumed
2363.4.6 by Vincent Ladeuil
Fix tests around stdin emptyness.
176
        self.assertEqual('', factory.stdin.readline())
4017.1.1 by John Arbash Meinel
Get a pb.tick() to work after calling pb.update()
177
178
    def test_text_tick_after_update(self):
4597.3.36 by Vincent Ladeuil
Fix imports and various cleanups in test_ui.
179
        ui_factory = _mod_ui_text.TextUIFactory(stdout=tests.StringIOWrapper(),
180
                                                stderr=tests.StringIOWrapper())
4017.1.1 by John Arbash Meinel
Get a pb.tick() to work after calling pb.update()
181
        pb = ui_factory.nested_progress_bar()
182
        try:
183
            pb.update('task', 0, 3)
184
            # Reset the clock, so that it actually tries to repaint itself
185
            ui_factory._progress_view._last_repaint = time.time() - 1.0
186
            pb.tick()
187
        finally:
188
            pb.finished()
4110.2.15 by Martin Pool
Fix bug in showing task progress and add a test
189
4222.2.1 by Jelmer Vernooij
Add get_username() call to the UIFactory.
190
    def test_text_ui_getusername(self):
4597.3.36 by Vincent Ladeuil
Fix imports and various cleanups in test_ui.
191
        factory = _mod_ui_text.TextUIFactory(None, None, None)
192
        factory.stdin = tests.StringIOWrapper("someuser\n\n")
193
        factory.stdout = tests.StringIOWrapper()
194
        factory.stderr = tests.StringIOWrapper()
4222.2.6 by Jelmer Vernooij
Remove use of NotATerminal.
195
        factory.stdout.encoding = "utf8"
4222.2.1 by Jelmer Vernooij
Add get_username() call to the UIFactory.
196
        # there is no output from the base factory
4368.3.1 by Vincent Ladeuil
Use stderr for UI prompt to address bug #376582.
197
        self.assertEqual("someuser",
5863.6.1 by Jelmer Vernooij
Require a unicode prompt to be passed into all methods that prompt.
198
                         factory.get_username(u'Hello %(host)s', host='some'))
4368.3.1 by Vincent Ladeuil
Use stderr for UI prompt to address bug #376582.
199
        self.assertEquals("Hello some: ", factory.stderr.getvalue())
200
        self.assertEquals('', factory.stdout.getvalue())
5863.6.1 by Jelmer Vernooij
Require a unicode prompt to be passed into all methods that prompt.
201
        self.assertEqual("", factory.get_username(u"Gebruiker"))
4222.2.1 by Jelmer Vernooij
Add get_username() call to the UIFactory.
202
        # stdin should be empty
203
        self.assertEqual('', factory.stdin.readline())
204
4222.2.2 by Jelmer Vernooij
Review from vila: Deal with UTF8 strings in prompts, fix typo.
205
    def test_text_ui_getusername_utf8(self):
4488.1.1 by Vincent Ladeuil
(vila) Cleanup imports in some test files
206
        ui = tests.TestUIFactory(stdin=u'someuser\u1234'.encode('utf8'),
207
                                 stdout=tests.StringIOWrapper(),
208
                                 stderr=tests.StringIOWrapper())
4368.3.1 by Vincent Ladeuil
Use stderr for UI prompt to address bug #376582.
209
        ui.stderr.encoding = ui.stdout.encoding = ui.stdin.encoding = "utf8"
4222.2.3 by Jelmer Vernooij
Also check for unicode usernames.
210
        pb = ui.nested_progress_bar()
211
        try:
212
            # there is no output from the base factory
4368.3.1 by Vincent Ladeuil
Use stderr for UI prompt to address bug #376582.
213
            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.
214
                ui.get_username, u'Hello\u1234 %(host)s', host=u'some\u1234')
4222.2.8 by Jelmer Vernooij
Fix copy-n-paste error.
215
            self.assertEquals(u"someuser\u1234", username.decode('utf8'))
4368.3.1 by Vincent Ladeuil
Use stderr for UI prompt to address bug #376582.
216
            self.assertEquals(u"Hello\u1234 some\u1234: ",
217
                              ui.stderr.getvalue().decode("utf8"))
218
            self.assertEquals('', ui.stdout.getvalue())
4222.2.3 by Jelmer Vernooij
Also check for unicode usernames.
219
        finally:
220
            pb.finished()
4222.2.2 by Jelmer Vernooij
Review from vila: Deal with UTF8 strings in prompts, fix typo.
221
4961.1.3 by Martin Pool
trace quietness now controls whether the progress bar appears
222
    def test_quietness(self):
5570.3.9 by Vincent Ladeuil
More use cases for overrideEnv, _cleanEnvironment *may* contain too much variables now.
223
        self.overrideEnv('BZR_PROGRESS_BAR', 'text')
4961.1.3 by Martin Pool
trace quietness now controls whether the progress bar appears
224
        ui_factory = _mod_ui_text.TextUIFactory(None,
225
            test_progress._TTYStringIO(),
226
            test_progress._TTYStringIO())
227
        self.assertIsInstance(ui_factory._progress_view,
228
            _mod_ui_text.TextProgressView)
229
        ui_factory.be_quiet(True)
230
        self.assertIsInstance(ui_factory._progress_view,
231
            _mod_ui_text.NullProgressView)
232
4634.144.10 by Martin Pool
Update test_ui for warning suppression
233
    def test_text_ui_show_user_warning(self):
4634.144.5 by Martin Pool
Cleaner presentation and tests for warn_cross_format_fetch
234
        from bzrlib.repofmt.groupcompress_repo import RepositoryFormat2a
5757.1.7 by Jelmer Vernooij
Fix more imports.
235
        from bzrlib.repofmt.knitpack_repo import RepositoryFormatKnitPack5
4634.144.5 by Martin Pool
Cleaner presentation and tests for warn_cross_format_fetch
236
        err = StringIO()
237
        out = StringIO()
238
        ui = tests.TextUIFactory(stdin=None, stdout=out, stderr=err)
239
        remote_fmt = remote.RemoteRepositoryFormat()
240
        remote_fmt._network_name = RepositoryFormatKnitPack5().network_name()
4634.144.10 by Martin Pool
Update test_ui for warning suppression
241
        ui.show_user_warning('cross_format_fetch', from_format=RepositoryFormat2a(),
242
            to_format=remote_fmt)
4634.144.5 by Martin Pool
Cleaner presentation and tests for warn_cross_format_fetch
243
        self.assertEquals('', out.getvalue())
244
        self.assertEquals("Doing on-the-fly conversion from RepositoryFormat2a() to "
245
            "RemoteRepositoryFormat(_network_name='Bazaar RepositoryFormatKnitPack5 "
246
            "(bzr 1.6)\\n').\nThis may take some time. Upgrade the repositories to "
247
            "the same format for better performance.\n",
248
            err.getvalue())
4634.144.10 by Martin Pool
Update test_ui for warning suppression
249
        # and now with it suppressed please
250
        err = StringIO()
251
        out = StringIO()
252
        ui = tests.TextUIFactory(stdin=None, stdout=out, stderr=err)
4634.144.11 by Martin Pool
Rename squelched to suppressed
253
        ui.suppressed_warnings.add('cross_format_fetch')
4634.144.10 by Martin Pool
Update test_ui for warning suppression
254
        ui.show_user_warning('cross_format_fetch', from_format=RepositoryFormat2a(),
255
            to_format=remote_fmt)
256
        self.assertEquals('', out.getvalue())
257
        self.assertEquals('', err.getvalue())
4634.144.5 by Martin Pool
Cleaner presentation and tests for warn_cross_format_fetch
258
4110.2.15 by Martin Pool
Fix bug in showing task progress and add a test
259
4884.1.1 by Vincent Ladeuil
Cleanup some test imports
260
class TestTextUIOutputStream(tests.TestCase):
4792.8.1 by Martin Pool
Add TextUIOutputStream coordinated with progress view
261
    """Tests for output stream that synchronizes with progress bar."""
262
263
    def test_output_clears_terminal(self):
4884.1.1 by Vincent Ladeuil
Cleanup some test imports
264
        stdout = tests.StringIOWrapper()
265
        stderr = tests.StringIOWrapper()
4792.8.1 by Martin Pool
Add TextUIOutputStream coordinated with progress view
266
        clear_calls = []
267
4884.1.1 by Vincent Ladeuil
Cleanup some test imports
268
        uif =  _mod_ui_text.TextUIFactory(None, stdout, stderr)
4792.8.1 by Martin Pool
Add TextUIOutputStream coordinated with progress view
269
        uif.clear_term = lambda: clear_calls.append('clear')
270
4884.1.1 by Vincent Ladeuil
Cleanup some test imports
271
        stream = _mod_ui_text.TextUIOutputStream(uif, uif.stdout)
4792.8.1 by Martin Pool
Add TextUIOutputStream coordinated with progress view
272
        stream.write("Hello world!\n")
273
        stream.write("there's more...\n")
4792.8.3 by Martin Pool
Add TextUIOutputStream.writelines
274
        stream.writelines(["1\n", "2\n", "3\n"])
4884.1.1 by Vincent Ladeuil
Cleanup some test imports
275
4792.8.1 by Martin Pool
Add TextUIOutputStream coordinated with progress view
276
        self.assertEqual(stdout.getvalue(),
277
            "Hello world!\n"
4792.8.3 by Martin Pool
Add TextUIOutputStream.writelines
278
            "there's more...\n"
279
            "1\n2\n3\n")
280
        self.assertEqual(['clear', 'clear', 'clear'],
4792.8.1 by Martin Pool
Add TextUIOutputStream coordinated with progress view
281
            clear_calls)
282
4792.8.8 by Martin Pool
Add TextUIOutputStream.flush
283
        stream.flush()
4792.8.1 by Martin Pool
Add TextUIOutputStream coordinated with progress view
284
285
4711.1.6 by Martin Pool
Separate TextUIFactory tests
286
class UITests(tests.TestCase):
287
288
    def test_progress_construction(self):
289
        """TextUIFactory constructs the right progress view.
290
        """
4597.3.36 by Vincent Ladeuil
Fix imports and various cleanups in test_ui.
291
        TTYStringIO = test_progress._TTYStringIO
292
        FileStringIO = tests.StringIOWrapper
4711.1.6 by Martin Pool
Separate TextUIFactory tests
293
        for (file_class, term, pb, expected_pb_class) in (
294
            # on an xterm, either use them or not as the user requests,
295
            # otherwise default on
4597.3.36 by Vincent Ladeuil
Fix imports and various cleanups in test_ui.
296
            (TTYStringIO, 'xterm', 'none', _mod_ui_text.NullProgressView),
297
            (TTYStringIO, 'xterm', 'text', _mod_ui_text.TextProgressView),
298
            (TTYStringIO, 'xterm', None, _mod_ui_text.TextProgressView),
4711.1.6 by Martin Pool
Separate TextUIFactory tests
299
            # on a dumb terminal, again if there's explicit configuration do
300
            # it, otherwise default off
4597.3.36 by Vincent Ladeuil
Fix imports and various cleanups in test_ui.
301
            (TTYStringIO, 'dumb', 'none', _mod_ui_text.NullProgressView),
302
            (TTYStringIO, 'dumb', 'text', _mod_ui_text.TextProgressView),
303
            (TTYStringIO, 'dumb', None, _mod_ui_text.NullProgressView),
4711.1.6 by Martin Pool
Separate TextUIFactory tests
304
            # 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.
305
            (FileStringIO, 'xterm', None, _mod_ui_text.NullProgressView),
306
            (FileStringIO, 'dumb', None, _mod_ui_text.NullProgressView),
4711.1.6 by Martin Pool
Separate TextUIFactory tests
307
            # however, it can still be forced on
4597.3.36 by Vincent Ladeuil
Fix imports and various cleanups in test_ui.
308
            (FileStringIO, 'dumb', 'text', _mod_ui_text.TextProgressView),
4711.1.6 by Martin Pool
Separate TextUIFactory tests
309
            ):
5570.3.9 by Vincent Ladeuil
More use cases for overrideEnv, _cleanEnvironment *may* contain too much variables now.
310
            self.overrideEnv('TERM', term)
311
            self.overrideEnv('BZR_PROGRESS_BAR', pb)
4711.1.6 by Martin Pool
Separate TextUIFactory tests
312
            stdin = file_class('')
313
            stderr = file_class()
314
            stdout = file_class()
4597.3.36 by Vincent Ladeuil
Fix imports and various cleanups in test_ui.
315
            uif = _mod_ui.make_ui_for_terminal(stdin, stdout, stderr)
316
            self.assertIsInstance(uif, _mod_ui_text.TextUIFactory,
4711.1.6 by Martin Pool
Separate TextUIFactory tests
317
                "TERM=%s BZR_PROGRESS_BAR=%s uif=%r" % (term, pb, uif,))
318
            self.assertIsInstance(uif.make_progress_view(),
319
                expected_pb_class,
320
                "TERM=%s BZR_PROGRESS_BAR=%s uif=%r" % (term, pb, uif,))
321
322
    def test_text_ui_non_terminal(self):
323
        """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.
324
        stdin = test_progress._NonTTYStringIO('')
325
        stderr = test_progress._NonTTYStringIO()
326
        stdout = test_progress._NonTTYStringIO()
4711.1.6 by Martin Pool
Separate TextUIFactory tests
327
        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.
328
            self.overrideEnv('TERM', term_type)
4597.3.36 by Vincent Ladeuil
Fix imports and various cleanups in test_ui.
329
            uif = _mod_ui.make_ui_for_terminal(stdin, stdout, stderr)
330
            self.assertIsInstance(uif, _mod_ui_text.TextUIFactory,
4711.1.6 by Martin Pool
Separate TextUIFactory tests
331
                'TERM=%r' % (term_type,))
332
333
4597.3.36 by Vincent Ladeuil
Fix imports and various cleanups in test_ui.
334
class SilentUITests(tests.TestCase):
4449.3.19 by Martin Pool
SilentUIFactory now always errors when asked for input
335
4449.3.36 by Martin Pool
Update tests: SilentUIFactory no longer does get_boolean or get_password
336
    def test_silent_factory_get_password(self):
337
        # A silent factory that can't do user interaction can't get a
338
        # password.  Possibly it should raise a more specific error but it
339
        # can't succeed.
4597.3.36 by Vincent Ladeuil
Fix imports and various cleanups in test_ui.
340
        ui = _mod_ui.SilentUIFactory()
341
        stdout = tests.StringIOWrapper()
4449.3.19 by Martin Pool
SilentUIFactory now always errors when asked for input
342
        self.assertRaises(
343
            NotImplementedError,
344
            self.apply_redirected,
345
            None, stdout, stdout, ui.get_password)
346
        # and it didn't write anything out either
347
        self.assertEqual('', stdout.getvalue())
348
349
    def test_silent_ui_getbool(self):
4597.3.36 by Vincent Ladeuil
Fix imports and various cleanups in test_ui.
350
        factory = _mod_ui.SilentUIFactory()
351
        stdout = tests.StringIOWrapper()
4449.3.19 by Martin Pool
SilentUIFactory now always errors when asked for input
352
        self.assertRaises(
353
            NotImplementedError,
354
            self.apply_redirected,
5863.6.1 by Jelmer Vernooij
Require a unicode prompt to be passed into all methods that prompt.
355
            None, stdout, stdout, factory.get_boolean, u"foo")
4449.3.42 by Martin Pool
Add basic test for CannedInputUIFactory
356
357
4597.3.36 by Vincent Ladeuil
Fix imports and various cleanups in test_ui.
358
class TestUIFactoryTests(tests.TestCase):
4580.2.2 by Martin Pool
Add test for bug 408201
359
360
    def test_test_ui_factory_progress(self):
361
        # 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
362
        # see https://bugs.launchpad.net/bzr/+bug/408201
4597.3.36 by Vincent Ladeuil
Fix imports and various cleanups in test_ui.
363
        ui = tests.TestUIFactory()
4580.2.2 by Martin Pool
Add test for bug 408201
364
        pb = ui.nested_progress_bar()
365
        pb.update('hello')
366
        pb.tick()
367
        pb.finished()
368
369
4597.3.36 by Vincent Ladeuil
Fix imports and various cleanups in test_ui.
370
class CannedInputUIFactoryTests(tests.TestCase):
371
4449.3.42 by Martin Pool
Add basic test for CannedInputUIFactory
372
    def test_canned_input_get_input(self):
4597.3.37 by Vincent Ladeuil
Allows ui factories to query users for an integer.
373
        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.
374
        self.assertEqual(True, uif.get_boolean(u'Extra cheese?'))
375
        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.
376
        self.assertEqual('password',
5863.6.1 by Jelmer Vernooij
Require a unicode prompt to be passed into all methods that prompt.
377
                         uif.get_password(u'Password for %(host)s',
4597.3.37 by Vincent Ladeuil
Allows ui factories to query users for an integer.
378
                                          host='example.com'))
5863.6.1 by Jelmer Vernooij
Require a unicode prompt to be passed into all methods that prompt.
379
        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
380
4503.2.1 by Vincent Ladeuil
Get a bool from a string.
381
382
class TestBoolFromString(tests.TestCase):
383
384
    def assertIsTrue(self, s, accepted_values=None):
385
        res = _mod_ui.bool_from_string(s, accepted_values=accepted_values)
386
        self.assertEquals(True, res)
387
388
    def assertIsFalse(self, s, accepted_values=None):
389
        res = _mod_ui.bool_from_string(s, accepted_values=accepted_values)
390
        self.assertEquals(False, res)
391
392
    def assertIsNone(self, s, accepted_values=None):
393
        res = _mod_ui.bool_from_string(s, accepted_values=accepted_values)
394
        self.assertIs(None, res)
395
396
    def test_know_valid_values(self):
397
        self.assertIsTrue('true')
398
        self.assertIsFalse('false')
399
        self.assertIsTrue('1')
400
        self.assertIsFalse('0')
401
        self.assertIsTrue('on')
402
        self.assertIsFalse('off')
403
        self.assertIsTrue('yes')
404
        self.assertIsFalse('no')
405
        self.assertIsTrue('y')
406
        self.assertIsFalse('n')
407
        # Also try some case variations
408
        self.assertIsTrue('True')
409
        self.assertIsFalse('False')
410
        self.assertIsTrue('On')
411
        self.assertIsFalse('Off')
412
        self.assertIsTrue('ON')
413
        self.assertIsFalse('OFF')
414
        self.assertIsTrue('oN')
415
        self.assertIsFalse('oFf')
416
417
    def test_invalid_values(self):
418
        self.assertIsNone(None)
419
        self.assertIsNone('doubt')
420
        self.assertIsNone('frue')
421
        self.assertIsNone('talse')
422
        self.assertIsNone('42')
423
424
    def test_provided_values(self):
425
        av = dict(y=True, n=False, yes=True, no=False)
426
        self.assertIsTrue('y', av)
427
        self.assertIsTrue('Y', av)
428
        self.assertIsTrue('Yes', av)
429
        self.assertIsFalse('n', av)
430
        self.assertIsFalse('N', av)
431
        self.assertIsFalse('No', av)
432
        self.assertIsNone('1', av)
433
        self.assertIsNone('0', av)
434
        self.assertIsNone('on', av)
435
        self.assertIsNone('off', av)
5422.1.1 by Martin Pool
Move CapturingUIFactory out of per_workingtree tests into somewhere reusable
436
437
5416.1.11 by Martin Pool
Add ConfirmationUserInterfacePolicy that lets specific confirmations be configured off.
438
class TestConfirmationUserInterfacePolicy(tests.TestCase):
439
440
    def test_confirm_action_default(self):
441
        base_ui = _mod_ui.NoninteractiveUIFactory()
442
        for answer in [True, False]:
443
            self.assertEquals(
444
                _mod_ui.ConfirmationUserInterfacePolicy(base_ui, answer, {})
445
                .confirm_action("Do something?",
446
                    "bzrlib.tests.do_something", {}),
447
                answer)
448
449
    def test_confirm_action_specific(self):
450
        base_ui = _mod_ui.NoninteractiveUIFactory()
451
        for default_answer in [True, False]:
452
            for specific_answer in [True, False]:
453
                for conf_id in ['given_id', 'other_id']:
454
                    wrapper = _mod_ui.ConfirmationUserInterfacePolicy(
455
                        base_ui, default_answer, dict(given_id=specific_answer))
456
                    result = wrapper.confirm_action("Do something?", conf_id, {})
457
                    if conf_id == 'given_id':
458
                        self.assertEquals(result, specific_answer)
459
                    else:
460
                        self.assertEquals(result, default_answer)
461
462
    def test_repr(self):
463
        base_ui = _mod_ui.NoninteractiveUIFactory()
464
        wrapper = _mod_ui.ConfirmationUserInterfacePolicy(
465
            base_ui, True, dict(a=2))
466
        self.assertThat(repr(wrapper),
467
            Equals("ConfirmationUserInterfacePolicy("
468
                "NoninteractiveUIFactory(), True, {'a': 2})"))
5416.2.4 by Martin Pool
resolve against trunk
469
470
5422.1.4 by Martin Pool
Rename CapturingUIFactory to ProgressRecordingUIFactory
471
class TestProgressRecordingUI(tests.TestCase):
5422.1.1 by Martin Pool
Move CapturingUIFactory out of per_workingtree tests into somewhere reusable
472
    """Test test-oriented UIFactory that records progress updates"""
473
474
    def test_nested_ignore_depth_beyond_one(self):
475
        # we only want to capture the first level out progress, not
476
        # want sub-components might do. So we have nested bars ignored.
5422.1.4 by Martin Pool
Rename CapturingUIFactory to ProgressRecordingUIFactory
477
        factory = ProgressRecordingUIFactory()
5422.1.1 by Martin Pool
Move CapturingUIFactory out of per_workingtree tests into somewhere reusable
478
        pb1 = factory.nested_progress_bar()
479
        pb1.update('foo', 0, 1)
480
        pb2 = factory.nested_progress_bar()
481
        pb2.update('foo', 0, 1)
482
        pb2.finished()
483
        pb1.finished()
484
        self.assertEqual([("update", 0, 1, 'foo')], factory._calls)