13
14
# You should have received a copy of the GNU General Public License
14
15
# along with this program; if not, write to the Free Software
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.
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.
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(), """
30
44
from bzrlib import (
33
49
from bzrlib.trace import mutter
34
from bzrlib.symbol_versioning import (
40
52
def _supports_progress(f):
41
"""Detect if we can use pretty progress bars on file F.
53
"""Detect if we can use pretty progress bars on the output stream f.
43
If this returns true we expect that a human may be looking at that
55
If this returns true we expect that a human may be looking at that
44
56
output, and that we can repaint a line to update it.
46
This doesn't check the policy for whether we *should* use them.
48
58
isatty = getattr(f, 'isatty', None)
53
# The following case also handles Win32 - on that platform $TERM is
54
# typically never set, so the case None is treated as a smart terminal,
55
# not dumb. <https://bugs.launchpad.net/bugs/334808> win32 files do have
56
# isatty methods that return true.
57
63
if os.environ.get('TERM') == 'dumb':
58
64
# e.g. emacs compile window
63
class ProgressTask(object):
64
"""Model component of a progress indicator.
66
Most code that needs to indicate progress should update one of these,
67
and it will in turn update the display, if one is present.
69
Code updating the task may also set fields as hints about how to display
70
it: show_pct, show_spinner, show_eta, show_count, show_bar. UIs
71
will not necessarily respect all these fields.
74
def __init__(self, parent_task=None, ui_factory=None, progress_view=None):
75
"""Construct a new progress task.
77
:param parent_task: Enclosing ProgressTask or None.
79
:param progress_view: ProgressView to display this ProgressTask.
81
:param ui_factory: The UI factory that will display updates;
82
deprecated in favor of passing progress_view directly.
84
Normally you should not call this directly but rather through
85
`ui_factory.nested_progress_bar`.
87
self._parent_task = parent_task
90
self.current_cnt = None
92
# TODO: deprecate passing ui_factory
93
self.ui_factory = ui_factory
94
self.progress_view = progress_view
96
self.show_spinner = True
97
self.show_eta = False,
98
self.show_count = True
102
return '%s(%r/%r, msg=%r)' % (
103
self.__class__.__name__,
108
def update(self, msg, current_cnt=None, total_cnt=None):
110
self.current_cnt = current_cnt
112
self.total_cnt = total_cnt
113
if self.progress_view:
114
self.progress_view.show_progress(self)
116
self.ui_factory._progress_updated(self)
119
self.update(self.msg)
122
if self.progress_view:
123
self.progress_view.task_finished(self)
125
self.ui_factory._progress_finished(self)
127
def make_sub_task(self):
128
return ProgressTask(self, ui_factory=self.ui_factory,
129
progress_view=self.progress_view)
131
def _overall_completion_fraction(self, child_fraction=0.0):
132
"""Return fractional completion of this task and its parents
134
Returns None if no completion can be computed."""
135
if self.current_cnt is not None and self.total_cnt:
136
own_fraction = (float(self.current_cnt) + child_fraction) / self.total_cnt
138
# if this task has no estimation, it just passes on directly
139
# whatever the child has measured...
140
own_fraction = child_fraction
141
if self._parent_task is None:
144
if own_fraction is None:
146
return self._parent_task._overall_completion_fraction(own_fraction)
148
def note(self, fmt_string, *args):
149
"""Record a note without disrupting the progress bar."""
150
# XXX: shouldn't be here; put it in mutter or the ui instead
152
self.ui_factory.note(fmt_string % args)
154
self.ui_factory.note(fmt_string)
157
# XXX: shouldn't be here; put it in mutter or the ui instead
158
if self.progress_view:
159
self.progress_view.clear()
161
self.ui_factory.clear_term()
164
@deprecated_function(deprecated_in((1, 16, 0)))
69
_progress_bar_types = {}
165
72
def ProgressBar(to_file=None, **kwargs):
166
73
"""Abstract factory"""
167
74
if to_file is None:
184
91
_progress_bar_types.keys())
185
92
return _progress_bar_types[requested_bar_type](to_file=to_file, **kwargs)
188
# NOTE: This is also deprecated; you should provide a ProgressView instead.
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()
189
158
class _BaseProgressBar(object):
191
160
def __init__(self,
453
426
# so just show an expanded spinning thingy
454
427
m = self.spin_pos % cols
455
428
ms = (' ' * m + '*').ljust(cols)
457
430
bar_str = '[' + ms + '] '
463
m = spin_str + bar_str + self.last_msg + count_str \
436
m = spin_str + bar_str + self.last_msg + count_str + pct_str + eta_str
465
437
self.to_file.write('\r%-*.*s' % (self.width - 1, self.width - 1, m))
466
438
self._have_output = True
467
439
#self.to_file.flush()
470
442
if self._have_output:
471
443
self.to_file.write('\r%s\r' % (' ' * (self.width - 1)))
472
444
self._have_output = False
473
#self.to_file.flush()
445
#self.to_file.flush()
448
_progress_bar_types['tty'] = TTYProgressBar
478
451
class ChildProgress(_BaseProgressBar):
479
452
"""A progress indicator that pushes its data to the parent"""
481
@deprecated_function(deprecated_in((1, 16, 0)))
482
454
def __init__(self, _stack, **kwargs):
483
455
_BaseProgressBar.__init__(self, _stack=_stack, **kwargs)
484
456
self.parent = _stack.top()