~bzr-pqm/bzr/bzr.dev

3948.2.6 by Martin Pool
ProgressBarStack is deprecated
1
# Copyright (C) 2005, 2008, 2009 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
15
# Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
16
17
"""Tests for the bzrlib ui
18
"""
19
20
import os
1534.5.6 by Robert Collins
split out converter logic into per-format objects.
21
from StringIO import StringIO
1704.2.9 by Martin Pool
Make text_factory test not depend on 80-col terminal
22
import re
1185.49.22 by John Arbash Meinel
Added get_password to the UIFactory, using it inside of sftp.py
23
import sys
4017.1.1 by John Arbash Meinel
Get a pb.tick() to work after calling pb.update()
24
import time
1185.49.22 by John Arbash Meinel
Added get_password to the UIFactory, using it inside of sftp.py
25
1681.1.2 by Robert Collins
* bzrlib.ui.text.TextUIFactory now accepts a bar_type parameter which
26
import bzrlib
1594.1.1 by Robert Collins
Introduce new bzr progress bar api. ui_factory.nested_progress_bar.
27
import bzrlib.errors as errors
2363.4.10 by Vincent Ladeuil
Complete tests.
28
from bzrlib.progress import (
29
    DotsProgressBar,
30
    ProgressBarStack,
31
    TTYProgressBar,
32
    )
3948.2.6 by Martin Pool
ProgressBarStack is deprecated
33
from bzrlib.symbol_versioning import (
34
    deprecated_in,
35
    )
2294.4.1 by Vincent Ladeuil
Add a UIFactory.get_login method, fix tests.
36
from bzrlib.tests import (
2363.4.10 by Vincent Ladeuil
Complete tests.
37
    TestCase,
2294.4.4 by Vincent Ladeuil
Provide a better implementation for testing passwords.
38
    TestUIFactory,
2294.4.1 by Vincent Ladeuil
Add a UIFactory.get_login method, fix tests.
39
    StringIOWrapper,
40
    )
1843.3.10 by John Arbash Meinel
ui tests were failing when output was redirected to a file. (thus blocked by the pqm)
41
from bzrlib.tests.test_progress import _TTYStringIO
2363.4.10 by Vincent Ladeuil
Complete tests.
42
from bzrlib.ui import (
43
    CLIUIFactory,
44
    SilentUIFactory,
45
    )
1185.49.22 by John Arbash Meinel
Added get_password to the UIFactory, using it inside of sftp.py
46
from bzrlib.ui.text import TextUIFactory
47
1681.1.2 by Robert Collins
* bzrlib.ui.text.TextUIFactory now accepts a bar_type parameter which
48
1185.49.22 by John Arbash Meinel
Added get_password to the UIFactory, using it inside of sftp.py
49
class UITests(TestCase):
50
51
    def test_silent_factory(self):
52
        ui = SilentUIFactory()
2294.4.1 by Vincent Ladeuil
Add a UIFactory.get_login method, fix tests.
53
        stdout = StringIO()
54
        self.assertEqual(None,
55
                         self.apply_redirected(None, stdout, stdout,
56
                                               ui.get_password))
57
        self.assertEqual('', stdout.getvalue())
58
        self.assertEqual(None,
59
                         self.apply_redirected(None, stdout, stdout,
60
                                               ui.get_password,
61
                                               u'Hello\u1234 %(user)s',
62
                                               user=u'some\u1234'))
63
        self.assertEqual('', stdout.getvalue())
64
65
    def test_text_factory_ascii_password(self):
2294.4.4 by Vincent Ladeuil
Provide a better implementation for testing passwords.
66
        ui = TestUIFactory(stdin='secret\n', stdout=StringIOWrapper())
2294.4.1 by Vincent Ladeuil
Add a UIFactory.get_login method, fix tests.
67
        pb = ui.nested_progress_bar()
68
        try:
69
            self.assertEqual('secret',
70
                             self.apply_redirected(ui.stdin, ui.stdout,
71
                                                   ui.stdout,
72
                                                   ui.get_password))
73
            # ': ' is appended to prompt
2294.4.4 by Vincent Ladeuil
Provide a better implementation for testing passwords.
74
            self.assertEqual(': ', ui.stdout.getvalue())
2363.4.3 by Vincent Ladeuil
Tidy-up tests.
75
            # stdin should be empty
2363.4.6 by Vincent Ladeuil
Fix tests around stdin emptyness.
76
            self.assertEqual('', ui.stdin.readline())
2294.4.1 by Vincent Ladeuil
Add a UIFactory.get_login method, fix tests.
77
        finally:
78
            pb.finished()
79
80
    def test_text_factory_utf8_password(self):
81
        """Test an utf8 password.
82
83
        We can't predict what encoding users will have for stdin, so we force
84
        it to utf8 to test that we transport the password correctly.
85
        """
2294.4.4 by Vincent Ladeuil
Provide a better implementation for testing passwords.
86
        ui = TestUIFactory(stdin=u'baz\u1234'.encode('utf8'),
87
                           stdout=StringIOWrapper())
88
        ui.stdin.encoding = 'utf8'
89
        ui.stdout.encoding = ui.stdin.encoding
2294.4.1 by Vincent Ladeuil
Add a UIFactory.get_login method, fix tests.
90
        pb = ui.nested_progress_bar()
91
        try:
92
            password = self.apply_redirected(ui.stdin, ui.stdout, ui.stdout,
93
                                             ui.get_password,
94
                                             u'Hello \u1234 %(user)s',
95
                                             user=u'some\u1234')
96
            # We use StringIO objects, we need to decode them
97
            self.assertEqual(u'baz\u1234', password.decode('utf8'))
98
            self.assertEqual(u'Hello \u1234 some\u1234: ',
99
                             ui.stdout.getvalue().decode('utf8'))
2363.4.3 by Vincent Ladeuil
Tidy-up tests.
100
            # stdin should be empty
2363.4.6 by Vincent Ladeuil
Fix tests around stdin emptyness.
101
            self.assertEqual('', ui.stdin.readline())
2294.4.1 by Vincent Ladeuil
Add a UIFactory.get_login method, fix tests.
102
        finally:
103
            pb.finished()
1534.5.6 by Robert Collins
split out converter logic into per-format objects.
104
105
    def test_progress_note(self):
106
        stderr = StringIO()
107
        stdout = StringIO()
3882.8.11 by Martin Pool
Choose the UIFactory class depending on the terminal capabilities
108
        ui_factory = TextUIFactory(stdin=StringIO(''),
109
            stderr=stderr,
110
            stdout=stdout)
1558.8.5 by Aaron Bentley
Pass note up the stack instead of using bzrlib.ui_factory
111
        pb = ui_factory.nested_progress_bar()
1558.8.4 by Aaron Bentley
Fixed test case for pb.note
112
        try:
113
            result = pb.note('t')
114
            self.assertEqual(None, result)
115
            self.assertEqual("t\n", stdout.getvalue())
1843.3.2 by John Arbash Meinel
Fix a ui test that depended on clearing
116
            # Since there was no update() call, there should be no clear() call
2363.4.4 by Vincent Ladeuil
More tidying-up.
117
            self.failIf(re.search(r'^\r {10,}\r$',
118
                                  stderr.getvalue()) is not None,
1843.3.2 by John Arbash Meinel
Fix a ui test that depended on clearing
119
                        'We cleared the stderr without anything to put there')
120
        finally:
121
            pb.finished()
122
123
    def test_progress_note_clears(self):
124
        stderr = StringIO()
125
        stdout = StringIO()
1843.3.10 by John Arbash Meinel
ui tests were failing when output was redirected to a file. (thus blocked by the pqm)
126
        # The PQM redirects the output to a file, so it
127
        # defaults to creating a Dots progress bar. we
128
        # need to force it to believe we are a TTY
3882.8.8 by Martin Pool
Progress and UI test cleanups
129
        ui_factory = TextUIFactory(
3882.8.11 by Martin Pool
Choose the UIFactory class depending on the terminal capabilities
130
            stdin=StringIO(''),
3882.8.4 by Martin Pool
All UI factories should support note()
131
            stdout=stdout, stderr=stderr)
1843.3.2 by John Arbash Meinel
Fix a ui test that depended on clearing
132
        pb = ui_factory.nested_progress_bar()
133
        try:
134
            # Create a progress update that isn't throttled
135
            pb.update('x', 1, 1)
136
            result = pb.note('t')
137
            self.assertEqual(None, result)
138
            self.assertEqual("t\n", stdout.getvalue())
1558.8.4 by Aaron Bentley
Fixed test case for pb.note
139
            # the exact contents will depend on the terminal width and we don't
140
            # care about that right now - but you're probably running it on at
141
            # least a 10-character wide terminal :)
1843.3.2 by John Arbash Meinel
Fix a ui test that depended on clearing
142
            self.assertContainsRe(stderr.getvalue(), r'\r {10,}\r$')
1558.8.4 by Aaron Bentley
Fixed test case for pb.note
143
        finally:
1558.8.5 by Aaron Bentley
Pass note up the stack instead of using bzrlib.ui_factory
144
            pb.finished()
1594.1.1 by Robert Collins
Introduce new bzr progress bar api. ui_factory.nested_progress_bar.
145
146
    def test_progress_nested(self):
147
        # test factory based nested and popping.
3882.8.11 by Martin Pool
Choose the UIFactory class depending on the terminal capabilities
148
        ui = TextUIFactory(None, None, None)
1594.1.1 by Robert Collins
Introduce new bzr progress bar api. ui_factory.nested_progress_bar.
149
        pb1 = ui.nested_progress_bar()
150
        pb2 = ui.nested_progress_bar()
3948.2.2 by Martin Pool
Corrections to finishing progress bars
151
        # You do get a warning if the outermost progress bar wasn't finished
152
        # first - it's not clear if this is really useful or if it should just
153
        # become orphaned -- mbp 20090120
3882.8.12 by Martin Pool
Give a warning, not an error, if a progress bar is not finished in order
154
        warnings, _ = self.callCatchWarnings(pb1.finished)
3948.2.2 by Martin Pool
Corrections to finishing progress bars
155
        if len(warnings) != 1:
156
            self.fail("unexpected warnings: %r" % (warnings,))
1594.1.1 by Robert Collins
Introduce new bzr progress bar api. ui_factory.nested_progress_bar.
157
        pb2.finished()
158
        pb1.finished()
159
160
    def test_progress_stack(self):
3943.8.1 by Marius Kruger
remove all trailing whitespace from bzr source
161
        # test the progress bar stack which the default text factory
1594.1.1 by Robert Collins
Introduce new bzr progress bar api. ui_factory.nested_progress_bar.
162
        # uses.
163
        stderr = StringIO()
164
        stdout = StringIO()
165
        # make a stack, which accepts parameters like a pb.
3948.2.6 by Martin Pool
ProgressBarStack is deprecated
166
        stack = self.applyDeprecated(
167
            deprecated_in((1, 12, 0)),
168
            ProgressBarStack,
169
            to_file=stderr, to_messages_file=stdout)
1594.1.1 by Robert Collins
Introduce new bzr progress bar api. ui_factory.nested_progress_bar.
170
        # but is not one
171
        self.assertFalse(getattr(stack, 'note', False))
172
        pb1 = stack.get_nested()
173
        pb2 = stack.get_nested()
3882.8.12 by Martin Pool
Give a warning, not an error, if a progress bar is not finished in order
174
        warnings, _ = self.callCatchWarnings(pb1.finished)
175
        self.assertEqual(len(warnings), 1)
1594.1.1 by Robert Collins
Introduce new bzr progress bar api. ui_factory.nested_progress_bar.
176
        pb2.finished()
177
        pb1.finished()
178
        # the text ui factory never actually removes the stack once its setup.
179
        # we need to be able to nest again correctly from here.
180
        pb1 = stack.get_nested()
181
        pb2 = stack.get_nested()
3882.8.12 by Martin Pool
Give a warning, not an error, if a progress bar is not finished in order
182
        warnings, _ = self.callCatchWarnings(pb1.finished)
183
        self.assertEqual(len(warnings), 1)
1594.1.1 by Robert Collins
Introduce new bzr progress bar api. ui_factory.nested_progress_bar.
184
        pb2.finished()
185
        pb1.finished()
1681.1.2 by Robert Collins
* bzrlib.ui.text.TextUIFactory now accepts a bar_type parameter which
186
1687.1.4 by Robert Collins
Add bzrlib.ui.ui_factory.get_boolean().
187
    def assert_get_bool_acceptance_of_user_input(self, factory):
2363.4.4 by Vincent Ladeuil
More tidying-up.
188
        factory.stdin = StringIO("y\nyes with garbage\n"
189
                                 "yes\nn\nnot an answer\n"
190
                                 "no\nfoo\n")
1687.1.4 by Robert Collins
Add bzrlib.ui.ui_factory.get_boolean().
191
        factory.stdout = StringIO()
192
        # there is no output from the base factory
193
        self.assertEqual(True, factory.get_boolean(""))
194
        self.assertEqual(True, factory.get_boolean(""))
195
        self.assertEqual(False, factory.get_boolean(""))
196
        self.assertEqual(False, factory.get_boolean(""))
197
        self.assertEqual("foo\n", factory.stdin.read())
2363.4.4 by Vincent Ladeuil
More tidying-up.
198
        # stdin should be empty
2363.4.6 by Vincent Ladeuil
Fix tests around stdin emptyness.
199
        self.assertEqual('', factory.stdin.readline())
1687.1.4 by Robert Collins
Add bzrlib.ui.ui_factory.get_boolean().
200
201
    def test_silent_ui_getbool(self):
2363.4.10 by Vincent Ladeuil
Complete tests.
202
        factory = SilentUIFactory()
1687.1.4 by Robert Collins
Add bzrlib.ui.ui_factory.get_boolean().
203
        self.assert_get_bool_acceptance_of_user_input(factory)
204
205
    def test_silent_factory_prompts_silently(self):
2363.4.10 by Vincent Ladeuil
Complete tests.
206
        factory = SilentUIFactory()
1687.1.4 by Robert Collins
Add bzrlib.ui.ui_factory.get_boolean().
207
        stdout = StringIO()
208
        factory.stdin = StringIO("y\n")
2363.4.4 by Vincent Ladeuil
More tidying-up.
209
        self.assertEqual(True,
210
                         self.apply_redirected(None, stdout, stdout,
211
                                               factory.get_boolean, "foo"))
1687.1.4 by Robert Collins
Add bzrlib.ui.ui_factory.get_boolean().
212
        self.assertEqual("", stdout.getvalue())
2363.4.4 by Vincent Ladeuil
More tidying-up.
213
        # stdin should be empty
2363.4.6 by Vincent Ladeuil
Fix tests around stdin emptyness.
214
        self.assertEqual('', factory.stdin.readline())
2363.4.4 by Vincent Ladeuil
More tidying-up.
215
1687.1.4 by Robert Collins
Add bzrlib.ui.ui_factory.get_boolean().
216
    def test_text_ui_getbool(self):
3882.8.11 by Martin Pool
Choose the UIFactory class depending on the terminal capabilities
217
        factory = TextUIFactory(None, None, None)
1687.1.4 by Robert Collins
Add bzrlib.ui.ui_factory.get_boolean().
218
        self.assert_get_bool_acceptance_of_user_input(factory)
219
220
    def test_text_factory_prompts_and_clears(self):
221
        # a get_boolean call should clear the pb before prompting
3882.8.10 by Martin Pool
Fix up test_ui for new progress bars
222
        out = _TTYStringIO()
3882.8.11 by Martin Pool
Choose the UIFactory class depending on the terminal capabilities
223
        factory = TextUIFactory(stdin=StringIO("yada\ny\n"), stdout=out, stderr=out)
3882.8.10 by Martin Pool
Fix up test_ui for new progress bars
224
        pb = factory.nested_progress_bar()
225
        pb.show_bar = False
226
        pb.show_spinner = False
227
        pb.show_count = False
228
        pb.update("foo", 0, 1)
2363.4.4 by Vincent Ladeuil
More tidying-up.
229
        self.assertEqual(True,
230
                         self.apply_redirected(None, factory.stdout,
231
                                               factory.stdout,
232
                                               factory.get_boolean,
233
                                               "what do you want"))
3882.8.10 by Martin Pool
Fix up test_ui for new progress bars
234
        output = out.getvalue()
235
        self.assertContainsRe(factory.stdout.getvalue(),
236
            "foo *\r\r  *\r*")
237
        self.assertContainsRe(factory.stdout.getvalue(),
238
            r"what do you want\? \[y/n\]: what do you want\? \[y/n\]: ")
239
        # stdin should have been totally consumed
2363.4.6 by Vincent Ladeuil
Fix tests around stdin emptyness.
240
        self.assertEqual('', factory.stdin.readline())
4017.1.1 by John Arbash Meinel
Get a pb.tick() to work after calling pb.update()
241
242
    def test_text_tick_after_update(self):
243
        ui_factory = TextUIFactory(stdout=StringIO(), stderr=StringIO())
244
        pb = ui_factory.nested_progress_bar()
245
        try:
246
            pb.update('task', 0, 3)
247
            # Reset the clock, so that it actually tries to repaint itself
248
            ui_factory._progress_view._last_repaint = time.time() - 1.0
249
            pb.tick()
250
        finally:
251
            pb.finished()