~bzr-pqm/bzr/bzr.dev

« back to all changes in this revision

Viewing changes to bzrlib/ui/text.py

  • Committer: Canonical.com Patch Queue Manager
  • Date: 2010-07-29 10:44:42 UTC
  • mfrom: (5361.1.1 2.3-lp-home)
  • Revision ID: pqm@pqm.ubuntu.com-20100729104442-5g1m4pumcss037ic
(spiv) Expand lp:~/ to lp:~username/ if a user has already logged in. (John
 A Meinel)

Show diffs side-by-side

added added

removed removed

Lines of Context:
1
 
# Copyright (C) 2005-2011 Canonical Ltd
 
1
# Copyright (C) 2005-2010 Canonical Ltd
2
2
#
3
3
# This program is free software; you can redistribute it and/or modify
4
4
# it under the terms of the GNU General Public License as published by
18
18
"""Text UI, write output to the console.
19
19
"""
20
20
 
 
21
import codecs
 
22
import getpass
21
23
import os
22
24
import sys
23
25
import time
 
26
import warnings
24
27
 
25
28
from bzrlib.lazy_import import lazy_import
26
29
lazy_import(globals(), """
27
 
import codecs
28
 
import getpass
29
 
import warnings
30
 
 
31
30
from bzrlib import (
32
31
    debug,
33
32
    progress,
34
33
    osutils,
 
34
    symbol_versioning,
35
35
    trace,
36
36
    )
37
37
 
43
43
    )
44
44
 
45
45
 
46
 
class _ChooseUI(object):
47
 
 
48
 
    """ Helper class for choose implementation.
49
 
    """
50
 
 
51
 
    def __init__(self, ui, msg, choices, default):
52
 
        self.ui = ui
53
 
        self._setup_mode()
54
 
        self._build_alternatives(msg, choices, default)
55
 
 
56
 
    def _setup_mode(self):
57
 
        """Setup input mode (line-based, char-based) and echo-back.
58
 
 
59
 
        Line-based input is used if the BZR_TEXTUI_INPUT environment
60
 
        variable is set to 'line-based', or if there is no controlling
61
 
        terminal.
62
 
        """
63
 
        if os.environ.get('BZR_TEXTUI_INPUT') != 'line-based' and \
64
 
           self.ui.stdin == sys.stdin and self.ui.stdin.isatty():
65
 
            self.line_based = False
66
 
            self.echo_back = True
67
 
        else:
68
 
            self.line_based = True
69
 
            self.echo_back = not self.ui.stdin.isatty()
70
 
 
71
 
    def _build_alternatives(self, msg, choices, default):
72
 
        """Parse choices string.
73
 
 
74
 
        Setup final prompt and the lists of choices and associated
75
 
        shortcuts.
76
 
        """
77
 
        index = 0
78
 
        help_list = []
79
 
        self.alternatives = {}
80
 
        choices = choices.split('\n')
81
 
        if default is not None and default not in range(0, len(choices)):
82
 
            raise ValueError("invalid default index")
83
 
        for c in choices:
84
 
            name = c.replace('&', '').lower()
85
 
            choice = (name, index)
86
 
            if name in self.alternatives:
87
 
                raise ValueError("duplicated choice: %s" % name)
88
 
            self.alternatives[name] = choice
89
 
            shortcut = c.find('&')
90
 
            if -1 != shortcut and (shortcut + 1) < len(c):
91
 
                help = c[:shortcut]
92
 
                help += '[' + c[shortcut + 1] + ']'
93
 
                help += c[(shortcut + 2):]
94
 
                shortcut = c[shortcut + 1]
95
 
            else:
96
 
                c = c.replace('&', '')
97
 
                shortcut = c[0]
98
 
                help = '[%s]%s' % (shortcut, c[1:])
99
 
            shortcut = shortcut.lower()
100
 
            if shortcut in self.alternatives:
101
 
                raise ValueError("duplicated shortcut: %s" % shortcut)
102
 
            self.alternatives[shortcut] = choice
103
 
            # Add redirections for default.
104
 
            if index == default:
105
 
                self.alternatives[''] = choice
106
 
                self.alternatives['\r'] = choice
107
 
            help_list.append(help)
108
 
            index += 1
109
 
 
110
 
        self.prompt = u'%s (%s): ' % (msg, ', '.join(help_list))
111
 
 
112
 
    def _getline(self):
113
 
        line = self.ui.stdin.readline()
114
 
        if '' == line:
115
 
            raise EOFError
116
 
        return line.strip()
117
 
 
118
 
    def _getchar(self):
119
 
        char = osutils.getchar()
120
 
        if char == chr(3): # INTR
121
 
            raise KeyboardInterrupt
122
 
        if char == chr(4): # EOF (^d, C-d)
123
 
            raise EOFError
124
 
        return char
125
 
 
126
 
    def interact(self):
127
 
        """Keep asking the user until a valid choice is made.
128
 
        """
129
 
        if self.line_based:
130
 
            getchoice = self._getline
131
 
        else:
132
 
            getchoice = self._getchar
133
 
        iter = 0
134
 
        while True:
135
 
            iter += 1
136
 
            if 1 == iter or self.line_based:
137
 
                self.ui.prompt(self.prompt)
138
 
            try:
139
 
                choice = getchoice()
140
 
            except EOFError:
141
 
                self.ui.stderr.write('\n')
142
 
                return None
143
 
            except KeyboardInterrupt:
144
 
                self.ui.stderr.write('\n')
145
 
                raise KeyboardInterrupt
146
 
            choice = choice.lower()
147
 
            if choice not in self.alternatives:
148
 
                # Not a valid choice, keep on asking.
149
 
                continue
150
 
            name, index = self.alternatives[choice]
151
 
            if self.echo_back:
152
 
                self.ui.stderr.write(name + '\n')
153
 
            return index
154
 
 
155
 
 
156
46
class TextUIFactory(UIFactory):
157
47
    """A UI factory for Text user interefaces."""
158
48
 
171
61
        # paints progress, network activity, etc
172
62
        self._progress_view = self.make_progress_view()
173
63
 
174
 
    def choose(self, msg, choices, default=None):
175
 
        """Prompt the user for a list of alternatives.
176
 
 
177
 
        Support both line-based and char-based editing.
178
 
 
179
 
        In line-based mode, both the shortcut and full choice name are valid
180
 
        answers, e.g. for choose('prompt', '&yes\n&no'): 'y', ' Y ', ' yes',
181
 
        'YES ' are all valid input lines for choosing 'yes'.
182
 
 
183
 
        An empty line, when in line-based mode, or pressing enter in char-based
184
 
        mode will select the default choice (if any).
185
 
 
186
 
        Choice is echoed back if:
187
 
        - input is char-based; which means a controlling terminal is available,
188
 
          and osutils.getchar is used
189
 
        - input is line-based, and no controlling terminal is available
190
 
        """
191
 
 
192
 
        choose_ui = _ChooseUI(self, msg, choices, default)
193
 
        return choose_ui.interact()
194
 
 
195
64
    def be_quiet(self, state):
196
65
        if state and not self._quiet:
197
66
            self.clear_term()
209
78
        # to clear it.  We might need to separately check for the case of
210
79
        self._progress_view.clear()
211
80
 
 
81
    def get_boolean(self, prompt):
 
82
        while True:
 
83
            self.prompt(prompt + "? [y/n]: ")
 
84
            line = self.stdin.readline().lower()
 
85
            if line in ('y\n', 'yes\n'):
 
86
                return True
 
87
            elif line in ('n\n', 'no\n'):
 
88
                return False
 
89
            elif line in ('', None):
 
90
                # end-of-file; possibly should raise an error here instead
 
91
                return None
 
92
 
212
93
    def get_integer(self, prompt):
213
94
        while True:
214
95
            self.prompt(prompt)
233
114
                password = password[:-1]
234
115
        return password
235
116
 
236
 
    def get_password(self, prompt=u'', **kwargs):
 
117
    def get_password(self, prompt='', **kwargs):
237
118
        """Prompt the user for a password.
238
119
 
239
120
        :param prompt: The prompt to present the user
317
198
        :param kwargs: Dictionary of arguments to insert into the prompt,
318
199
            to allow UIs to reformat the prompt.
319
200
        """
320
 
        if type(prompt) != unicode:
321
 
            raise ValueError("prompt %r not a unicode string" % prompt)
322
201
        if kwargs:
323
202
            # See <https://launchpad.net/bugs/365891>
324
203
            prompt = prompt % kwargs
325
204
        prompt = prompt.encode(osutils.get_terminal_encoding(), 'replace')
326
205
        self.clear_term()
327
 
        self.stdout.flush()
328
206
        self.stderr.write(prompt)
329
207
 
330
208
    def report_transport_activity(self, transport, byte_count, direction):
422
300
        # correspond reliably to overall command progress
423
301
        self.enable_bar = False
424
302
 
425
 
    def _avail_width(self):
426
 
        # we need one extra space for terminals that wrap on last char
427
 
        w = osutils.terminal_width() 
428
 
        if w is None:
429
 
            return None
430
 
        else:
431
 
            return w - 1
432
 
 
433
303
    def _show_line(self, s):
434
304
        # sys.stderr.write("progress %r\n" % s)
435
 
        width = self._avail_width()
 
305
        width = osutils.terminal_width()
436
306
        if width is not None:
 
307
            # we need one extra space for terminals that wrap on last char
 
308
            width = width - 1
437
309
            s = '%-*.*s' % (width, width, s)
438
310
        self._term_file.write('\r' + s + '\r')
439
311
 
476
348
            return ''
477
349
 
478
350
    def _format_task(self, task):
479
 
        """Format task-specific parts of progress bar.
480
 
 
481
 
        :returns: (text_part, counter_part) both unicode strings.
482
 
        """
483
351
        if not task.show_count:
484
352
            s = ''
485
353
        elif task.current_cnt is not None and task.total_cnt is not None:
495
363
            t = t._parent_task
496
364
            if t.msg:
497
365
                m = t.msg + ':' + m
498
 
        return m, s
 
366
        return m + s
499
367
 
500
368
    def _render_line(self):
501
369
        bar_string = self._render_bar()
502
370
        if self._last_task:
503
 
            task_part, counter_part = self._format_task(self._last_task)
 
371
            task_msg = self._format_task(self._last_task)
504
372
        else:
505
 
            task_part = counter_part = ''
 
373
            task_msg = ''
506
374
        if self._last_task and not self._last_task.show_transport_activity:
507
375
            trans = ''
508
376
        else:
509
377
            trans = self._last_transport_msg
510
 
        # the bar separates the transport activity from the message, so even
511
 
        # if there's no bar or spinner, we must show something if both those
512
 
        # fields are present
513
 
        if (task_part or trans) and not bar_string:
514
 
            bar_string = '| '
515
 
        # preferentially truncate the task message if we don't have enough
516
 
        # space
517
 
        avail_width = self._avail_width()
518
 
        if avail_width is not None:
519
 
            # if terminal avail_width is unknown, don't truncate
520
 
            current_len = len(bar_string) + len(trans) + len(task_part) + len(counter_part)
521
 
            gap = current_len - avail_width
522
 
            if gap > 0:
523
 
                task_part = task_part[:-gap-2] + '..'
524
 
        s = trans + bar_string + task_part + counter_part
525
 
        if avail_width is not None:
526
 
            if len(s) < avail_width:
527
 
                s = s.ljust(avail_width)
528
 
            elif len(s) > avail_width:
529
 
                s = s[:avail_width]
530
 
        return s
 
378
            if trans:
 
379
                trans += ' | '
 
380
        return (bar_string + trans + task_msg)
531
381
 
532
382
    def _repaint(self):
533
383
        s = self._render_line()
589
439
            rate = (self._bytes_since_update
590
440
                    / (now - self._transport_update_time))
591
441
            # using base-10 units (see HACKING.txt).
592
 
            msg = ("%6dkB %5dkB/s " %
 
442
            msg = ("%6dkB %5dkB/s" %
593
443
                    (self._total_byte_count / 1000, int(rate) / 1000,))
594
444
            self._transport_update_time = now
595
445
            self._last_repaint = now