Show More
ui.py
1845 lines
| 67.0 KiB
| text/x-python
|
PythonLexer
/ mercurial / ui.py
mpm@selenic.com
|
r207 | # ui.py - user interface bits for mercurial | ||
# | ||||
Thomas Arendsen Hein
|
r4635 | # Copyright 2005-2007 Matt Mackall <mpm@selenic.com> | ||
mpm@selenic.com
|
r207 | # | ||
Martin Geisler
|
r8225 | # This software may be used and distributed according to the terms of the | ||
Matt Mackall
|
r10263 | # GNU General Public License version 2 or any later version. | ||
mpm@selenic.com
|
r207 | |||
Gregory Szorc
|
r25989 | from __future__ import absolute_import | ||
Simon Farnsworth
|
r30976 | import collections | ||
Kostia Balytskyi
|
r30480 | import contextlib | ||
Gregory Szorc
|
r25989 | import errno | ||
import getpass | ||||
Pierre-Yves David
|
r25629 | import inspect | ||
Gregory Szorc
|
r25989 | import os | ||
Matt Mackall
|
r27392 | import re | ||
Augie Fackler
|
r30992 | import signal | ||
Gregory Szorc
|
r25989 | import socket | ||
Augie Fackler
|
r30992 | import subprocess | ||
Gregory Szorc
|
r25989 | import sys | ||
import tempfile | ||||
import traceback | ||||
from .i18n import _ | ||||
from .node import hex | ||||
from . import ( | ||||
Pierre-Yves David
|
r31087 | color, | ||
Gregory Szorc
|
r25989 | config, | ||
r32984 | configitems, | |||
Pulkit Goyal
|
r30277 | encoding, | ||
Gregory Szorc
|
r25989 | error, | ||
formatter, | ||||
progress, | ||||
Pulkit Goyal
|
r30519 | pycompat, | ||
Jun Wu
|
r31679 | rcutil, | ||
Gregory Szorc
|
r25989 | scmutil, | ||
util, | ||||
) | ||||
mpm@selenic.com
|
r207 | |||
liscju
|
r29378 | urlreq = util.urlreq | ||
Simon Farnsworth
|
r30979 | # for use with str.translate(None, _keepalnum), to keep just alphanumerics | ||
Yuya Nishihara
|
r31253 | _keepalnum = ''.join(c for c in map(pycompat.bytechr, range(256)) | ||
if not c.isalnum()) | ||||
Simon Farnsworth
|
r30979 | |||
Augie Fackler
|
r32872 | # The config knobs that will be altered (if unset) by ui.tweakdefaults. | ||
tweakrc = """ | ||||
[ui] | ||||
# The rollback command is dangerous. As a rule, don't use it. | ||||
rollback = False | ||||
Martin von Zweigbergk
|
r35066 | # Make `hg status` report copy information | ||
statuscopies = yes | ||||
Augie Fackler
|
r32872 | |||
[commands] | ||||
# Make `hg status` emit cwd-relative paths by default. | ||||
status.relative = yes | ||||
Augie Fackler
|
r34708 | # Refuse to perform an `hg update` that would cause a file content merge | ||
update.check = noconflict | ||||
Augie Fackler
|
r32872 | |||
[diff] | ||||
git = 1 | ||||
""" | ||||
Matt Mackall
|
r22419 | samplehgrcs = { | ||
'user': | ||||
Yuya Nishihara
|
r33650 | b"""# example user config (see 'hg help config' for more info) | ||
Matt Mackall
|
r22419 | [ui] | ||
# name and email, e.g. | ||||
# username = Jane Doe <jdoe@example.com> | ||||
username = | ||||
Augie Fackler
|
r34569 | # We recommend enabling tweakdefaults to get slight improvements to | ||
# the UI over time. Make sure to set HGPLAIN in the environment when | ||||
# writing scripts! | ||||
# tweakdefaults = True | ||||
Pierre-Yves David
|
r32094 | # uncomment to disable color in command output | ||
Pierre-Yves David
|
r32095 | # (see 'hg help color' for details) | ||
Pierre-Yves David
|
r32094 | # color = never | ||
Pierre-Yves David
|
r31124 | |||
Pierre-Yves David
|
r32101 | # uncomment to disable command output pagination | ||
# (see 'hg help pager' for details) | ||||
Pierre-Yves David
|
r32105 | # paginate = never | ||
Pierre-Yves David
|
r32101 | |||
Matt Mackall
|
r22419 | [extensions] | ||
# uncomment these lines to enable some popular extensions | ||||
timeless
|
r29978 | # (see 'hg help extensions' for more info) | ||
Matt Mackall
|
r22419 | # | ||
Pierre-Yves David
|
r32097 | # churn = | ||
Pierre-Yves David
|
r32098 | """, | ||
Matt Mackall
|
r22419 | |||
Jordi Gutiérrez Hermoso
|
r22837 | 'cloned': | ||
Yuya Nishihara
|
r33650 | b"""# example repository config (see 'hg help config' for more info) | ||
Jordi Gutiérrez Hermoso
|
r22837 | [paths] | ||
default = %s | ||||
# path aliases to other clones of this repo in URLs or filesystem paths | ||||
timeless
|
r29978 | # (see 'hg help config.paths' for more info) | ||
Jordi Gutiérrez Hermoso
|
r22837 | # | ||
Rishabh Madan
|
r31064 | # default:pushurl = ssh://jdoe@example.net/hg/jdoes-fork | ||
# my-fork = ssh://jdoe@example.net/hg/jdoes-fork | ||||
# my-clone = /home/jdoe/jdoes-clone | ||||
Jordi Gutiérrez Hermoso
|
r22837 | |||
[ui] | ||||
# name and email (local to this repository, optional), e.g. | ||||
# username = Jane Doe <jdoe@example.com> | ||||
""", | ||||
Matt Mackall
|
r22419 | 'local': | ||
Yuya Nishihara
|
r33650 | b"""# example repository config (see 'hg help config' for more info) | ||
Jordi Gutiérrez Hermoso
|
r22836 | [paths] | ||
# path aliases to other clones of this repo in URLs or filesystem paths | ||||
timeless
|
r29978 | # (see 'hg help config.paths' for more info) | ||
Jordi Gutiérrez Hermoso
|
r22836 | # | ||
Rishabh Madan
|
r31064 | # default = http://example.com/hg/example-repo | ||
# default:pushurl = ssh://jdoe@example.net/hg/jdoes-fork | ||||
# my-fork = ssh://jdoe@example.net/hg/jdoes-fork | ||||
# my-clone = /home/jdoe/jdoes-clone | ||||
Jordi Gutiérrez Hermoso
|
r22836 | |||
[ui] | ||||
# name and email (local to this repository, optional), e.g. | ||||
# username = Jane Doe <jdoe@example.com> | ||||
Matt Mackall
|
r22419 | """, | ||
'global': | ||||
Yuya Nishihara
|
r33650 | b"""# example system-wide hg config (see 'hg help config' for more info) | ||
Matt Mackall
|
r22419 | |||
Pierre-Yves David
|
r31124 | [ui] | ||
Pierre-Yves David
|
r32094 | # uncomment to disable color in command output | ||
Pierre-Yves David
|
r32095 | # (see 'hg help color' for details) | ||
Pierre-Yves David
|
r32094 | # color = never | ||
Pierre-Yves David
|
r31124 | |||
Pierre-Yves David
|
r32101 | # uncomment to disable command output pagination | ||
# (see 'hg help pager' for details) | ||||
Pierre-Yves David
|
r32105 | # paginate = never | ||
Pierre-Yves David
|
r32101 | |||
Matt Mackall
|
r22419 | [extensions] | ||
# uncomment these lines to enable some popular extensions | ||||
timeless
|
r29978 | # (see 'hg help extensions' for more info) | ||
Matt Mackall
|
r22419 | # | ||
# blackbox = | ||||
Pierre-Yves David
|
r32097 | # churn = | ||
Pierre-Yves David
|
r32098 | """, | ||
Matt Mackall
|
r22419 | } | ||
Augie Fackler
|
r34483 | def _maybestrurl(maybebytes): | ||
if maybebytes is None: | ||||
return None | ||||
return pycompat.strurl(maybebytes) | ||||
def _maybebytesurl(maybestr): | ||||
if maybestr is None: | ||||
return None | ||||
return pycompat.bytesurl(maybestr) | ||||
Kyle Lippincott
|
r30945 | |||
class httppasswordmgrdbproxy(object): | ||||
"""Delays loading urllib2 until it's needed.""" | ||||
def __init__(self): | ||||
self._mgr = None | ||||
def _get_mgr(self): | ||||
if self._mgr is None: | ||||
self._mgr = urlreq.httppasswordmgrwithdefaultrealm() | ||||
return self._mgr | ||||
Augie Fackler
|
r34427 | def add_password(self, realm, uris, user, passwd): | ||
Augie Fackler
|
r34483 | if isinstance(uris, tuple): | ||
uris = tuple(_maybestrurl(u) for u in uris) | ||||
else: | ||||
uris = _maybestrurl(uris) | ||||
return self._get_mgr().add_password( | ||||
_maybestrurl(realm), uris, | ||||
_maybestrurl(user), _maybestrurl(passwd)) | ||||
Kyle Lippincott
|
r30945 | |||
Augie Fackler
|
r34427 | def find_user_password(self, realm, uri): | ||
Augie Fackler
|
r34483 | return tuple(_maybebytesurl(v) for v in | ||
self._get_mgr().find_user_password(_maybestrurl(realm), | ||||
_maybestrurl(uri))) | ||||
Kyle Lippincott
|
r30945 | |||
Augie Fackler
|
r30992 | def _catchterm(*args): | ||
raise error.SignalInterrupt | ||||
Kyle Lippincott
|
r30945 | |||
r32958 | # unique object used to detect no default value has been provided when | |||
# retrieving configuration value. | ||||
_unset = object() | ||||
Saurabh Singh
|
r34883 | # _reqexithandlers: callbacks run at the end of a request | ||
_reqexithandlers = [] | ||||
Eric Hopper
|
r1559 | class ui(object): | ||
Matt Mackall
|
r8190 | def __init__(self, src=None): | ||
Yuya Nishihara
|
r30559 | """Create a fresh new ui object if no src given | ||
Use uimod.ui.load() to create a ui which knows global and user configs. | ||||
In most cases, you should use ui.copy() to create a copy of an existing | ||||
ui object. | ||||
""" | ||||
Pierre-Yves David
|
r21132 | # _buffers: used for temporary capture of output | ||
Matt Mackall
|
r8202 | self._buffers = [] | ||
Gregory Szorc
|
r27106 | # 3-tuple describing how each buffer in the stack behaves. | ||
# Values are (capture stderr, capture subprocesses, apply labels). | ||||
Pierre-Yves David
|
r21132 | self._bufferstates = [] | ||
Gregory Szorc
|
r27106 | # When a buffer is active, defines whether we are expanding labels. | ||
# This exists to prevent an extra list lookup. | ||||
self._bufferapplylabels = None | ||||
Bryan O'Sullivan
|
r9851 | self.quiet = self.verbose = self.debugflag = self.tracebackflag = False | ||
Matt Mackall
|
r8208 | self._reportuntrusted = True | ||
r32984 | self._knownconfig = configitems.coreitems | |||
Matt Mackall
|
r8203 | self._ocfg = config.config() # overlay | ||
self._tcfg = config.config() # trusted | ||||
self._ucfg = config.config() # untrusted | ||||
Martin Geisler
|
r8478 | self._trustusers = set() | ||
self._trustgroups = set() | ||||
Idan Kamara
|
r17048 | self.callhooks = True | ||
Gregory Szorc
|
r29109 | # Insecure server connections requested. | ||
self.insecureconnections = False | ||||
Simon Farnsworth
|
r30976 | # Blocked time | ||
self.logblockedtimes = False | ||||
Pierre-Yves David
|
r31106 | # color mode: see mercurial/color.py for possible value | ||
self._colormode = None | ||||
Pierre-Yves David
|
r31113 | self._terminfoparams = {} | ||
Pierre-Yves David
|
r31115 | self._styles = {} | ||
Matt Mackall
|
r8136 | |||
Matt Mackall
|
r8190 | if src: | ||
Idan Kamara
|
r14612 | self.fout = src.fout | ||
self.ferr = src.ferr | ||||
self.fin = src.fin | ||||
Augie Fackler
|
r30992 | self.pageractive = src.pageractive | ||
Augie Fackler
|
r31026 | self._disablepager = src._disablepager | ||
Augie Fackler
|
r32872 | self._tweaked = src._tweaked | ||
Idan Kamara
|
r14612 | |||
Matt Mackall
|
r8203 | self._tcfg = src._tcfg.copy() | ||
self._ucfg = src._ucfg.copy() | ||||
self._ocfg = src._ocfg.copy() | ||||
Matt Mackall
|
r8201 | self._trustusers = src._trustusers.copy() | ||
self._trustgroups = src._trustgroups.copy() | ||||
Sune Foldager
|
r9887 | self.environ = src.environ | ||
Idan Kamara
|
r17048 | self.callhooks = src.callhooks | ||
Gregory Szorc
|
r29109 | self.insecureconnections = src.insecureconnections | ||
Pierre-Yves David
|
r31106 | self._colormode = src._colormode | ||
Pierre-Yves David
|
r31113 | self._terminfoparams = src._terminfoparams.copy() | ||
Pierre-Yves David
|
r31115 | self._styles = src._styles.copy() | ||
Pierre-Yves David
|
r31106 | |||
Matt Mackall
|
r8143 | self.fixconfig() | ||
liscju
|
r29378 | |||
self.httppasswordmgrdb = src.httppasswordmgrdb | ||||
Simon Farnsworth
|
r30976 | self._blockedtimes = src._blockedtimes | ||
Matt Mackall
|
r8143 | else: | ||
Yuya Nishihara
|
r30473 | self.fout = util.stdout | ||
self.ferr = util.stderr | ||||
self.fin = util.stdin | ||||
Augie Fackler
|
r30992 | self.pageractive = False | ||
Augie Fackler
|
r31026 | self._disablepager = False | ||
Augie Fackler
|
r32872 | self._tweaked = False | ||
Idan Kamara
|
r14612 | |||
Sune Foldager
|
r9887 | # shared read-only environment | ||
Pulkit Goyal
|
r30637 | self.environ = encoding.environ | ||
Dirkjan Ochtman
|
r8222 | |||
Kyle Lippincott
|
r30945 | self.httppasswordmgrdb = httppasswordmgrdbproxy() | ||
Simon Farnsworth
|
r30976 | self._blockedtimes = collections.defaultdict(int) | ||
liscju
|
r29378 | |||
Matt Harbison
|
r30832 | allowed = self.configlist('experimental', 'exportableenviron') | ||
if '*' in allowed: | ||||
self._exportableenviron = self.environ | ||||
else: | ||||
self._exportableenviron = {} | ||||
for k in allowed: | ||||
if k in self.environ: | ||||
self._exportableenviron[k] = self.environ[k] | ||||
Yuya Nishihara
|
r30559 | @classmethod | ||
def load(cls): | ||||
"""Create a ui and load global and user configs""" | ||||
u = cls() | ||||
Jun Wu
|
r31685 | # we always trust global config files and environment variables | ||
Jun Wu
|
r31683 | for t, f in rcutil.rccomponents(): | ||
if t == 'path': | ||||
u.readconfig(f, trust=True) | ||||
Jun Wu
|
r31685 | elif t == 'items': | ||
sections = set() | ||||
for section, name, value, source in f: | ||||
# do not set u._ocfg | ||||
# XXX clean this up once immutable config object is a thing | ||||
u._tcfg.set(section, name, value, source) | ||||
u._ucfg.set(section, name, value, source) | ||||
sections.add(section) | ||||
for section in sections: | ||||
u.fixconfig(section=section) | ||||
Jun Wu
|
r31683 | else: | ||
raise error.ProgrammingError('unknown rctype: %s' % t) | ||||
Augie Fackler
|
r32872 | u._maybetweakdefaults() | ||
Yuya Nishihara
|
r30559 | return u | ||
Augie Fackler
|
r32872 | def _maybetweakdefaults(self): | ||
if not self.configbool('ui', 'tweakdefaults'): | ||||
return | ||||
if self._tweaked or self.plain('tweakdefaults'): | ||||
return | ||||
# Note: it is SUPER IMPORTANT that you set self._tweaked to | ||||
# True *before* any calls to setconfig(), otherwise you'll get | ||||
# infinite recursion between setconfig and this method. | ||||
# | ||||
# TODO: We should extract an inner method in setconfig() to | ||||
# avoid this weirdness. | ||||
self._tweaked = True | ||||
tmpcfg = config.config() | ||||
tmpcfg.parse('<tweakdefaults>', tweakrc) | ||||
for section in tmpcfg: | ||||
for name, value in tmpcfg.items(section): | ||||
if not self.hasconfig(section, name): | ||||
self.setconfig(section, name, value, "<tweakdefaults>") | ||||
Matt Mackall
|
r8189 | def copy(self): | ||
Ronny Pfannschmidt
|
r8220 | return self.__class__(self) | ||
Thomas Arendsen Hein
|
r1839 | |||
Yuya Nishihara
|
r29366 | def resetstate(self): | ||
"""Clear internal state that shouldn't persist across commands""" | ||||
if self._progbar: | ||||
self._progbar.resetstate() # reset last-print time of progress bar | ||||
Kyle Lippincott
|
r30945 | self.httppasswordmgrdb = httppasswordmgrdbproxy() | ||
Yuya Nishihara
|
r29366 | |||
Simon Farnsworth
|
r30976 | @contextlib.contextmanager | ||
def timeblockedsection(self, key): | ||||
Simon Farnsworth
|
r30978 | # this is open-coded below - search for timeblockedsection to find them | ||
Simon Farnsworth
|
r30976 | starttime = util.timer() | ||
try: | ||||
yield | ||||
finally: | ||||
self._blockedtimes[key + '_blocked'] += \ | ||||
(util.timer() - starttime) * 1000 | ||||
Yuya Nishihara
|
r29366 | |||
Matt Mackall
|
r16135 | def formatter(self, topic, opts): | ||
Yuya Nishihara
|
r32573 | return formatter.formatter(self, self, topic, opts) | ||
Matt Mackall
|
r16135 | |||
Matt Mackall
|
r14859 | def _trusted(self, fp, f): | ||
Alexis S. L. Carvalho
|
r3677 | st = util.fstat(fp) | ||
Martin Geisler
|
r8657 | if util.isowner(st): | ||
Alexis S. L. Carvalho
|
r3677 | return True | ||
Matt Mackall
|
r8141 | |||
Matt Mackall
|
r8201 | tusers, tgroups = self._trustusers, self._trustgroups | ||
Matt Mackall
|
r8141 | if '*' in tusers or '*' in tgroups: | ||
return True | ||||
user = util.username(st.st_uid) | ||||
group = util.groupname(st.st_gid) | ||||
if user in tusers or group in tgroups or user == util.username(): | ||||
return True | ||||
Matt Mackall
|
r8204 | if self._reportuntrusted: | ||
Martin Geisler
|
r16939 | self.warn(_('not trusting file %s from untrusted ' | ||
Matt Mackall
|
r8141 | 'user %s, group %s\n') % (f, user, group)) | ||
return False | ||||
Alexis S. L. Carvalho
|
r3551 | |||
Matt Mackall
|
r8200 | def readconfig(self, filename, root=None, trust=False, | ||
Alexander Solovyov
|
r8345 | sections=None, remap=None): | ||
Matt Mackall
|
r8142 | try: | ||
Augie Fackler
|
r30348 | fp = open(filename, u'rb') | ||
Matt Mackall
|
r8142 | except IOError: | ||
if not sections: # ignore unless we were looking for something | ||||
return | ||||
raise | ||||
Matt Mackall
|
r8139 | |||
Matt Mackall
|
r8203 | cfg = config.config() | ||
Matt Mackall
|
r14859 | trusted = sections or trust or self._trusted(fp, filename) | ||
Alexis S. L. Carvalho
|
r3552 | |||
Matt Mackall
|
r8142 | try: | ||
Alexander Solovyov
|
r8345 | cfg.read(filename, fp, sections=sections, remap=remap) | ||
Matt Mackall
|
r15407 | fp.close() | ||
Gregory Szorc
|
r25660 | except error.ConfigError as inst: | ||
Matt Mackall
|
r8142 | if trusted: | ||
Matt Mackall
|
r8144 | raise | ||
Martin Geisler
|
r16938 | self.warn(_("ignored: %s\n") % str(inst)) | ||
Alexis S. L. Carvalho
|
r3552 | |||
Brodie Rao
|
r10455 | if self.plain(): | ||
Brodie Rao
|
r10507 | for k in ('debug', 'fallbackencoding', 'quiet', 'slash', | ||
Mathias De Maré
|
r24663 | 'logtemplate', 'statuscopies', 'style', | ||
Brodie Rao
|
r10507 | 'traceback', 'verbose'): | ||
Brodie Rao
|
r10455 | if k in cfg['ui']: | ||
del cfg['ui'][k] | ||||
"Yann E. MORIN"
|
r14373 | for k, v in cfg.items('defaults'): | ||
del cfg['defaults'][k] | ||||
Martin von Zweigbergk
|
r31588 | for k, v in cfg.items('commands'): | ||
del cfg['commands'][k] | ||||
"Yann E. MORIN"
|
r14373 | # Don't remove aliases from the configuration if in the exceptionlist | ||
if self.plain('alias'): | ||||
Brodie Rao
|
r10506 | for k, v in cfg.items('alias'): | ||
del cfg['alias'][k] | ||||
Siddharth Agarwal
|
r24883 | if self.plain('revsetalias'): | ||
for k, v in cfg.items('revsetalias'): | ||||
del cfg['revsetalias'][k] | ||||
Yuya Nishihara
|
r28958 | if self.plain('templatealias'): | ||
for k, v in cfg.items('templatealias'): | ||||
del cfg['templatealias'][k] | ||||
Brodie Rao
|
r10455 | |||
Matt Mackall
|
r8142 | if trusted: | ||
Matt Mackall
|
r8203 | self._tcfg.update(cfg) | ||
self._tcfg.update(self._ocfg) | ||||
self._ucfg.update(cfg) | ||||
self._ucfg.update(self._ocfg) | ||||
Matt Mackall
|
r8139 | |||
Alexis S. L. Carvalho
|
r3347 | if root is None: | ||
root = os.path.expanduser('~') | ||||
self.fixconfig(root=root) | ||||
Alexis S. L. Carvalho
|
r3014 | |||
Nicolas Dumazet
|
r12764 | def fixconfig(self, root=None, section=None): | ||
if section in (None, 'paths'): | ||||
# expand vars and ~ | ||||
# translate paths relative to root (or home) into absolute paths | ||||
Pulkit Goyal
|
r30519 | root = root or pycompat.getcwd() | ||
Nicolas Dumazet
|
r12764 | for c in self._tcfg, self._ucfg, self._ocfg: | ||
for n, p in c.items('paths'): | ||||
Gregory Szorc
|
r29412 | # Ignore sub-options. | ||
if ':' in n: | ||||
continue | ||||
Nicolas Dumazet
|
r12764 | if not p: | ||
continue | ||||
if '%%' in p: | ||||
Yuya Nishihara
|
r30618 | s = self.configsource('paths', n) or 'none' | ||
Nicolas Dumazet
|
r12764 | self.warn(_("(deprecated '%%' in path %s=%s from %s)\n") | ||
Yuya Nishihara
|
r30618 | % (n, p, s)) | ||
Nicolas Dumazet
|
r12764 | p = p.replace('%%', '%') | ||
p = util.expandpath(p) | ||||
Brodie Rao
|
r14076 | if not util.hasscheme(p) and not os.path.isabs(p): | ||
Nicolas Dumazet
|
r12764 | p = os.path.normpath(os.path.join(root, p)) | ||
c.set("paths", n, p) | ||||
Alexis S. L. Carvalho
|
r3347 | |||
Nicolas Dumazet
|
r12764 | if section in (None, 'ui'): | ||
# update ui options | ||||
self.debugflag = self.configbool('ui', 'debug') | ||||
self.verbose = self.debugflag or self.configbool('ui', 'verbose') | ||||
self.quiet = not self.debugflag and self.configbool('ui', 'quiet') | ||||
if self.verbose and self.quiet: | ||||
self.quiet = self.verbose = False | ||||
Ry4an Brase
|
r13493 | self._reportuntrusted = self.debugflag or self.configbool("ui", | ||
Jun Wu
|
r33499 | "report_untrusted") | ||
self.tracebackflag = self.configbool('ui', 'traceback') | ||||
Simon Farnsworth
|
r30976 | self.logblockedtimes = self.configbool('ui', 'logblockedtimes') | ||
Alexis S. L. Carvalho
|
r3350 | |||
Nicolas Dumazet
|
r12764 | if section in (None, 'trusted'): | ||
# update trust information | ||||
self._trustusers.update(self.configlist('trusted', 'users')) | ||||
self._trustgroups.update(self.configlist('trusted', 'groups')) | ||||
Alexis S. L. Carvalho
|
r3551 | |||
Pierre-Yves David
|
r15919 | def backupconfig(self, section, item): | ||
return (self._ocfg.backup(section, item), | ||||
self._tcfg.backup(section, item), | ||||
self._ucfg.backup(section, item),) | ||||
def restoreconfig(self, data): | ||||
self._ocfg.restore(data[0]) | ||||
self._tcfg.restore(data[1]) | ||||
self._ucfg.restore(data[2]) | ||||
Mads Kiilerich
|
r20788 | def setconfig(self, section, name, value, source=''): | ||
Mads Kiilerich
|
r20787 | for cfg in (self._ocfg, self._tcfg, self._ucfg): | ||
Mads Kiilerich
|
r20788 | cfg.set(section, name, value, source) | ||
Nicolas Dumazet
|
r12764 | self.fixconfig(section=section) | ||
Augie Fackler
|
r32872 | self._maybetweakdefaults() | ||
mpm@selenic.com
|
r960 | |||
Matt Mackall
|
r8199 | def _data(self, untrusted): | ||
Matt Mackall
|
r8203 | return untrusted and self._ucfg or self._tcfg | ||
Alexis S. L. Carvalho
|
r3552 | |||
Matt Mackall
|
r8182 | def configsource(self, section, name, untrusted=False): | ||
Yuya Nishihara
|
r30618 | return self._data(untrusted).source(section, name) | ||
Matt Mackall
|
r8182 | |||
r32958 | def config(self, section, name, default=_unset, untrusted=False): | |||
r33058 | """return the plain string version of a config""" | |||
value = self._config(section, name, default=default, | ||||
untrusted=untrusted) | ||||
if value is _unset: | ||||
return None | ||||
return value | ||||
def _config(self, section, name, default=_unset, untrusted=False): | ||||
Yuya Nishihara
|
r34949 | value = itemdefault = default | ||
David Demelier
|
r33329 | item = self._knownconfig.get(section, {}).get(name) | ||
alternates = [(section, name)] | ||||
if item is not None: | ||||
alternates.extend(item.alias) | ||||
Yuya Nishihara
|
r34949 | if callable(item.default): | ||
itemdefault = item.default() | ||||
else: | ||||
itemdefault = item.default | ||||
Boris Feld
|
r34859 | else: | ||
msg = ("accessing unregistered config item: '%s.%s'") | ||||
msg %= (section, name) | ||||
self.develwarn(msg, 2, 'warn-config-unknown') | ||||
David Demelier
|
r33329 | |||
if default is _unset: | ||||
if item is None: | ||||
value = default | ||||
Boris Feld
|
r33471 | elif item.default is configitems.dynamicdefault: | ||
value = None | ||||
msg = "config item requires an explicit default value: '%s.%s'" | ||||
msg %= (section, name) | ||||
self.develwarn(msg, 2, 'warn-config-default') | ||||
David Demelier
|
r33329 | else: | ||
Yuya Nishihara
|
r34949 | value = itemdefault | ||
Boris Feld
|
r33471 | elif (item is not None | ||
Yuya Nishihara
|
r34949 | and item.default is not configitems.dynamicdefault | ||
and default != itemdefault): | ||||
msg = ("specifying a mismatched default value for a registered " | ||||
David Demelier
|
r33329 | "config item: '%s.%s' '%s'") | ||
msg %= (section, name, default) | ||||
self.develwarn(msg, 2, 'warn-config-default') | ||||
r32987 | ||||
David Demelier
|
r33329 | for s, n in alternates: | ||
candidate = self._data(untrusted).get(s, n, None) | ||||
r33058 | if candidate is not None: | |||
value = candidate | ||||
David Demelier
|
r33329 | section = s | ||
Matt Mackall
|
r15035 | name = n | ||
break | ||||
Matt Mackall
|
r8204 | if self.debugflag and not untrusted and self._reportuntrusted: | ||
David Demelier
|
r33329 | for s, n in alternates: | ||
uvalue = self._ucfg.get(s, n) | ||||
Augie Fackler
|
r19536 | if uvalue is not None and uvalue != value: | ||
self.debug("ignoring untrusted configuration option " | ||||
David Demelier
|
r33329 | "%s.%s = %s\n" % (s, n, uvalue)) | ||
Alexis S. L. Carvalho
|
r3552 | return value | ||
Alexis S. L. Carvalho
|
r3341 | |||
r32967 | def configsuboptions(self, section, name, default=_unset, untrusted=False): | |||
Gregory Szorc
|
r27252 | """Get a config option and all sub-options. | ||
Some config options have sub-options that are declared with the | ||||
format "key:opt = value". This method is used to return the main | ||||
option and all its declared sub-options. | ||||
Returns a 2-tuple of ``(option, sub-options)``, where `sub-options`` | ||||
is a dict of defined sub-options where keys and values are strings. | ||||
""" | ||||
r32966 | main = self.config(section, name, default, untrusted=untrusted) | |||
Gregory Szorc
|
r27252 | data = self._data(untrusted) | ||
sub = {} | ||||
prefix = '%s:' % name | ||||
for k, v in data.items(section): | ||||
if k.startswith(prefix): | ||||
sub[k[len(prefix):]] = v | ||||
if self.debugflag and not untrusted and self._reportuntrusted: | ||||
for k, v in sub.items(): | ||||
uvalue = self._ucfg.get(section, '%s:%s' % (name, k)) | ||||
if uvalue is not None and uvalue != v: | ||||
self.debug('ignoring untrusted configuration option ' | ||||
'%s:%s.%s = %s\n' % (section, name, k, uvalue)) | ||||
return main, sub | ||||
r32965 | def configpath(self, section, name, default=_unset, untrusted=False): | |||
Simon Heimberg
|
r14924 | 'get a path config item, expanded relative to repo root or config file' | ||
Matt Mackall
|
r13238 | v = self.config(section, name, default, untrusted) | ||
Simon Heimberg
|
r14923 | if v is None: | ||
return None | ||||
Matt Mackall
|
r13238 | if not os.path.isabs(v) or "://" not in v: | ||
src = self.configsource(section, name, untrusted) | ||||
if ':' in src: | ||||
Simon Heimberg
|
r14922 | base = os.path.dirname(src.rsplit(':')[0]) | ||
Matt Mackall
|
r13238 | v = os.path.join(base, os.path.expanduser(v)) | ||
return v | ||||
r32959 | def configbool(self, section, name, default=_unset, untrusted=False): | |||
Sune Foldager
|
r14171 | """parse a configuration element as a boolean | ||
Yuya Nishihara
|
r34133 | >>> u = ui(); s = b'foo' | ||
>>> u.setconfig(s, b'true', b'yes') | ||||
>>> u.configbool(s, b'true') | ||||
Sune Foldager
|
r14171 | True | ||
Yuya Nishihara
|
r34133 | >>> u.setconfig(s, b'false', b'no') | ||
>>> u.configbool(s, b'false') | ||||
Sune Foldager
|
r14171 | False | ||
Yuya Nishihara
|
r34133 | >>> u.configbool(s, b'unknown') | ||
Sune Foldager
|
r14171 | False | ||
Yuya Nishihara
|
r34133 | >>> u.configbool(s, b'unknown', True) | ||
Sune Foldager
|
r14171 | True | ||
Yuya Nishihara
|
r34133 | >>> u.setconfig(s, b'invalid', b'somevalue') | ||
>>> u.configbool(s, b'invalid') | ||||
Sune Foldager
|
r14171 | Traceback (most recent call last): | ||
... | ||||
ConfigError: foo.invalid is not a boolean ('somevalue') | ||||
""" | ||||
r33059 | v = self._config(section, name, default, untrusted=untrusted) | |||
Martin Geisler
|
r8527 | if v is None: | ||
r33059 | return v | |||
if v is _unset: | ||||
r32959 | if default is _unset: | |||
return False | ||||
Matt Mackall
|
r8144 | return default | ||
Dirkjan Ochtman
|
r10243 | if isinstance(v, bool): | ||
return v | ||||
Augie Fackler
|
r12087 | b = util.parsebool(v) | ||
if b is None: | ||||
Sune Foldager
|
r14171 | raise error.ConfigError(_("%s.%s is not a boolean ('%s')") | ||
Matt Mackall
|
r8144 | % (section, name, v)) | ||
Augie Fackler
|
r12087 | return b | ||
Alexis S. L. Carvalho
|
r3552 | |||
r32960 | def configwith(self, convert, section, name, default=_unset, | |||
Bryan O'Sullivan
|
r30926 | desc=None, untrusted=False): | ||
"""parse a configuration element with a conversion function | ||||
Yuya Nishihara
|
r34133 | >>> u = ui(); s = b'foo' | ||
>>> u.setconfig(s, b'float1', b'42') | ||||
>>> u.configwith(float, s, b'float1') | ||||
Bryan O'Sullivan
|
r30926 | 42.0 | ||
Yuya Nishihara
|
r34133 | >>> u.setconfig(s, b'float2', b'-4.25') | ||
>>> u.configwith(float, s, b'float2') | ||||
Jun Wu
|
r30932 | -4.25 | ||
Yuya Nishihara
|
r34133 | >>> u.configwith(float, s, b'unknown', 7) | ||
r32960 | 7.0 | |||
Yuya Nishihara
|
r34133 | >>> u.setconfig(s, b'invalid', b'somevalue') | ||
>>> u.configwith(float, s, b'invalid') | ||||
Bryan O'Sullivan
|
r30926 | Traceback (most recent call last): | ||
... | ||||
ConfigError: foo.invalid is not a valid float ('somevalue') | ||||
Yuya Nishihara
|
r34133 | >>> u.configwith(float, s, b'invalid', desc=b'womble') | ||
Bryan O'Sullivan
|
r30926 | Traceback (most recent call last): | ||
... | ||||
ConfigError: foo.invalid is not a valid womble ('somevalue') | ||||
""" | ||||
r32960 | v = self.config(section, name, default, untrusted) | |||
Bryan O'Sullivan
|
r30926 | if v is None: | ||
r32960 | return v # do not attempt to convert None | |||
Bryan O'Sullivan
|
r30926 | try: | ||
return convert(v) | ||||
Boris Feld
|
r32462 | except (ValueError, error.ParseError): | ||
Bryan O'Sullivan
|
r30926 | if desc is None: | ||
Yuya Nishihara
|
r34205 | desc = pycompat.sysbytes(convert.__name__) | ||
Bryan O'Sullivan
|
r30926 | raise error.ConfigError(_("%s.%s is not a valid %s ('%s')") | ||
% (section, name, desc, v)) | ||||
r32961 | def configint(self, section, name, default=_unset, untrusted=False): | |||
Sune Foldager
|
r14171 | """parse a configuration element as an integer | ||
Yuya Nishihara
|
r34133 | >>> u = ui(); s = b'foo' | ||
>>> u.setconfig(s, b'int1', b'42') | ||||
>>> u.configint(s, b'int1') | ||||
Sune Foldager
|
r14171 | 42 | ||
Yuya Nishihara
|
r34133 | >>> u.setconfig(s, b'int2', b'-42') | ||
>>> u.configint(s, b'int2') | ||||
Sune Foldager
|
r14171 | -42 | ||
Yuya Nishihara
|
r34133 | >>> u.configint(s, b'unknown', 7) | ||
Sune Foldager
|
r14171 | 7 | ||
Yuya Nishihara
|
r34133 | >>> u.setconfig(s, b'invalid', b'somevalue') | ||
>>> u.configint(s, b'invalid') | ||||
Sune Foldager
|
r14171 | Traceback (most recent call last): | ||
... | ||||
Bryan O'Sullivan
|
r30927 | ConfigError: foo.invalid is not a valid integer ('somevalue') | ||
Sune Foldager
|
r14171 | """ | ||
Bryan O'Sullivan
|
r30927 | return self.configwith(int, section, name, default, 'integer', | ||
untrusted) | ||||
Sune Foldager
|
r14171 | |||
r32962 | def configbytes(self, section, name, default=_unset, untrusted=False): | |||
Bryan O'Sullivan
|
r19065 | """parse a configuration element as a quantity in bytes | ||
Units can be specified as b (bytes), k or kb (kilobytes), m or | ||||
mb (megabytes), g or gb (gigabytes). | ||||
Yuya Nishihara
|
r34133 | >>> u = ui(); s = b'foo' | ||
>>> u.setconfig(s, b'val1', b'42') | ||||
>>> u.configbytes(s, b'val1') | ||||
Bryan O'Sullivan
|
r19065 | 42 | ||
Yuya Nishihara
|
r34133 | >>> u.setconfig(s, b'val2', b'42.5 kb') | ||
>>> u.configbytes(s, b'val2') | ||||
Bryan O'Sullivan
|
r19065 | 43520 | ||
Yuya Nishihara
|
r34133 | >>> u.configbytes(s, b'unknown', b'7 MB') | ||
Bryan O'Sullivan
|
r19065 | 7340032 | ||
Yuya Nishihara
|
r34133 | >>> u.setconfig(s, b'invalid', b'somevalue') | ||
>>> u.configbytes(s, b'invalid') | ||||
Bryan O'Sullivan
|
r19065 | Traceback (most recent call last): | ||
... | ||||
ConfigError: foo.invalid is not a byte quantity ('somevalue') | ||||
""" | ||||
r33060 | value = self._config(section, name, default, untrusted) | |||
if value is _unset: | ||||
r32962 | if default is _unset: | |||
default = 0 | ||||
Bryan O'Sullivan
|
r19195 | value = default | ||
Augie Fackler
|
r33586 | if not isinstance(value, bytes): | ||
r32962 | return value | |||
Bryan O'Sullivan
|
r19065 | try: | ||
Bryan O'Sullivan
|
r19195 | return util.sizetoint(value) | ||
except error.ParseError: | ||||
Bryan O'Sullivan
|
r19065 | raise error.ConfigError(_("%s.%s is not a byte quantity ('%s')") | ||
Bryan O'Sullivan
|
r19195 | % (section, name, value)) | ||
Bryan O'Sullivan
|
r19065 | |||
r32963 | def configlist(self, section, name, default=_unset, untrusted=False): | |||
Sune Foldager
|
r14171 | """parse a configuration element as a list of comma/space separated | ||
strings | ||||
Yuya Nishihara
|
r34133 | >>> u = ui(); s = b'foo' | ||
>>> u.setconfig(s, b'list1', b'this,is "a small" ,test') | ||||
>>> u.configlist(s, b'list1') | ||||
Sune Foldager
|
r14171 | ['this', 'is', 'a small', 'test'] | ||
Augie Fackler
|
r34958 | >>> u.setconfig(s, b'list2', b'this, is "a small" , test ') | ||
>>> u.configlist(s, b'list2') | ||||
['this', 'is', 'a small', 'test'] | ||||
Sune Foldager
|
r14171 | """ | ||
Jun Wu
|
r31481 | # default is not always a list | ||
r32963 | v = self.configwith(config.parselist, section, name, default, | |||
Jun Wu
|
r31481 | 'list', untrusted) | ||
r32963 | if isinstance(v, bytes): | |||
return config.parselist(v) | ||||
elif v is None: | ||||
return [] | ||||
return v | ||||
Thomas Arendsen Hein
|
r2499 | |||
r32964 | def configdate(self, section, name, default=_unset, untrusted=False): | |||
Boris Feld
|
r32408 | """parse a configuration element as a tuple of ints | ||
Yuya Nishihara
|
r34133 | >>> u = ui(); s = b'foo' | ||
>>> u.setconfig(s, b'date', b'0 0') | ||||
>>> u.configdate(s, b'date') | ||||
Boris Feld
|
r32408 | (0, 0) | ||
""" | ||||
if self.config(section, name, default, untrusted): | ||||
Boris Feld
|
r32462 | return self.configwith(util.parsedate, section, name, default, | ||
Boris Feld
|
r32408 | 'date', untrusted) | ||
r32964 | if default is _unset: | |||
return None | ||||
Boris Feld
|
r32408 | return default | ||
Bryan O'Sullivan
|
r27696 | def hasconfig(self, section, name, untrusted=False): | ||
return self._data(untrusted).hasitem(section, name) | ||||
Bryan O'Sullivan
|
r4487 | def has_section(self, section, untrusted=False): | ||
Vadim Gelfer
|
r2343 | '''tell whether section exists in config.''' | ||
Matt Mackall
|
r8199 | return section in self._data(untrusted) | ||
Alexis S. L. Carvalho
|
r3552 | |||
Gregory Szorc
|
r27253 | def configitems(self, section, untrusted=False, ignoresub=False): | ||
Matt Mackall
|
r8199 | items = self._data(untrusted).items(section) | ||
Gregory Szorc
|
r27253 | if ignoresub: | ||
newitems = {} | ||||
for k, v in items: | ||||
if ':' not in k: | ||||
newitems[k] = v | ||||
items = newitems.items() | ||||
Matt Mackall
|
r8204 | if self.debugflag and not untrusted and self._reportuntrusted: | ||
Dirkjan Ochtman
|
r8222 | for k, v in self._ucfg.items(section): | ||
Matt Mackall
|
r8203 | if self._tcfg.get(section, k) != v: | ||
David Soria Parra
|
r14708 | self.debug("ignoring untrusted configuration option " | ||
"%s.%s = %s\n" % (section, k, v)) | ||||
Matt Mackall
|
r8144 | return items | ||
mpm@selenic.com
|
r285 | |||
Alexis S. L. Carvalho
|
r3552 | def walkconfig(self, untrusted=False): | ||
Matt Mackall
|
r8203 | cfg = self._data(untrusted) | ||
for section in cfg.sections(): | ||||
Alexis S. L. Carvalho
|
r3552 | for name, value in self.configitems(section, untrusted): | ||
Martin Geisler
|
r13576 | yield section, name, value | ||
Bryan O'Sullivan
|
r1028 | |||
"Yann E. MORIN"
|
r14372 | def plain(self, feature=None): | ||
Dan Villiom Podlaski Christiansen
|
r11325 | '''is plain mode active? | ||
Brodie Rao
|
r13849 | Plain mode means that all configuration variables which affect | ||
the behavior and output of Mercurial should be | ||||
ignored. Additionally, the output should be stable, | ||||
reproducible and suitable for use in scripts or applications. | ||||
The only way to trigger plain mode is by setting either the | ||||
`HGPLAIN' or `HGPLAINEXCEPT' environment variables. | ||||
Dan Villiom Podlaski Christiansen
|
r11325 | |||
"Yann E. MORIN"
|
r14372 | The return value can either be | ||
- False if HGPLAIN is not set, or feature is in HGPLAINEXCEPT | ||||
Yuya Nishihara
|
r35180 | - False if feature is disabled by default and not included in HGPLAIN | ||
"Yann E. MORIN"
|
r14372 | - True otherwise | ||
Dan Villiom Podlaski Christiansen
|
r11325 | ''' | ||
Pulkit Goyal
|
r30277 | if ('HGPLAIN' not in encoding.environ and | ||
'HGPLAINEXCEPT' not in encoding.environ): | ||||
Brodie Rao
|
r13849 | return False | ||
Pulkit Goyal
|
r30277 | exceptions = encoding.environ.get('HGPLAINEXCEPT', | ||
'').strip().split(',') | ||||
Yuya Nishihara
|
r35180 | # TODO: add support for HGPLAIN=+feature,-feature syntax | ||
if '+strictflags' not in encoding.environ.get('HGPLAIN', '').split(','): | ||||
exceptions.append('strictflags') | ||||
"Yann E. MORIN"
|
r14372 | if feature and exceptions: | ||
return feature not in exceptions | ||||
return True | ||||
Brodie Rao
|
r10455 | |||
Boris Feld
|
r34850 | def username(self, acceptempty=False): | ||
Thomas Arendsen Hein
|
r1985 | """Return default username to be used in commits. | ||
Searched in this order: $HGUSER, [ui] section of hgrcs, $EMAIL | ||||
and stop searching if one of these is set. | ||||
Boris Feld
|
r34850 | If not found and acceptempty is True, returns None. | ||
Benoit Boissinot
|
r6862 | If not found and ui.askusername is True, ask the user, else use | ||
($LOGNAME or $USER or $LNAME or $USERNAME) + "@full.hostname". | ||||
Boris Feld
|
r34850 | If no username could be found, raise an Abort error. | ||
Thomas Arendsen Hein
|
r1985 | """ | ||
Pulkit Goyal
|
r30277 | user = encoding.environ.get("HGUSER") | ||
Thomas Arendsen Hein
|
r1985 | if user is None: | ||
David Demelier
|
r33329 | user = self.config("ui", "username") | ||
Chad Dombrova
|
r11225 | if user is not None: | ||
user = os.path.expandvars(user) | ||||
Thomas Arendsen Hein
|
r1985 | if user is None: | ||
Pulkit Goyal
|
r30277 | user = encoding.environ.get("EMAIL") | ||
Boris Feld
|
r34850 | if user is None and acceptempty: | ||
return user | ||||
Benoit Boissinot
|
r6862 | if user is None and self.configbool("ui", "askusername"): | ||
Martin Geisler
|
r7600 | user = self.prompt(_("enter a commit username:"), default=None) | ||
Martin Geisler
|
r9613 | if user is None and not self.interactive(): | ||
Benoit Boissinot
|
r3721 | try: | ||
user = '%s@%s' % (util.getuser(), socket.getfqdn()) | ||||
Martin Geisler
|
r16940 | self.warn(_("no username found, using '%s' instead\n") % user) | ||
Benoit Boissinot
|
r3721 | except KeyError: | ||
Thomas Arendsen Hein
|
r4044 | pass | ||
if not user: | ||||
Pierre-Yves David
|
r26587 | raise error.Abort(_('no username supplied'), | ||
timeless
|
r28962 | hint=_("use 'hg config --edit' " | ||
Matt Mackall
|
r20580 | 'to set your username')) | ||
Matt Mackall
|
r6351 | if "\n" in user: | ||
Pierre-Yves David
|
r26587 | raise error.Abort(_("username %s contains a newline\n") | ||
% repr(user)) | ||||
Thomas Arendsen Hein
|
r1985 | return user | ||
Matt Mackall
|
r608 | |||
Thomas Arendsen Hein
|
r1129 | def shortuser(self, user): | ||
"""Return a short representation of a user name or email address.""" | ||||
Matt Mackall
|
r10282 | if not self.verbose: | ||
user = util.shortuser(user) | ||||
Thomas Arendsen Hein
|
r1129 | return user | ||
Vadim Gelfer
|
r2494 | def expandpath(self, loc, default=None): | ||
Thomas Arendsen Hein
|
r1892 | """Return repository location relative to cwd or from [paths]""" | ||
Gregory Szorc
|
r26189 | try: | ||
p = self.paths.getpath(loc) | ||||
if p: | ||||
return p.rawloc | ||||
except error.RepoError: | ||||
pass | ||||
if default: | ||||
try: | ||||
p = self.paths.getpath(default) | ||||
if p: | ||||
return p.rawloc | ||||
except error.RepoError: | ||||
pass | ||||
Gregory Szorc
|
r24250 | return loc | ||
@util.propertycache | ||||
def paths(self): | ||||
return paths(self) | ||||
mpm@selenic.com
|
r506 | |||
Gregory Szorc
|
r27106 | def pushbuffer(self, error=False, subproc=False, labeled=False): | ||
Mads Kiilerich
|
r23139 | """install a buffer to capture standard output of the ui object | ||
Pierre-Yves David
|
r21132 | |||
Pierre-Yves David
|
r24848 | If error is True, the error output will be captured too. | ||
If subproc is True, output from subprocesses (typically hooks) will be | ||||
Gregory Szorc
|
r27106 | captured too. | ||
Brodie Rao
|
r10815 | |||
If labeled is True, any labels associated with buffered | ||||
output will be handled. By default, this has no effect | ||||
on the output returned, but extensions and GUI tools may | ||||
handle this argument and returned styled output. If output | ||||
is being buffered so it can be captured and parsed or | ||||
processed, labeled should not be set to True. | ||||
Gregory Szorc
|
r27106 | """ | ||
Matt Mackall
|
r8202 | self._buffers.append([]) | ||
Gregory Szorc
|
r27106 | self._bufferstates.append((error, subproc, labeled)) | ||
self._bufferapplylabels = labeled | ||||
Matt Mackall
|
r3737 | |||
Gregory Szorc
|
r27109 | def popbuffer(self): | ||
'''pop the last buffer and return the buffered output''' | ||||
Pierre-Yves David
|
r21132 | self._bufferstates.pop() | ||
Gregory Szorc
|
r27106 | if self._bufferstates: | ||
self._bufferapplylabels = self._bufferstates[-1][2] | ||||
else: | ||||
self._bufferapplylabels = None | ||||
Matt Mackall
|
r8202 | return "".join(self._buffers.pop()) | ||
Matt Mackall
|
r3737 | |||
Brodie Rao
|
r10815 | def write(self, *args, **opts): | ||
'''write args to output | ||||
Pierre-Yves David
|
r31091 | By default, this method simply writes to the buffer or stdout. | ||
Color mode can be set on the UI class to have the output decorated | ||||
with color modifier before being written to stdout. | ||||
Brodie Rao
|
r10815 | |||
Pierre-Yves David
|
r31091 | The color used is controlled by an optional keyword argument, "label". | ||
This should be a string containing label names separated by space. | ||||
Label names take the form of "topic.type". For example, ui.debug() | ||||
issues a label of "ui.debug". | ||||
Brodie Rao
|
r10815 | |||
When labeling output for a specific command, a label of | ||||
"cmdname.type" is recommended. For example, status issues | ||||
a label of "status.modified" for modified files. | ||||
''' | ||||
timeless
|
r28633 | if self._buffers and not opts.get('prompt', False): | ||
Pierre-Yves David
|
r31091 | if self._bufferapplylabels: | ||
label = opts.get('label', '') | ||||
self._buffers[-1].extend(self.label(a, label) for a in args) | ||||
else: | ||||
self._buffers[-1].extend(args) | ||||
elif self._colormode == 'win32': | ||||
# windows color printing is its own can of crab, defer to | ||||
# the color module and that is it. | ||||
Pierre-Yves David
|
r31114 | color.win32print(self, self._write, *args, **opts) | ||
Matt Mackall
|
r3737 | else: | ||
Pierre-Yves David
|
r31091 | msgs = args | ||
if self._colormode is not None: | ||||
label = opts.get('label', '') | ||||
msgs = [self.label(a, label) for a in args] | ||||
self._write(*msgs, **opts) | ||||
Pierre-Yves David
|
r31090 | |||
def _write(self, *msgs, **opts): | ||||
Yuya Nishihara
|
r31128 | self._progclear() | ||
# opencode timeblockedsection because this is a critical path | ||||
starttime = util.timer() | ||||
try: | ||||
for a in msgs: | ||||
self.fout.write(a) | ||||
Bryan O'Sullivan
|
r31961 | except IOError as err: | ||
raise error.StdioError(err) | ||||
Yuya Nishihara
|
r31128 | finally: | ||
self._blockedtimes['stdio_blocked'] += \ | ||||
(util.timer() - starttime) * 1000 | ||||
mpm@selenic.com
|
r565 | |||
Brodie Rao
|
r10815 | def write_err(self, *args, **opts): | ||
Pierre-Yves David
|
r25499 | self._progclear() | ||
Pierre-Yves David
|
r31092 | if self._bufferstates and self._bufferstates[-1][0]: | ||
Pierre-Yves David
|
r31094 | self.write(*args, **opts) | ||
elif self._colormode == 'win32': | ||||
# windows color printing is its own can of crab, defer to | ||||
# the color module and that is it. | ||||
Pierre-Yves David
|
r31114 | color.win32print(self, self._write_err, *args, **opts) | ||
Pierre-Yves David
|
r31094 | else: | ||
msgs = args | ||||
if self._colormode is not None: | ||||
label = opts.get('label', '') | ||||
msgs = [self.label(a, label) for a in args] | ||||
self._write_err(*msgs, **opts) | ||||
Pierre-Yves David
|
r31093 | |||
def _write_err(self, *msgs, **opts): | ||||
Benoit Boissinot
|
r1989 | try: | ||
Simon Farnsworth
|
r30978 | with self.timeblockedsection('stdio'): | ||
if not getattr(self.fout, 'closed', False): | ||||
self.fout.flush() | ||||
Pierre-Yves David
|
r31093 | for a in msgs: | ||
Simon Farnsworth
|
r30978 | self.ferr.write(a) | ||
# stderr may be buffered under win32 when redirected to files, | ||||
# including stdout. | ||||
if not getattr(self.ferr, 'closed', False): | ||||
self.ferr.flush() | ||||
Gregory Szorc
|
r25660 | except IOError as inst: | ||
Gregory Szorc
|
r33859 | if inst.errno not in (errno.EPIPE, errno.EIO, errno.EBADF): | ||
raise error.StdioError(inst) | ||||
mpm@selenic.com
|
r565 | |||
Vadim Gelfer
|
r1837 | def flush(self): | ||
Simon Farnsworth
|
r30978 | # opencode timeblockedsection because this is a critical path | ||
starttime = util.timer() | ||||
try: | ||||
Bryan O'Sullivan
|
r31963 | try: | ||
self.fout.flush() | ||||
except IOError as err: | ||||
Gregory Szorc
|
r33859 | if err.errno not in (errno.EPIPE, errno.EIO, errno.EBADF): | ||
raise error.StdioError(err) | ||||
Bryan O'Sullivan
|
r31963 | finally: | ||
try: | ||||
self.ferr.flush() | ||||
except IOError as err: | ||||
Gregory Szorc
|
r33859 | if err.errno not in (errno.EPIPE, errno.EIO, errno.EBADF): | ||
raise error.StdioError(err) | ||||
Simon Farnsworth
|
r30978 | finally: | ||
self._blockedtimes['stdio_blocked'] += \ | ||||
(util.timer() - starttime) * 1000 | ||||
Vadim Gelfer
|
r1837 | |||
Matt Mackall
|
r16751 | def _isatty(self, fh): | ||
Jun Wu
|
r33499 | if self.configbool('ui', 'nontty'): | ||
Matt Mackall
|
r16751 | return False | ||
return util.isatty(fh) | ||||
Vadim Gelfer
|
r1837 | |||
Augie Fackler
|
r31026 | def disablepager(self): | ||
self._disablepager = True | ||||
Augie Fackler
|
r30994 | |||
Augie Fackler
|
r30992 | def pager(self, command): | ||
"""Start a pager for subsequent command output. | ||||
Commands which produce a long stream of output should call | ||||
this function to activate the user's preferred pagination | ||||
mechanism (which may be no pager). Calling this function | ||||
precludes any future use of interactive functionality, such as | ||||
prompting the user or activating curses. | ||||
Args: | ||||
command: The full, non-aliased name of the command. That is, "log" | ||||
not "history, "summary" not "summ", etc. | ||||
""" | ||||
Augie Fackler
|
r31026 | if (self._disablepager | ||
FUJIWARA Katsunori
|
r33622 | or self.pageractive): | ||
# how pager should do is already determined | ||||
return | ||||
if not command.startswith('internal-always-') and ( | ||||
# explicit --pager=on (= 'internal-always-' prefix) should | ||||
# take precedence over disabling factors below | ||||
command in self.configlist('pager', 'ignore') | ||||
Jun Wu
|
r33499 | or not self.configbool('ui', 'paginate') | ||
Augie Fackler
|
r30997 | or not self.configbool('pager', 'attend-' + command, True) | ||
Augie Fackler
|
r30992 | # TODO: if we want to allow HGPLAINEXCEPT=pager, | ||
# formatted() will need some adjustment. | ||||
or not self.formatted() | ||||
or self.plain() | ||||
Jun Wu
|
r34023 | or self._buffers | ||
Augie Fackler
|
r30992 | # TODO: expose debugger-enabled on the UI object | ||
Augie Fackler
|
r31341 | or '--debugger' in pycompat.sysargv): | ||
Augie Fackler
|
r30992 | # We only want to paginate if the ui appears to be | ||
# interactive, the user didn't say HGPLAIN or | ||||
# HGPLAINEXCEPT=pager, and the user didn't specify --debug. | ||||
return | ||||
Yuya Nishihara
|
r32078 | pagercmd = self.config('pager', 'pager', rcutil.fallbackpager) | ||
Yuya Nishihara
|
r31079 | if not pagercmd: | ||
return | ||||
Jun Wu
|
r31954 | pagerenv = {} | ||
for name, value in rcutil.defaultpagerenv().items(): | ||||
if name not in encoding.environ: | ||||
pagerenv[name] = value | ||||
Yuya Nishihara
|
r31079 | self.debug('starting pager for command %r\n' % command) | ||
Yuya Nishihara
|
r31490 | self.flush() | ||
Matt Harbison
|
r31690 | |||
wasformatted = self.formatted() | ||||
Augie Fackler
|
r30992 | if util.safehasattr(signal, "SIGPIPE"): | ||
signal.signal(signal.SIGPIPE, _catchterm) | ||||
Jun Wu
|
r31954 | if self._runpager(pagercmd, pagerenv): | ||
Matt Harbison
|
r31690 | self.pageractive = True | ||
# Preserve the formatted-ness of the UI. This is important | ||||
# because we mess with stdout, which might confuse | ||||
# auto-detection of things being formatted. | ||||
self.setconfig('ui', 'formatted', wasformatted, 'pager') | ||||
self.setconfig('ui', 'interactive', False, 'pager') | ||||
Matt Harbison
|
r31691 | |||
# If pagermode differs from color.mode, reconfigure color now that | ||||
# pageractive is set. | ||||
cm = self._colormode | ||||
if cm != self.config('color', 'pagermode', cm): | ||||
color.setup(self) | ||||
Matt Harbison
|
r31690 | else: | ||
# If the pager can't be spawned in dispatch when --pager=on is | ||||
# given, don't try again when the command runs, to avoid a duplicate | ||||
# warning about a missing pager command. | ||||
self.disablepager() | ||||
Augie Fackler
|
r30992 | |||
Jun Wu
|
r31954 | def _runpager(self, command, env=None): | ||
Augie Fackler
|
r30992 | """Actually start the pager and set up file descriptors. | ||
This is separate in part so that extensions (like chg) can | ||||
override how a pager is invoked. | ||||
""" | ||||
Augie Fackler
|
r31479 | if command == 'cat': | ||
# Save ourselves some work. | ||||
Matt Harbison
|
r31690 | return False | ||
Augie Fackler
|
r31478 | # If the command doesn't contain any of these characters, we | ||
# assume it's a binary and exec it directly. This means for | ||||
# simple pager command configurations, we can degrade | ||||
# gracefully and tell the user about their broken pager. | ||||
shell = any(c in command for c in "|&;<>()$`\\\"' \t\n*?[#~=%") | ||||
Matt Harbison
|
r31624 | |||
Jun Wu
|
r34646 | if pycompat.iswindows and not shell: | ||
Matt Harbison
|
r31624 | # Window's built-in `more` cannot be invoked with shell=False, but | ||
# its `more.com` can. Hide this implementation detail from the | ||||
# user so we can also get sane bad PAGER behavior. MSYS has | ||||
# `more.exe`, so do a cmd.exe style resolution of the executable to | ||||
# determine which one to use. | ||||
fullcmd = util.findexe(command) | ||||
if not fullcmd: | ||||
self.warn(_("missing pager command '%s', skipping pager\n") | ||||
% command) | ||||
Matt Harbison
|
r31690 | return False | ||
Matt Harbison
|
r31624 | |||
command = fullcmd | ||||
Augie Fackler
|
r31478 | try: | ||
pager = subprocess.Popen( | ||||
command, shell=shell, bufsize=-1, | ||||
close_fds=util.closefds, stdin=subprocess.PIPE, | ||||
Jun Wu
|
r31954 | stdout=util.stdout, stderr=util.stderr, | ||
env=util.shellenviron(env)) | ||||
Augie Fackler
|
r31478 | except OSError as e: | ||
if e.errno == errno.ENOENT and not shell: | ||||
self.warn(_("missing pager command '%s', skipping pager\n") | ||||
% command) | ||||
Matt Harbison
|
r31690 | return False | ||
Augie Fackler
|
r31478 | raise | ||
Augie Fackler
|
r30992 | |||
# back up original file descriptors | ||||
stdoutfd = os.dup(util.stdout.fileno()) | ||||
stderrfd = os.dup(util.stderr.fileno()) | ||||
os.dup2(pager.stdin.fileno(), util.stdout.fileno()) | ||||
if self._isatty(util.stderr): | ||||
os.dup2(pager.stdin.fileno(), util.stderr.fileno()) | ||||
Bryan O'Sullivan
|
r31958 | @self.atexit | ||
Augie Fackler
|
r30992 | def killpager(): | ||
if util.safehasattr(signal, "SIGINT"): | ||||
signal.signal(signal.SIGINT, signal.SIG_IGN) | ||||
# restore original fds, closing pager.stdin copies in the process | ||||
os.dup2(stdoutfd, util.stdout.fileno()) | ||||
os.dup2(stderrfd, util.stderr.fileno()) | ||||
pager.stdin.close() | ||||
pager.wait() | ||||
Matt Harbison
|
r31690 | return True | ||
Saurabh Singh
|
r34883 | @property | ||
def _exithandlers(self): | ||||
return _reqexithandlers | ||||
Bryan O'Sullivan
|
r31956 | def atexit(self, func, *args, **kwargs): | ||
'''register a function to run after dispatching a request | ||||
Handlers do not stay registered across request boundaries.''' | ||||
self._exithandlers.append((func, args, kwargs)) | ||||
return func | ||||
Simon Farnsworth
|
r28542 | def interface(self, feature): | ||
"""what interface to use for interactive console features? | ||||
The interface is controlled by the value of `ui.interface` but also by | ||||
the value of feature-specific configuration. For example: | ||||
ui.interface.histedit = text | ||||
ui.interface.chunkselector = curses | ||||
Here the features are "histedit" and "chunkselector". | ||||
The configuration above means that the default interfaces for commands | ||||
is curses, the interface for histedit is text and the interface for | ||||
selecting chunk is crecord (the best curses interface available). | ||||
Mads Kiilerich
|
r30332 | Consider the following example: | ||
Simon Farnsworth
|
r28542 | ui.interface = curses | ||
ui.interface.histedit = text | ||||
Then histedit will use the text interface and chunkselector will use | ||||
the default curses interface (crecord at the moment). | ||||
""" | ||||
alldefaults = frozenset(["text", "curses"]) | ||||
featureinterfaces = { | ||||
"chunkselector": [ | ||||
"text", | ||||
"curses", | ||||
] | ||||
} | ||||
# Feature-specific interface | ||||
if feature not in featureinterfaces.keys(): | ||||
# Programming error, not user error | ||||
raise ValueError("Unknown feature requested %s" % feature) | ||||
availableinterfaces = frozenset(featureinterfaces[feature]) | ||||
if alldefaults > availableinterfaces: | ||||
# Programming error, not user error. We need a use case to | ||||
# define the right thing to do here. | ||||
raise ValueError( | ||||
"Feature %s does not handle all default interfaces" % | ||||
feature) | ||||
if self.plain(): | ||||
return "text" | ||||
# Default interface for all the features | ||||
defaultinterface = "text" | ||||
Jun Wu
|
r33499 | i = self.config("ui", "interface") | ||
Simon Farnsworth
|
r28542 | if i in alldefaults: | ||
defaultinterface = i | ||||
choseninterface = defaultinterface | ||||
Boris Feld
|
r34617 | f = self.config("ui", "interface.%s" % feature) | ||
Simon Farnsworth
|
r28542 | if f in availableinterfaces: | ||
choseninterface = f | ||||
if i is not None and defaultinterface != i: | ||||
if f is not None: | ||||
self.warn(_("invalid value for ui.interface: %s\n") % | ||||
(i,)) | ||||
else: | ||||
self.warn(_("invalid value for ui.interface: %s (using %s)\n") % | ||||
(i, choseninterface)) | ||||
if f is not None and choseninterface != f: | ||||
self.warn(_("invalid value for ui.interface.%s: %s (using %s)\n") % | ||||
(feature, f, choseninterface)) | ||||
return choseninterface | ||||
Matt Mackall
|
r8208 | def interactive(self): | ||
Dan Villiom Podlaski Christiansen
|
r11325 | '''is interactive input allowed? | ||
An interactive session is a session where input can be reasonably read | ||||
from `sys.stdin'. If this function returns false, any attempt to read | ||||
from stdin should fail with an error, unless a sensible default has been | ||||
specified. | ||||
Interactiveness is triggered by the value of the `ui.interactive' | ||||
configuration variable or - if it is unset - when `sys.stdin' points | ||||
to a terminal device. | ||||
This function refers to input only; for output, see `ui.formatted()'. | ||||
''' | ||||
r33061 | i = self.configbool("ui", "interactive") | |||
Patrick Mezard
|
r8538 | if i is None: | ||
Idan Kamara
|
r14515 | # some environments replace stdin without implementing isatty | ||
# usually those are non-interactive | ||||
Matt Mackall
|
r16751 | return self._isatty(self.fin) | ||
Ronny Pfannschmidt
|
r10077 | |||
Patrick Mezard
|
r8538 | return i | ||
Matt Mackall
|
r8208 | |||
Augie Fackler
|
r12689 | def termwidth(self): | ||
'''how wide is the terminal in columns? | ||||
''' | ||||
Pulkit Goyal
|
r30277 | if 'COLUMNS' in encoding.environ: | ||
Augie Fackler
|
r12689 | try: | ||
Pulkit Goyal
|
r30277 | return int(encoding.environ['COLUMNS']) | ||
Augie Fackler
|
r12689 | except ValueError: | ||
pass | ||||
Yuya Nishihara
|
r30314 | return scmutil.termsize(self)[0] | ||
Augie Fackler
|
r12689 | |||
Dan Villiom Podlaski Christiansen
|
r11324 | def formatted(self): | ||
Dan Villiom Podlaski Christiansen
|
r11325 | '''should formatted output be used? | ||
It is often desirable to format the output to suite the output medium. | ||||
Examples of this are truncating long lines or colorizing messages. | ||||
However, this is not often not desirable when piping output into other | ||||
utilities, e.g. `grep'. | ||||
Formatted output is triggered by the value of the `ui.formatted' | ||||
configuration variable or - if it is unset - when `sys.stdout' points | ||||
to a terminal device. Please note that `ui.formatted' should be | ||||
considered an implementation detail; it is not intended for use outside | ||||
Mercurial or its extensions. | ||||
This function refers to output only; for input, see `ui.interactive()'. | ||||
This function always returns false when in plain mode, see `ui.plain()'. | ||||
''' | ||||
Dan Villiom Podlaski Christiansen
|
r11324 | if self.plain(): | ||
return False | ||||
Jun Wu
|
r33499 | i = self.configbool("ui", "formatted") | ||
Dan Villiom Podlaski Christiansen
|
r11324 | if i is None: | ||
Idan Kamara
|
r14515 | # some environments replace stdout without implementing isatty | ||
# usually those are non-interactive | ||||
Matt Mackall
|
r16751 | return self._isatty(self.fout) | ||
Dan Villiom Podlaski Christiansen
|
r11324 | |||
return i | ||||
Dirkjan Ochtman
|
r5337 | def _readline(self, prompt=''): | ||
Matt Mackall
|
r16751 | if self._isatty(self.fin): | ||
Bryan O'Sullivan
|
r5036 | try: | ||
# magically add command line editing support, where | ||||
# available | ||||
import readline | ||||
# force demandimport to really load the module | ||||
readline.read_history_file | ||||
Brendan Cully
|
r7496 | # windows sometimes raises something other than ImportError | ||
except Exception: | ||||
Bryan O'Sullivan
|
r5036 | pass | ||
Idan Kamara
|
r14614 | |||
Idan Kamara
|
r15000 | # call write() so output goes through subclassed implementation | ||
# e.g. color extension on Windows | ||||
timeless
|
r28633 | self.write(prompt, prompt=True) | ||
Kostia Balytskyi
|
r33662 | self.flush() | ||
Idan Kamara
|
r15000 | |||
Yuya Nishihara
|
r22291 | # prompt ' ' must exist; otherwise readline may delete entire line | ||
# - http://bugs.python.org/issue12833 | ||||
Simon Farnsworth
|
r30978 | with self.timeblockedsection('stdio'): | ||
Augie Fackler
|
r33838 | line = util.bytesinput(self.fin, self.fout, r' ') | ||
Idan Kamara
|
r14614 | |||
Steve Borho
|
r5613 | # When stdin is in binary mode on Windows, it can cause | ||
# raw_input() to emit an extra trailing carriage return | ||||
Yuya Nishihara
|
r31775 | if pycompat.oslinesep == '\r\n' and line and line[-1] == '\r': | ||
Steve Borho
|
r5613 | line = line[:-1] | ||
return line | ||||
Bryan O'Sullivan
|
r5036 | |||
Simon Heimberg
|
r9048 | def prompt(self, msg, default="y"): | ||
"""Prompt user with msg, read response. | ||||
If ui is not interactive, the default is returned. | ||||
Kirill Smelkov
|
r5751 | """ | ||
Matt Mackall
|
r8208 | if not self.interactive(): | ||
Yuya Nishihara
|
r28039 | self.write(msg, ' ', default or '', "\n") | ||
Peter Arrenbrecht
|
r7320 | return default | ||
Simon Heimberg
|
r9048 | try: | ||
Idan Kamara
|
r15053 | r = self._readline(self.label(msg, 'ui.prompt')) | ||
Simon Heimberg
|
r9048 | if not r: | ||
Mads Kiilerich
|
r22589 | r = default | ||
Yuya Nishihara
|
r23053 | if self.configbool('ui', 'promptecho'): | ||
Mads Kiilerich
|
r22589 | self.write(r, "\n") | ||
Simon Heimberg
|
r9048 | return r | ||
except EOFError: | ||||
Siddharth Agarwal
|
r26896 | raise error.ResponseExpected() | ||
Simon Heimberg
|
r9048 | |||
FUJIWARA Katsunori
|
r20265 | @staticmethod | ||
def extractchoices(prompt): | ||||
"""Extract prompt message and list of choices from specified prompt. | ||||
This returns tuple "(message, choices)", and "choices" is the | ||||
list of tuple "(response character, text without &)". | ||||
Matt Mackall
|
r27392 | |||
Yuya Nishihara
|
r34133 | >>> ui.extractchoices(b"awake? $$ &Yes $$ &No") | ||
Matt Mackall
|
r27392 | ('awake? ', [('y', 'Yes'), ('n', 'No')]) | ||
Yuya Nishihara
|
r34133 | >>> ui.extractchoices(b"line\\nbreak? $$ &Yes $$ &No") | ||
Matt Mackall
|
r27392 | ('line\\nbreak? ', [('y', 'Yes'), ('n', 'No')]) | ||
Yuya Nishihara
|
r34133 | >>> ui.extractchoices(b"want lots of $$money$$?$$Ye&s$$N&o") | ||
Matt Mackall
|
r27392 | ('want lots of $$money$$?', [('s', 'Yes'), ('o', 'No')]) | ||
FUJIWARA Katsunori
|
r20265 | """ | ||
Matt Mackall
|
r27392 | |||
# Sadly, the prompt string may have been built with a filename | ||||
# containing "$$" so let's try to find the first valid-looking | ||||
# prompt to start parsing. Sadly, we also can't rely on | ||||
# choices containing spaces, ASCII, or basically anything | ||||
# except an ampersand followed by a character. | ||||
Pulkit Goyal
|
r33096 | m = re.match(br'(?s)(.+?)\$\$([^\$]*&[^ \$].*)', prompt) | ||
Matt Mackall
|
r27392 | msg = m.group(1) | ||
choices = [p.strip(' ') for p in m.group(2).split('$$')] | ||||
Augie Fackler
|
r33680 | def choicetuple(s): | ||
ampidx = s.index('&') | ||||
return s[ampidx + 1:ampidx + 2].lower(), s.replace('&', '', 1) | ||||
return (msg, [choicetuple(s) for s in choices]) | ||||
FUJIWARA Katsunori
|
r20265 | |||
Matt Mackall
|
r19226 | def promptchoice(self, prompt, default=0): | ||
"""Prompt user with a message, read response, and ensure it matches | ||||
one of the provided choices. The prompt is formatted as follows: | ||||
"would you like fries with that (Yn)? $$ &Yes $$ &No" | ||||
The index of the choice is returned. Responses are case | ||||
insensitive. If ui is not interactive, the default is | ||||
returned. | ||||
Simon Heimberg
|
r9048 | """ | ||
Matt Mackall
|
r19226 | |||
FUJIWARA Katsunori
|
r20265 | msg, choices = self.extractchoices(prompt) | ||
resps = [r for r, t in choices] | ||||
Thomas Arendsen Hein
|
r5671 | while True: | ||
Simon Heimberg
|
r9048 | r = self.prompt(msg, resps[default]) | ||
if r.lower() in resps: | ||||
return resps.index(r.lower()) | ||||
self.write(_("unrecognized response\n")) | ||||
Vadim Gelfer
|
r2281 | def getpass(self, prompt=None, default=None): | ||
Matt Mackall
|
r10282 | if not self.interactive(): | ||
return default | ||||
Steve Borho
|
r7798 | try: | ||
Matt Mackall
|
r19880 | self.write_err(self.label(prompt or _('password: '), 'ui.prompt')) | ||
Yuya Nishihara
|
r21195 | # disable getpass() only if explicitly specified. it's still valid | ||
# to interact with tty even if fin is not a tty. | ||||
Simon Farnsworth
|
r30978 | with self.timeblockedsection('stdio'): | ||
if self.configbool('ui', 'nontty'): | ||||
l = self.fin.readline() | ||||
if not l: | ||||
raise EOFError | ||||
return l.rstrip('\n') | ||||
else: | ||||
return getpass.getpass('') | ||||
Steve Borho
|
r7798 | except EOFError: | ||
Siddharth Agarwal
|
r26896 | raise error.ResponseExpected() | ||
Brodie Rao
|
r10815 | def status(self, *msg, **opts): | ||
'''write status message to output (if ui.quiet is False) | ||||
This adds an output label of "ui.status". | ||||
''' | ||||
Matt Mackall
|
r10282 | if not self.quiet: | ||
Augie Fackler
|
r31177 | opts[r'label'] = opts.get(r'label', '') + ' ui.status' | ||
Brodie Rao
|
r10815 | self.write(*msg, **opts) | ||
def warn(self, *msg, **opts): | ||||
'''write warning message to output (stderr) | ||||
This adds an output label of "ui.warning". | ||||
''' | ||||
Augie Fackler
|
r31177 | opts[r'label'] = opts.get(r'label', '') + ' ui.warning' | ||
Brodie Rao
|
r10815 | self.write_err(*msg, **opts) | ||
def note(self, *msg, **opts): | ||||
'''write note to output (if ui.verbose is True) | ||||
This adds an output label of "ui.note". | ||||
''' | ||||
Matt Mackall
|
r10282 | if self.verbose: | ||
Augie Fackler
|
r31177 | opts[r'label'] = opts.get(r'label', '') + ' ui.note' | ||
Brodie Rao
|
r10815 | self.write(*msg, **opts) | ||
def debug(self, *msg, **opts): | ||||
'''write debug message to output (if ui.debugflag is True) | ||||
This adds an output label of "ui.debug". | ||||
''' | ||||
Matt Mackall
|
r10282 | if self.debugflag: | ||
Augie Fackler
|
r31177 | opts[r'label'] = opts.get(r'label', '') + ' ui.debug' | ||
Brodie Rao
|
r10815 | self.write(*msg, **opts) | ||
FUJIWARA Katsunori
|
r26750 | |||
Sean Farley
|
r30835 | def edit(self, text, user, extra=None, editform=None, pending=None, | ||
Michael Bolin
|
r34030 | repopath=None, action=None): | ||
if action is None: | ||||
self.develwarn('action is None but will soon be a required ' | ||||
'parameter to ui.edit()') | ||||
Jordi Gutiérrez Hermoso
|
r28635 | extra_defaults = { | ||
'prefix': 'editor', | ||||
'suffix': '.txt', | ||||
} | ||||
Mykola Nikishov
|
r27153 | if extra is not None: | ||
Michael Bolin
|
r34030 | if extra.get('suffix') is not None: | ||
self.develwarn('extra.suffix is not None but will soon be ' | ||||
'ignored by ui.edit()') | ||||
Mykola Nikishov
|
r27153 | extra_defaults.update(extra) | ||
extra = extra_defaults | ||||
Sean Farley
|
r30835 | |||
Michael Bolin
|
r34056 | if action == 'diff': | ||
suffix = '.diff' | ||||
elif action: | ||||
Michael Bolin
|
r34030 | suffix = '.%s.hg.txt' % action | ||
else: | ||||
suffix = extra['suffix'] | ||||
Sean Farley
|
r30848 | rdir = None | ||
Sean Farley
|
r30835 | if self.configbool('experimental', 'editortmpinhg'): | ||
Sean Farley
|
r30848 | rdir = repopath | ||
Mykola Nikishov
|
r27153 | (fd, name) = tempfile.mkstemp(prefix='hg-' + extra['prefix'] + '-', | ||
Michael Bolin
|
r34030 | suffix=suffix, | ||
Sean Farley
|
r30848 | dir=rdir) | ||
Thomas Arendsen Hein
|
r1984 | try: | ||
Yuya Nishihara
|
r31778 | f = os.fdopen(fd, r'wb') | ||
f.write(util.tonativeeol(text)) | ||||
Thomas Arendsen Hein
|
r1984 | f.close() | ||
Alexander Drozdov
|
r20605 | environ = {'HGUSER': user} | ||
Alexander Drozdov
|
r20606 | if 'transplant_source' in extra: | ||
environ.update({'HGREVISION': hex(extra['transplant_source'])}) | ||||
Alexander Drozdov
|
r24687 | for label in ('intermediate-source', 'source', 'rebase_source'): | ||
Alexander Drozdov
|
r20605 | if label in extra: | ||
environ.update({'HGREVISION': extra[label]}) | ||||
break | ||||
FUJIWARA Katsunori
|
r22205 | if editform: | ||
environ.update({'HGEDITFORM': editform}) | ||||
FUJIWARA Katsunori
|
r26750 | if pending: | ||
environ.update({'HG_PENDING': pending}) | ||||
Alexander Drozdov
|
r20605 | |||
Osku Salerma
|
r5660 | editor = self.geteditor() | ||
mpm@selenic.com
|
r207 | |||
Yuya Nishihara
|
r23269 | self.system("%s \"%s\"" % (editor, name), | ||
Alexander Drozdov
|
r20605 | environ=environ, | ||
Simon Farnsworth
|
r30980 | onerr=error.Abort, errprefix=_("edit failed"), | ||
blockedtag='editor') | ||||
Matt Mackall
|
r608 | |||
Yuya Nishihara
|
r31778 | f = open(name, r'rb') | ||
t = util.fromnativeeol(f.read()) | ||||
Thomas Arendsen Hein
|
r1984 | f.close() | ||
finally: | ||||
os.unlink(name) | ||||
Radoslaw "AstralStorm" Szkodzinski
|
r662 | |||
mpm@selenic.com
|
r207 | return t | ||
Vadim Gelfer
|
r2200 | |||
Simon Farnsworth
|
r30979 | def system(self, cmd, environ=None, cwd=None, onerr=None, errprefix=None, | ||
blockedtag=None): | ||||
Yuya Nishihara
|
r23269 | '''execute shell command with appropriate output stream. command | ||
output will be redirected if fout is not stdout. | ||||
Yuya Nishihara
|
r31108 | |||
if command fails and onerr is None, return status, else raise onerr | ||||
object as exception. | ||||
Yuya Nishihara
|
r23269 | ''' | ||
Simon Farnsworth
|
r30979 | if blockedtag is None: | ||
Simon Farnsworth
|
r31535 | # Long cmds tend to be because of an absolute path on cmd. Keep | ||
# the tail end instead | ||||
cmdsuffix = cmd.translate(None, _keepalnum)[-85:] | ||||
blockedtag = 'unknown_system_' + cmdsuffix | ||||
Pierre-Yves David
|
r24848 | out = self.fout | ||
Augie Fackler
|
r25149 | if any(s[1] for s in self._bufferstates): | ||
Pierre-Yves David
|
r24848 | out = self | ||
Simon Farnsworth
|
r30979 | with self.timeblockedsection(blockedtag): | ||
Yuya Nishihara
|
r31108 | rc = self._runsystem(cmd, environ=environ, cwd=cwd, out=out) | ||
if rc and onerr: | ||||
errmsg = '%s %s' % (os.path.basename(cmd.split(None, 1)[0]), | ||||
util.explainexit(rc)[0]) | ||||
if errprefix: | ||||
errmsg = '%s: %s' % (errprefix, errmsg) | ||||
raise onerr(errmsg) | ||||
return rc | ||||
Yuya Nishihara
|
r31107 | |||
Yuya Nishihara
|
r31108 | def _runsystem(self, cmd, environ, cwd, out): | ||
Yuya Nishihara
|
r31107 | """actually execute the given shell command (can be overridden by | ||
extensions like chg)""" | ||||
Yuya Nishihara
|
r31108 | return util.system(cmd, environ=environ, cwd=cwd, out=out) | ||
Yuya Nishihara
|
r23269 | |||
Matt Harbison
|
r18966 | def traceback(self, exc=None, force=False): | ||
'''print exception traceback if traceback printing enabled or forced. | ||||
Vadim Gelfer
|
r2335 | only to call in exception handler. returns true if traceback | ||
printed.''' | ||||
Matt Harbison
|
r18966 | if self.tracebackflag or force: | ||
Matt Harbison
|
r18965 | if exc is None: | ||
exc = sys.exc_info() | ||||
cause = getattr(exc[1], 'cause', None) | ||||
if cause is not None: | ||||
causetb = traceback.format_tb(cause[2]) | ||||
exctb = traceback.format_tb(exc[2]) | ||||
exconly = traceback.format_exception_only(cause[0], cause[1]) | ||||
# exclude frame where 'exc' was chained and rethrown from exctb | ||||
self.write_err('Traceback (most recent call last):\n', | ||||
''.join(exctb[:-1]), | ||||
''.join(causetb), | ||||
''.join(exconly)) | ||||
else: | ||||
Matt Harbison
|
r25568 | output = traceback.format_exception(exc[0], exc[1], exc[2]) | ||
Augie Fackler
|
r31178 | data = r''.join(output) | ||
if pycompat.ispy3: | ||||
enc = pycompat.sysstr(encoding.encoding) | ||||
data = data.encode(enc, errors=r'replace') | ||||
self.write_err(data) | ||||
Matt Harbison
|
r18966 | return self.tracebackflag or force | ||
Osku Salerma
|
r5660 | |||
def geteditor(self): | ||||
'''return editor to use''' | ||||
Pulkit Goyal
|
r30641 | if pycompat.sysplatform == 'plan9': | ||
Steven Stallion
|
r16383 | # vi is the MIPS instruction simulator on Plan 9. We | ||
# instead default to E to plumb commit messages to | ||||
# avoid confusion. | ||||
editor = 'E' | ||||
else: | ||||
editor = 'vi' | ||||
Pulkit Goyal
|
r30277 | return (encoding.environ.get("HGEDITOR") or | ||
Jun Wu
|
r31687 | self.config("ui", "editor", editor)) | ||
Matt Mackall
|
r9153 | |||
Pierre-Yves David
|
r25499 | @util.propertycache | ||
def _progbar(self): | ||||
"""setup the progbar singleton to the ui object""" | ||||
if (self.quiet or self.debugflag | ||||
Jun Wu
|
r33499 | or self.configbool('progress', 'disable') | ||
Pierre-Yves David
|
r25499 | or not progress.shouldprint(self)): | ||
return None | ||||
return getprogbar(self) | ||||
def _progclear(self): | ||||
"""clear progress bar output if any. use it before any output""" | ||||
Mark Thomas
|
r34346 | if not haveprogbar(): # nothing loaded yet | ||
Pierre-Yves David
|
r25499 | return | ||
if self._progbar is not None and self._progbar.printed: | ||||
self._progbar.clear() | ||||
Matt Mackall
|
r9153 | def progress(self, topic, pos, item="", unit="", total=None): | ||
'''show a progress message | ||||
timeless
|
r28598 | By default a textual progress bar will be displayed if an operation | ||
takes too long. 'topic' is the current operation, 'item' is a | ||||
Mads Kiilerich
|
r17424 | non-numeric marker of the current position (i.e. the currently | ||
in-process file), 'pos' is the current numeric position (i.e. | ||||
Brodie Rao
|
r9398 | revision, bytes, etc.), unit is a corresponding unit label, | ||
Matt Mackall
|
r9153 | and total is the highest expected pos. | ||
Augie Fackler
|
r10425 | Multiple nested topics may be active at a time. | ||
All topics should be marked closed by setting pos to None at | ||||
termination. | ||||
Matt Mackall
|
r9153 | ''' | ||
Pierre-Yves David
|
r25499 | if self._progbar is not None: | ||
self._progbar.progress(topic, pos, item=item, unit=unit, | ||||
total=total) | ||||
Pierre-Yves David
|
r25125 | if pos is None or not self.configbool('progress', 'debug'): | ||
Matt Mackall
|
r9153 | return | ||
Brodie Rao
|
r9398 | if unit: | ||
unit = ' ' + unit | ||||
Matt Mackall
|
r9153 | if item: | ||
item = ' ' + item | ||||
if total: | ||||
pct = 100.0 * pos / total | ||||
Augie Fackler
|
r34272 | self.debug('%s:%s %d/%d%s (%4.2f%%)\n' | ||
Brodie Rao
|
r9398 | % (topic, item, pos, total, unit, pct)) | ||
Matt Mackall
|
r9153 | else: | ||
Augie Fackler
|
r34272 | self.debug('%s:%s %d%s\n' % (topic, item, pos, unit)) | ||
Brodie Rao
|
r10815 | |||
Durham Goode
|
r18669 | def log(self, service, *msg, **opts): | ||
Matt Mackall
|
r11984 | '''hook for logging facility extensions | ||
service should be a readily-identifiable subsystem, which will | ||||
allow filtering. | ||||
Augie Fackler
|
r26235 | |||
*msg should be a newline-terminated format string to log, and | ||||
then any values to %-format into that format string. | ||||
**opts currently has no defined meanings. | ||||
Matt Mackall
|
r11984 | ''' | ||
Brodie Rao
|
r10815 | def label(self, msg, label): | ||
'''style msg based on supplied label | ||||
Pierre-Yves David
|
r31087 | If some color mode is enabled, this will add the necessary control | ||
characters to apply such color. In addition, 'debug' color mode adds | ||||
markup showing which label affects a piece of text. | ||||
Brodie Rao
|
r10815 | |||
ui.write(s, 'label') is equivalent to | ||||
ui.write(ui.label(s, 'label')). | ||||
''' | ||||
Pierre-Yves David
|
r31087 | if self._colormode is not None: | ||
return color.colorlabel(self, msg, label) | ||||
Brodie Rao
|
r10815 | return msg | ||
Gregory Szorc
|
r24250 | |||
Pierre-Yves David
|
r29095 | def develwarn(self, msg, stacklevel=1, config=None): | ||
Pierre-Yves David
|
r27274 | """issue a developer warning message | ||
Use 'stacklevel' to report the offender some layers further up in the | ||||
stack. | ||||
""" | ||||
Pierre-Yves David
|
r29095 | if not self.configbool('devel', 'all-warnings'): | ||
Kyle Lippincott
|
r35125 | if config is None or not self.configbool('devel', config): | ||
Pierre-Yves David
|
r29095 | return | ||
Pierre-Yves David
|
r25629 | msg = 'devel-warn: ' + msg | ||
Pierre-Yves David
|
r27274 | stacklevel += 1 # get in develwarn | ||
Pierre-Yves David
|
r25629 | if self.tracebackflag: | ||
Pierre-Yves David
|
r27274 | util.debugstacktrace(msg, stacklevel, self.ferr, self.fout) | ||
timeless
|
r28498 | self.log('develwarn', '%s at:\n%s' % | ||
(msg, ''.join(util.getstackframes(stacklevel)))) | ||||
Pierre-Yves David
|
r25629 | else: | ||
curframe = inspect.currentframe() | ||||
calframe = inspect.getouterframes(curframe, 2) | ||||
Pierre-Yves David
|
r27274 | self.write_err('%s at: %s:%s (%s)\n' | ||
% ((msg,) + calframe[stacklevel][1:4])) | ||||
timeless
|
r28498 | self.log('develwarn', '%s at: %s:%s (%s)\n', | ||
msg, *calframe[stacklevel][1:4]) | ||||
Jun Wu
|
r29762 | curframe = calframe = None # avoid cycles | ||
Pierre-Yves David
|
r25629 | |||
Pierre-Yves David
|
r27275 | def deprecwarn(self, msg, version): | ||
"""issue a deprecation warning | ||||
- msg: message explaining what is deprecated and how to upgrade, | ||||
- version: last version where the API will be supported, | ||||
""" | ||||
Pierre-Yves David
|
r29082 | if not (self.configbool('devel', 'all-warnings') | ||
or self.configbool('devel', 'deprec-warn')): | ||||
return | ||||
Pierre-Yves David
|
r27275 | msg += ("\n(compatibility will be dropped after Mercurial-%s," | ||
" update your code.)") % version | ||||
Pierre-Yves David
|
r29096 | self.develwarn(msg, stacklevel=2, config='deprec-warn') | ||
Pierre-Yves David
|
r25629 | |||
Matt Harbison
|
r30832 | def exportableenviron(self): | ||
"""The environment variables that are safe to export, e.g. through | ||||
hgweb. | ||||
""" | ||||
return self._exportableenviron | ||||
Kostia Balytskyi
|
r30480 | @contextlib.contextmanager | ||
def configoverride(self, overrides, source=""): | ||||
"""Context manager for temporary config overrides | ||||
`overrides` must be a dict of the following structure: | ||||
{(section, name) : value}""" | ||||
backups = {} | ||||
Gregory Szorc
|
r30537 | try: | ||
for (section, name), value in overrides.items(): | ||||
backups[(section, name)] = self.backupconfig(section, name) | ||||
self.setconfig(section, name, value, source) | ||||
yield | ||||
finally: | ||||
for __, backup in backups.items(): | ||||
self.restoreconfig(backup) | ||||
# just restoring ui.quiet config to the previous value is not enough | ||||
# as it does not update ui.quiet class member | ||||
if ('ui', 'quiet') in overrides: | ||||
self.fixconfig(section='ui') | ||||
Kostia Balytskyi
|
r30480 | |||
Gregory Szorc
|
r24250 | class paths(dict): | ||
"""Represents a collection of paths and their configs. | ||||
Data is initially derived from ui instances and the config files they have | ||||
loaded. | ||||
""" | ||||
def __init__(self, ui): | ||||
dict.__init__(self) | ||||
Gregory Szorc
|
r27266 | for name, loc in ui.configitems('paths', ignoresub=True): | ||
Gregory Szorc
|
r24250 | # No location is the same as not existing. | ||
if not loc: | ||||
continue | ||||
Gregory Szorc
|
r27266 | loc, sub = ui.configsuboptions('paths', name) | ||
self[name] = path(ui, name, rawloc=loc, suboptions=sub) | ||||
Gregory Szorc
|
r24250 | |||
def getpath(self, name, default=None): | ||||
Yuya Nishihara
|
r27561 | """Return a ``path`` from a string, falling back to default. | ||
Gregory Szorc
|
r26056 | |||
``name`` can be a named path or locations. Locations are filesystem | ||||
paths or URIs. | ||||
Gregory Szorc
|
r24250 | |||
Gregory Szorc
|
r26189 | Returns None if ``name`` is not a registered path, a URI, or a local | ||
path to a repo. | ||||
Gregory Szorc
|
r24250 | """ | ||
Gregory Szorc
|
r26189 | # Only fall back to default if no path was requested. | ||
if name is None: | ||||
Yuya Nishihara
|
r27561 | if not default: | ||
default = () | ||||
elif not isinstance(default, (tuple, list)): | ||||
default = (default,) | ||||
for k in default: | ||||
Gregory Szorc
|
r26189 | try: | ||
Yuya Nishihara
|
r27561 | return self[k] | ||
Gregory Szorc
|
r26189 | except KeyError: | ||
Yuya Nishihara
|
r27561 | continue | ||
return None | ||||
Gregory Szorc
|
r26189 | |||
# Most likely empty string. | ||||
# This may need to raise in the future. | ||||
if not name: | ||||
return None | ||||
Gregory Szorc
|
r24250 | try: | ||
return self[name] | ||||
except KeyError: | ||||
Gregory Szorc
|
r26056 | # Try to resolve as a local path or URI. | ||
try: | ||||
Gregory Szorc
|
r27265 | # We don't pass sub-options in, so no need to pass ui instance. | ||
return path(None, None, rawloc=name) | ||||
Gregory Szorc
|
r26056 | except ValueError: | ||
Gregory Szorc
|
r26189 | raise error.RepoError(_('repository %s does not exist') % | ||
name) | ||||
Gregory Szorc
|
r26056 | |||
Gregory Szorc
|
r27266 | _pathsuboptions = {} | ||
def pathsuboption(option, attr): | ||||
"""Decorator used to declare a path sub-option. | ||||
Arguments are the sub-option name and the attribute it should set on | ||||
``path`` instances. | ||||
The decorated function will receive as arguments a ``ui`` instance, | ||||
``path`` instance, and the string value of this option from the config. | ||||
The function should return the value that will be set on the ``path`` | ||||
instance. | ||||
This decorator can be used to perform additional verification of | ||||
sub-options and to change the type of sub-options. | ||||
""" | ||||
def register(func): | ||||
_pathsuboptions[option] = (attr, func) | ||||
return func | ||||
return register | ||||
@pathsuboption('pushurl', 'pushloc') | ||||
def pushurlpathoption(ui, path, value): | ||||
u = util.url(value) | ||||
# Actually require a URL. | ||||
if not u.scheme: | ||||
ui.warn(_('(paths.%s:pushurl not a URL; ignoring)\n') % path.name) | ||||
return None | ||||
# Don't support the #foo syntax in the push URL to declare branch to | ||||
# push. | ||||
if u.fragment: | ||||
ui.warn(_('("#fragment" in paths.%s:pushurl not supported; ' | ||||
'ignoring)\n') % path.name) | ||||
u.fragment = None | ||||
return str(u) | ||||
Gregory Szorc
|
r29413 | @pathsuboption('pushrev', 'pushrev') | ||
def pushrevpathoption(ui, path, value): | ||||
return value | ||||
Gregory Szorc
|
r24250 | class path(object): | ||
"""Represents an individual path and its configuration.""" | ||||
Gregory Szorc
|
r27266 | def __init__(self, ui, name, rawloc=None, suboptions=None): | ||
Gregory Szorc
|
r24250 | """Construct a path from its config options. | ||
Gregory Szorc
|
r27265 | ``ui`` is the ``ui`` instance the path is coming from. | ||
Gregory Szorc
|
r24250 | ``name`` is the symbolic name of the path. | ||
``rawloc`` is the raw location, as defined in the config. | ||||
Gregory Szorc
|
r26064 | ``pushloc`` is the raw locations pushes should be made to. | ||
Gregory Szorc
|
r26056 | |||
If ``name`` is not defined, we require that the location be a) a local | ||||
filesystem path with a .hg directory or b) a URL. If not, | ||||
``ValueError`` is raised. | ||||
Gregory Szorc
|
r24250 | """ | ||
Gregory Szorc
|
r26056 | if not rawloc: | ||
raise ValueError('rawloc must be defined') | ||||
# Locations may define branches via syntax <base>#<branch>. | ||||
u = util.url(rawloc) | ||||
branch = None | ||||
if u.fragment: | ||||
branch = u.fragment | ||||
u.fragment = None | ||||
self.url = u | ||||
self.branch = branch | ||||
Gregory Szorc
|
r24250 | self.name = name | ||
Gregory Szorc
|
r26056 | self.rawloc = rawloc | ||
Augie Fackler
|
r31350 | self.loc = '%s' % u | ||
Gregory Szorc
|
r26056 | |||
# When given a raw location but not a symbolic name, validate the | ||||
# location is valid. | ||||
Durham Goode
|
r26076 | if not name and not u.scheme and not self._isvalidlocalpath(self.loc): | ||
Gregory Szorc
|
r26056 | raise ValueError('location is not a URL or path to a local ' | ||
'repo: %s' % rawloc) | ||||
Pierre-Yves David
|
r25498 | |||
Gregory Szorc
|
r27266 | suboptions = suboptions or {} | ||
# Now process the sub-options. If a sub-option is registered, its | ||||
# attribute will always be present. The value will be None if there | ||||
# was no valid sub-option. | ||||
for suboption, (attr, func) in _pathsuboptions.iteritems(): | ||||
if suboption not in suboptions: | ||||
setattr(self, attr, None) | ||||
continue | ||||
value = func(ui, self, suboptions[suboption]) | ||||
setattr(self, attr, value) | ||||
Durham Goode
|
r26076 | def _isvalidlocalpath(self, path): | ||
"""Returns True if the given path is a potentially valid repository. | ||||
This is its own function so that extensions can change the definition of | ||||
'valid' in this case (like when pulling from a git repo into a hg | ||||
one).""" | ||||
return os.path.isdir(os.path.join(path, '.hg')) | ||||
Gregory Szorc
|
r26064 | @property | ||
Gregory Szorc
|
r27266 | def suboptions(self): | ||
"""Return sub-options and their values for this path. | ||||
This is intended to be used for presentation purposes. | ||||
""" | ||||
d = {} | ||||
for subopt, (attr, _func) in _pathsuboptions.iteritems(): | ||||
value = getattr(self, attr) | ||||
if value is not None: | ||||
d[subopt] = value | ||||
return d | ||||
Gregory Szorc
|
r26064 | |||
Pierre-Yves David
|
r25498 | # we instantiate one globally shared progress bar to avoid | ||
# competing progress bars when multiple UI objects get created | ||||
_progresssingleton = None | ||||
def getprogbar(ui): | ||||
global _progresssingleton | ||||
if _progresssingleton is None: | ||||
# passing 'ui' object to the singleton is fishy, | ||||
# this is how the extension used to work but feel free to rework it. | ||||
_progresssingleton = progress.progbar(ui) | ||||
return _progresssingleton | ||||
Mark Thomas
|
r34346 | |||
def haveprogbar(): | ||||
return _progresssingleton is not None | ||||