1
# Copyright (C) 2005 Aaron Bentley <aaron.bentley@utoronto.ca>
2
# Copyright (C) 2005, 2006 Canonical Ltd
1
# Copyright (C) 2005, 2006, 2008, 2009 Canonical Ltd
4
3
# This program is free software; you can redistribute it and/or modify
5
4
# it under the terms of the GNU General Public License as published by
14
13
# You should have received a copy of the GNU General Public License
15
14
# along with this program; if not, write to the Free Software
16
# Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
19
"""Simple text-mode progress indicator.
21
To display an indicator, create a ProgressBar object. Call it,
22
passing Progress objects indicating the current state. When done,
25
Progress is suppressed when output is not sent to a terminal, so as
26
not to clutter log files.
15
# Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
18
"""Progress indicators.
20
The usual way to use this is via bzrlib.ui.ui_factory.nested_progress_bar which
21
will manage a conceptual stack of nested activities.
29
# TODO: should be a global option e.g. --silent that disables progress
30
# indicators, preferably without needing to adjust all code that
31
# potentially calls them.
33
# TODO: If not on a tty perhaps just print '......' for the benefit of IDEs, etc
35
# TODO: Optionally show elapsed time instead/as well as ETA; nicer
36
# when the rate is unpredictable
42
from bzrlib.lazy_import import lazy_import
43
lazy_import(globals(), """
44
30
from bzrlib import (
49
33
from bzrlib.trace import mutter
34
from bzrlib.symbol_versioning import (
40
# XXX: deprecated; can be removed when the ProgressBar factory is removed
52
41
def _supports_progress(f):
53
42
"""Detect if we can use pretty progress bars on the output stream f.
55
If this returns true we expect that a human may be looking at that
44
If this returns true we expect that a human may be looking at that
56
45
output, and that we can repaint a line to update it.
58
47
isatty = getattr(f, 'isatty', None)
69
_progress_bar_types = {}
58
class ProgressTask(object):
59
"""Model component of a progress indicator.
61
Most code that needs to indicate progress should update one of these,
62
and it will in turn update the display, if one is present.
64
Code updating the task may also set fields as hints about how to display
65
it: show_pct, show_spinner, show_eta, show_count, show_bar. UIs
66
will not necessarily respect all these fields.
69
def __init__(self, parent_task=None, ui_factory=None):
70
"""Construct a new progress task.
72
Normally you should not call this directly but rather through
73
`ui_factory.nested_progress_bar`.
75
self._parent_task = parent_task
78
self.current_cnt = None
80
self.ui_factory = ui_factory
82
self.show_spinner = True
83
self.show_eta = False,
84
self.show_count = True
88
return '%s(%r/%r, msg=%r)' % (
89
self.__class__.__name__,
94
def update(self, msg, current_cnt=None, total_cnt=None):
96
self.current_cnt = current_cnt
98
self.total_cnt = total_cnt
99
self.ui_factory._progress_updated(self)
102
self.update(self.msg)
105
self.ui_factory._progress_finished(self)
107
def make_sub_task(self):
108
return ProgressTask(self, self.ui_factory)
110
def _overall_completion_fraction(self, child_fraction=0.0):
111
"""Return fractional completion of this task and its parents
113
Returns None if no completion can be computed."""
114
if self.current_cnt is not None and self.total_cnt:
115
own_fraction = (float(self.current_cnt) + child_fraction) / self.total_cnt
117
# if this task has no estimation, it just passes on directly
118
# whatever the child has measured...
119
own_fraction = child_fraction
120
if self._parent_task is None:
123
if own_fraction is None:
125
return self._parent_task._overall_completion_fraction(own_fraction)
127
def note(self, fmt_string, *args):
128
"""Record a note without disrupting the progress bar."""
129
# XXX: shouldn't be here; put it in mutter or the ui instead
131
self.ui_factory.note(fmt_string % args)
133
self.ui_factory.note(fmt_string)
136
# XXX: shouldn't be here; put it in mutter or the ui instead
137
self.ui_factory.clear_term()
140
@deprecated_function(deprecated_in((1, 16, 0)))
72
141
def ProgressBar(to_file=None, **kwargs):
73
142
"""Abstract factory"""
74
143
if to_file is None:
91
160
_progress_bar_types.keys())
92
161
return _progress_bar_types[requested_bar_type](to_file=to_file, **kwargs)
95
class ProgressBarStack(object):
96
"""A stack of progress bars."""
105
to_messages_file=None,
107
"""Setup the stack with the parameters the progress bars should have."""
110
if to_messages_file is None:
111
to_messages_file = sys.stdout
112
self._to_file = to_file
113
self._show_pct = show_pct
114
self._show_spinner = show_spinner
115
self._show_eta = show_eta
116
self._show_bar = show_bar
117
self._show_count = show_count
118
self._to_messages_file = to_messages_file
120
self._klass = klass or ProgressBar
123
if len(self._stack) != 0:
124
return self._stack[-1]
129
if len(self._stack) != 0:
130
return self._stack[0]
134
def get_nested(self):
135
"""Return a nested progress bar."""
136
if len(self._stack) == 0:
139
func = self.top().child_progress
140
new_bar = func(to_file=self._to_file,
141
show_pct=self._show_pct,
142
show_spinner=self._show_spinner,
143
show_eta=self._show_eta,
144
show_bar=self._show_bar,
145
show_count=self._show_count,
146
to_messages_file=self._to_messages_file,
148
self._stack.append(new_bar)
151
def return_pb(self, bar):
152
"""Return bar after its been used."""
153
if bar is not self._stack[-1]:
154
raise errors.MissingProgressBarFinish()
158
164
class _BaseProgressBar(object):
160
166
def __init__(self,
230
237
return DummyProgress(**kwargs)
233
_progress_bar_types['dummy'] = DummyProgress
234
_progress_bar_types['none'] = DummyProgress
237
240
class DotsProgressBar(_BaseProgressBar):
242
@deprecated_function(deprecated_in((1, 16, 0)))
239
243
def __init__(self, **kwargs):
240
244
_BaseProgressBar.__init__(self, **kwargs)
241
245
self.last_msg = None
242
246
self.need_nl = False
247
251
def update(self, msg=None, current_cnt=None, total_cnt=None):
248
252
if msg and msg != self.last_msg:
318
318
self.last_updates = self.last_updates[-self._max_last_updates:]
319
319
self.last_update = now
323
self.update(self.last_msg, self.last_cnt, self.last_total,
323
self.update(self.last_msg, self.last_cnt, self.last_total,
324
324
self.child_fraction)
326
326
def child_update(self, message, current, total):
331
331
elif self.last_cnt + child_fraction <= self.last_total:
332
332
self.child_fraction = child_fraction
334
mutter('not updating child fraction')
335
333
if self.last_msg is None:
336
334
self.last_msg = ''
339
def update(self, msg, current_cnt=None, total_cnt=None,
341
"""Update and redraw progress bar."""
337
def update(self, msg, current_cnt=None, total_cnt=None,
339
"""Update and redraw progress bar.
343
342
msg = self.last_msg
371
373
self.last_total = total_cnt
372
374
self.child_fraction = child_fraction
374
# each function call takes 20ms/4000 = 0.005 ms,
376
# each function call takes 20ms/4000 = 0.005 ms,
375
377
# but multiple that by 4000 calls -> starts to cost.
376
378
# so anything to make this function call faster
377
379
# will improve base 'diff' time by up to 0.1 seconds.
381
383
if self.show_eta and self.start_time and self.last_total:
382
eta = get_eta(self.start_time, self.last_cnt + self.child_fraction,
384
eta = get_eta(self.start_time, self.last_cnt + self.child_fraction,
383
385
self.last_total, last_updates = self.last_updates)
384
386
eta_str = " " + str_tdelta(eta)
388
390
if self.show_spinner:
389
spin_str = self.SPIN_CHARS[self.spin_pos % 4] + ' '
391
spin_str = self.SPIN_CHARS[self.spin_pos % 4] + ' '
409
411
# make both fields the same size
410
412
t = '%i' % (self.last_total)
411
413
c = '%*i' % (len(t), self.last_cnt)
412
count_str = ' ' + c + '/' + t
414
count_str = ' ' + c + '/' + t
414
416
if self.show_bar:
415
417
# progress bar, if present, soaks up all remaining space
426
428
# so just show an expanded spinning thingy
427
429
m = self.spin_pos % cols
428
430
ms = (' ' * m + '*').ljust(cols)
430
432
bar_str = '[' + ms + '] '
436
m = spin_str + bar_str + self.last_msg + count_str + pct_str + eta_str
438
m = spin_str + bar_str + self.last_msg + count_str \
437
440
self.to_file.write('\r%-*.*s' % (self.width - 1, self.width - 1, m))
438
441
self._have_output = True
439
442
#self.to_file.flush()
442
445
if self._have_output:
443
446
self.to_file.write('\r%s\r' % (' ' * (self.width - 1)))
444
447
self._have_output = False
445
#self.to_file.flush()
448
_progress_bar_types['tty'] = TTYProgressBar
448
#self.to_file.flush()
451
451
class ChildProgress(_BaseProgressBar):
452
452
"""A progress indicator that pushes its data to the parent"""
454
@deprecated_function(deprecated_in((1, 16, 0)))
454
455
def __init__(self, _stack, **kwargs):
455
456
_BaseProgressBar.__init__(self, _stack=_stack, **kwargs)
456
457
self.parent = _stack.top()
548
548
self.cur_phase = 0
550
550
self.cur_phase += 1
551
assert self.cur_phase < self.total
552
551
self.pb.update(self.message, self.cur_phase, self.total)
557
result = doctest.testmod()
560
print "All tests passed"
562
print "No tests to run"
568
print 'dumb-terminal test:'
569
pb = DotsProgressBar()
571
pb.update('Leoparden', i, 99)
577
print 'smart-terminal test:'
578
pb = ProgressBar(show_pct=True, show_bar=True, show_spinner=False)
580
pb.update('Elephanten', i, 99)
588
if __name__ == "__main__":
554
_progress_bar_types = {}
555
_progress_bar_types['dummy'] = DummyProgress
556
_progress_bar_types['none'] = DummyProgress
557
_progress_bar_types['tty'] = TTYProgressBar
558
_progress_bar_types['dots'] = DotsProgressBar