/brz/remove-bazaar

To get this branch, use:
bzr branch http://gegoxaren.bato24.eu/bzr/brz/remove-bazaar
3260.2.1 by Alexander Belchenko
Don't ask a password if there is no real terminal. (#69851)
1
# Copyright (C) 2005, 2006, 2007, 2008 Canonical Ltd
1887.1.1 by Adeodato Simó
Do not separate paragraphs in the copyright statement with blank lines,
2
#
1185.1.29 by Robert Collins
merge merge tweaks from aaron, which includes latest .dev
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.
1887.1.1 by Adeodato Simó
Do not separate paragraphs in the copyright statement with blank lines,
7
#
1185.1.29 by Robert Collins
merge merge tweaks from aaron, which includes latest .dev
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.
1887.1.1 by Adeodato Simó
Do not separate paragraphs in the copyright statement with blank lines,
12
#
1185.1.29 by Robert Collins
merge merge tweaks from aaron, which includes latest .dev
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
"""UI abstraction.
18
19
This tells the library how to display things to the user.  Through this
20
layer different applications can choose the style of UI.
21
22
At the moment this layer is almost trivial: the application can just
23
choose the style of progress bar.
24
25
Set the ui_factory member to define the behaviour.  The default
26
displays no output.
27
"""
28
3882.8.11 by Martin Pool
Choose the UIFactory class depending on the terminal capabilities
29
import os
1687.1.4 by Robert Collins
Add bzrlib.ui.ui_factory.get_boolean().
30
import sys
3882.8.12 by Martin Pool
Give a warning, not an error, if a progress bar is not finished in order
31
import warnings
1594.1.1 by Robert Collins
Introduce new bzr progress bar api. ui_factory.nested_progress_bar.
32
1996.3.32 by John Arbash Meinel
from bzrlib.ui lazy import progress, and make progress import lazily
33
from bzrlib.lazy_import import lazy_import
34
lazy_import(globals(), """
2294.4.4 by Vincent Ladeuil
Provide a better implementation for testing passwords.
35
import getpass
36
1996.3.32 by John Arbash Meinel
from bzrlib.ui lazy import progress, and make progress import lazily
37
from bzrlib import (
3260.2.1 by Alexander Belchenko
Don't ask a password if there is no real terminal. (#69851)
38
    errors,
2461.1.2 by Vincent Ladeuil
Take jam's remark into account.
39
    osutils,
1996.3.32 by John Arbash Meinel
from bzrlib.ui lazy import progress, and make progress import lazily
40
    progress,
2323.6.2 by Martin Pool
Move responsibility for suggesting upgrades to ui object
41
    trace,
1996.3.32 by John Arbash Meinel
from bzrlib.ui lazy import progress, and make progress import lazily
42
    )
43
""")
44
1185.1.29 by Robert Collins
merge merge tweaks from aaron, which includes latest .dev
45
1185.49.21 by John Arbash Meinel
Refactored bzrlib/ui.py into a module with the possibility for multiple ui forms.
46
class UIFactory(object):
47
    """UI abstraction.
48
49
    This tells the library how to display things to the user.  Through this
50
    layer different applications can choose the style of UI.
51
    """
1594.1.1 by Robert Collins
Introduce new bzr progress bar api. ui_factory.nested_progress_bar.
52
1594.1.3 by Robert Collins
Fixup pb usage to use nested_progress_bar.
53
    def __init__(self):
3882.7.7 by Martin Pool
Change progress bars to a more MVC style
54
        self._task_stack = []
1594.1.3 by Robert Collins
Fixup pb usage to use nested_progress_bar.
55
1185.49.22 by John Arbash Meinel
Added get_password to the UIFactory, using it inside of sftp.py
56
    def get_password(self, prompt='', **kwargs):
57
        """Prompt the user for a password.
58
59
        :param prompt: The prompt to present the user
60
        :param kwargs: Arguments which will be expanded into the prompt.
61
                       This lets front ends display different things if
62
                       they so choose.
2294.4.1 by Vincent Ladeuil
Add a UIFactory.get_login method, fix tests.
63
64
        :return: The password string, return None if the user canceled the
65
                 request. Note that we do not touch the encoding, users may
66
                 have whatever they see fit and the password should be
67
                 transported as is.
1185.49.22 by John Arbash Meinel
Added get_password to the UIFactory, using it inside of sftp.py
68
        """
1594.1.1 by Robert Collins
Introduce new bzr progress bar api. ui_factory.nested_progress_bar.
69
        raise NotImplementedError(self.get_password)
2294.4.1 by Vincent Ladeuil
Add a UIFactory.get_login method, fix tests.
70
1594.1.1 by Robert Collins
Introduce new bzr progress bar api. ui_factory.nested_progress_bar.
71
    def nested_progress_bar(self):
72
        """Return a nested progress bar.
73
2095.4.5 by mbp at sourcefrog
Use regular progress-bar classes, not a special mechanism
74
        When the bar has been finished with, it should be released by calling
1594.1.1 by Robert Collins
Introduce new bzr progress bar api. ui_factory.nested_progress_bar.
75
        bar.finished().
76
        """
3882.7.7 by Martin Pool
Change progress bars to a more MVC style
77
        if self._task_stack:
3882.8.3 by Martin Pool
Move display of transport throughput into TextProgressView
78
            t = progress.ProgressTask(self._task_stack[-1], self)
3882.7.7 by Martin Pool
Change progress bars to a more MVC style
79
        else:
3882.8.3 by Martin Pool
Move display of transport throughput into TextProgressView
80
            t = progress.ProgressTask(None, self)
3882.7.7 by Martin Pool
Change progress bars to a more MVC style
81
        self._task_stack.append(t)
82
        return t
83
84
    def progress_finished(self, task):
85
        if task != self._task_stack[-1]:
3882.8.12 by Martin Pool
Give a warning, not an error, if a progress bar is not finished in order
86
            warnings.warn("%r is not currently active" % (task,))
87
        else:
88
            del self._task_stack[-1]
1594.1.1 by Robert Collins
Introduce new bzr progress bar api. ui_factory.nested_progress_bar.
89
1558.8.1 by Aaron Bentley
Fix overall progress bar's interaction with 'note' and 'warning'
90
    def clear_term(self):
91
        """Prepare the terminal for output.
92
93
        This will, for example, clear text progress bars, and leave the
94
        cursor at the leftmost position."""
95
        raise NotImplementedError(self.clear_term)
96
1687.1.4 by Robert Collins
Add bzrlib.ui.ui_factory.get_boolean().
97
    def get_boolean(self, prompt):
98
        """Get a boolean question answered from the user. 
99
100
        :param prompt: a message to prompt the user with. Should be a single
101
        line without terminating \n.
102
        :return: True or False for y/yes or n/no.
103
        """
104
        raise NotImplementedError(self.get_boolean)
105
2323.6.2 by Martin Pool
Move responsibility for suggesting upgrades to ui object
106
    def recommend_upgrade(self,
107
        current_format_name,
108
        basedir):
2323.6.4 by Martin Pool
BzrDir._check_supported now also takes care of recommending upgrades, which
109
        # this should perhaps be in the TextUIFactory and the default can do
110
        # nothing
2323.6.2 by Martin Pool
Move responsibility for suggesting upgrades to ui object
111
        trace.warning("%s is deprecated "
112
            "and a better format is available.\n"
113
            "It is recommended that you upgrade by "
114
            "running the command\n"
115
            "  bzr upgrade %s",
116
            current_format_name,
117
            basedir)
2323.6.4 by Martin Pool
BzrDir._check_supported now also takes care of recommending upgrades, which
118
3882.7.5 by Martin Pool
Further mockup of transport-based activity indicator.
119
    def report_transport_activity(self, transport, byte_count, direction):
120
        """Called by transports as they do IO.
121
        
122
        This may update a progress bar, spinner, or similar display.
123
        By default it does nothing.
124
        """
125
        pass
126
127
2461.1.1 by Vincent Ladeuil
Fix 110204 by letting TestUIFactory encode password prompt.
128
1687.1.4 by Robert Collins
Add bzrlib.ui.ui_factory.get_boolean().
129
class CLIUIFactory(UIFactory):
3882.8.11 by Martin Pool
Choose the UIFactory class depending on the terminal capabilities
130
    """Common behaviour for command line UI factories.
131
    
132
    This is suitable for dumb terminals that can't repaint existing text."""
1687.1.4 by Robert Collins
Add bzrlib.ui.ui_factory.get_boolean().
133
3882.8.11 by Martin Pool
Choose the UIFactory class depending on the terminal capabilities
134
    def __init__(self, stdin=None, stdout=None, stderr=None):
135
        UIFactory.__init__(self)
136
        self.stdin = stdin or sys.stdin
137
        self.stdout = stdout or sys.stdout
138
        self.stderr = stderr or sys.stderr
1687.1.4 by Robert Collins
Add bzrlib.ui.ui_factory.get_boolean().
139
140
    def get_boolean(self, prompt):
141
        self.clear_term()
142
        # FIXME: make a regexp and handle case variations as well.
143
        while True:
2294.4.1 by Vincent Ladeuil
Add a UIFactory.get_login method, fix tests.
144
            self.prompt(prompt + "? [y/n]: ")
1687.1.4 by Robert Collins
Add bzrlib.ui.ui_factory.get_boolean().
145
            line = self.stdin.readline()
146
            if line in ('y\n', 'yes\n'):
147
                return True
148
            if line in ('n\n', 'no\n'):
149
                return False
150
2294.4.4 by Vincent Ladeuil
Provide a better implementation for testing passwords.
151
    def get_non_echoed_password(self, prompt):
3260.2.1 by Alexander Belchenko
Don't ask a password if there is no real terminal. (#69851)
152
        if not sys.stdin.isatty():
153
            raise errors.NotATerminal()
2461.1.2 by Vincent Ladeuil
Take jam's remark into account.
154
        encoding = osutils.get_terminal_encoding()
155
        return getpass.getpass(prompt.encode(encoding, 'replace'))
2294.4.4 by Vincent Ladeuil
Provide a better implementation for testing passwords.
156
157
    def get_password(self, prompt='', **kwargs):
158
        """Prompt the user for a password.
159
160
        :param prompt: The prompt to present the user
161
        :param kwargs: Arguments which will be expanded into the prompt.
162
                       This lets front ends display different things if
163
                       they so choose.
164
        :return: The password string, return None if the user 
165
                 canceled the request.
166
        """
167
        prompt += ': '
2461.1.1 by Vincent Ladeuil
Fix 110204 by letting TestUIFactory encode password prompt.
168
        prompt = (prompt % kwargs)
2294.4.4 by Vincent Ladeuil
Provide a better implementation for testing passwords.
169
        # There's currently no way to say 'i decline to enter a password'
170
        # as opposed to 'my password is empty' -- does it matter?
171
        return self.get_non_echoed_password(prompt)
172
1687.1.4 by Robert Collins
Add bzrlib.ui.ui_factory.get_boolean().
173
    def prompt(self, prompt):
174
        """Emit prompt on the CLI."""
3945.1.1 by Vincent Ladeuil
Restore a working UI implementation suitable for emacs shells.
175
        self.stdout.write(prompt)
176
177
    def note(self, msg):
178
        """Write an already-formatted message."""
179
        self.stdout.write(msg + '\n')
1687.1.4 by Robert Collins
Add bzrlib.ui.ui_factory.get_boolean().
180
3882.7.10 by Martin Pool
CLIUIFactory implements (as stubs) progress methods
181
    def clear_term(self):
182
        pass
183
184
    def show_progress(self, task):
185
        pass
186
187
    def progress_finished(self, task):
188
        pass
189
1687.1.4 by Robert Collins
Add bzrlib.ui.ui_factory.get_boolean().
190
191
class SilentUIFactory(CLIUIFactory):
1185.49.21 by John Arbash Meinel
Refactored bzrlib/ui.py into a module with the possibility for multiple ui forms.
192
    """A UI Factory which never prints anything.
193
194
    This is the default UI, if another one is never registered.
195
    """
1594.1.1 by Robert Collins
Introduce new bzr progress bar api. ui_factory.nested_progress_bar.
196
3882.8.11 by Martin Pool
Choose the UIFactory class depending on the terminal capabilities
197
    def __init__(self):
198
        CLIUIFactory.__init__(self)
199
1185.49.22 by John Arbash Meinel
Added get_password to the UIFactory, using it inside of sftp.py
200
    def get_password(self, prompt='', **kwargs):
201
        return None
202
3945.1.1 by Vincent Ladeuil
Restore a working UI implementation suitable for emacs shells.
203
    def prompt(self, prompt):
204
        pass
1558.8.1 by Aaron Bentley
Fix overall progress bar's interaction with 'note' and 'warning'
205
3882.8.4 by Martin Pool
All UI factories should support note()
206
    def note(self, msg):
207
        pass
208
3882.8.2 by Martin Pool
ProgressTask holds a reference to the ui that displays it
209
1558.8.1 by Aaron Bentley
Fix overall progress bar's interaction with 'note' and 'warning'
210
def clear_decorator(func, *args, **kwargs):
211
    """Decorator that clears the term"""
212
    ui_factory.clear_term()
213
    func(*args, **kwargs)
214
1185.49.22 by John Arbash Meinel
Added get_password to the UIFactory, using it inside of sftp.py
215
1185.1.29 by Robert Collins
merge merge tweaks from aaron, which includes latest .dev
216
ui_factory = SilentUIFactory()
1534.5.9 by Robert Collins
Advise users running upgrade on a checkout to also run it on the branch.
217
"""IMPORTANT: never import this symbol directly. ONLY ever access it as 
218
ui.ui_factory."""
3882.8.11 by Martin Pool
Choose the UIFactory class depending on the terminal capabilities
219
220
221
def make_ui_for_terminal(stdin, stdout, stderr):
222
    """Construct and return a suitable UIFactory for a text mode program.
223
224
    If stdout is a smart terminal, this gets a smart UIFactory with 
225
    progress indicators, etc.  If it's a dumb terminal, just plain text output.
226
    """
3945.1.3 by Vincent Ladeuil
Restore line eaten by a gremlin.
227
    cls = None
3882.8.11 by Martin Pool
Choose the UIFactory class depending on the terminal capabilities
228
    isatty = getattr(stdin, 'isatty', None)
229
    if isatty is None:
230
        cls = CLIUIFactory
231
    elif not isatty():
232
        cls = CLIUIFactory
233
    elif os.environ.get('TERM') in (None, 'dumb', ''):
234
        # e.g. emacs compile window
235
        cls = CLIUIFactory
3945.1.1 by Vincent Ladeuil
Restore a working UI implementation suitable for emacs shells.
236
    # User may know better, otherwise default to TextUIFactory
237
    if (   os.environ.get('BZR_USE_TEXT_UI', None) is not None
238
        or cls is None):
3882.8.11 by Martin Pool
Choose the UIFactory class depending on the terminal capabilities
239
        from bzrlib.ui.text import TextUIFactory
240
        cls = TextUIFactory
241
    return cls(stdin=stdin, stdout=stdout, stderr=stderr)