ui.py
452 lines
| 16.6 KiB
| text/x-python
|
PythonLexer
/ mercurial / ui.py
mpm@selenic.com
|
r207 | # ui.py - user interface bits for mercurial | ||
# | ||||
Vadim Gelfer
|
r2859 | # Copyright 2005, 2006 Matt Mackall <mpm@selenic.com> | ||
mpm@selenic.com
|
r207 | # | ||
# This software may be used and distributed according to the terms | ||||
# of the GNU General Public License, incorporated herein by reference. | ||||
Benoit Boissinot
|
r1400 | from i18n import gettext as _ | ||
Bryan O'Sullivan
|
r613 | from demandload import * | ||
Matt Mackall
|
r2889 | demandload(globals(), "errno getpass os re socket sys tempfile") | ||
Thomas Arendsen Hein
|
r3385 | demandload(globals(), "ConfigParser traceback util") | ||
mpm@selenic.com
|
r207 | |||
Alexis S. L. Carvalho
|
r3344 | def dupconfig(orig): | ||
Alexis S. L. Carvalho
|
r3425 | new = util.configparser(orig.defaults()) | ||
Alexis S. L. Carvalho
|
r3344 | updateconfig(orig, new) | ||
return new | ||||
Alexis S. L. Carvalho
|
r3433 | def updateconfig(source, dest, sections=None): | ||
if not sections: | ||||
sections = source.sections() | ||||
for section in sections: | ||||
Alexis S. L. Carvalho
|
r3344 | if not dest.has_section(section): | ||
dest.add_section(section) | ||||
for name, value in source.items(section, raw=True): | ||||
dest.set(section, name, value) | ||||
Eric Hopper
|
r1559 | class ui(object): | ||
mpm@selenic.com
|
r207 | def __init__(self, verbose=False, debug=False, quiet=False, | ||
Thomas Arendsen Hein
|
r3557 | interactive=True, traceback=False, report_untrusted=True, | ||
parentui=None): | ||||
Alexis S. L. Carvalho
|
r3344 | self.overlay = None | ||
Matt Mackall
|
r3737 | self.buffers = [] | ||
Thomas Arendsen Hein
|
r1839 | if parentui is None: | ||
Thomas Arendsen Hein
|
r1874 | # this is the parent of all ui children | ||
self.parentui = None | ||||
Alexis S. L. Carvalho
|
r3014 | self.readhooks = [] | ||
Alexis S. L. Carvalho
|
r3350 | self.quiet = quiet | ||
self.verbose = verbose | ||||
self.debugflag = debug | ||||
self.interactive = interactive | ||||
self.traceback = traceback | ||||
Thomas Arendsen Hein
|
r3557 | self.report_untrusted = report_untrusted | ||
Alexis S. L. Carvalho
|
r3551 | self.trusted_users = {} | ||
self.trusted_groups = {} | ||||
Alexis S. L. Carvalho
|
r3552 | # if ucdata is not None, its keys must be a superset of cdata's | ||
Alexis S. L. Carvalho
|
r3425 | self.cdata = util.configparser() | ||
Alexis S. L. Carvalho
|
r3552 | self.ucdata = None | ||
Alexis S. L. Carvalho
|
r3676 | # we always trust global config files | ||
self.check_trusted = False | ||||
Vadim Gelfer
|
r1951 | self.readconfig(util.rcpath()) | ||
Alexis S. L. Carvalho
|
r3676 | self.check_trusted = True | ||
Thomas Arendsen Hein
|
r1839 | self.updateopts(verbose, debug, quiet, interactive) | ||
Vadim Gelfer
|
r1866 | else: | ||
Thomas Arendsen Hein
|
r1874 | # parentui may point to an ui object which is already a child | ||
self.parentui = parentui.parentui or parentui | ||||
Alexis S. L. Carvalho
|
r3338 | self.readhooks = self.parentui.readhooks[:] | ||
Alexis S. L. Carvalho
|
r3551 | self.trusted_users = parentui.trusted_users.copy() | ||
self.trusted_groups = parentui.trusted_groups.copy() | ||||
Alexis S. L. Carvalho
|
r3344 | self.cdata = dupconfig(self.parentui.cdata) | ||
Alexis S. L. Carvalho
|
r3552 | if self.parentui.ucdata: | ||
self.ucdata = dupconfig(self.parentui.ucdata) | ||||
Alexis S. L. Carvalho
|
r3344 | if self.parentui.overlay: | ||
self.overlay = dupconfig(self.parentui.overlay) | ||||
Thomas Arendsen Hein
|
r1839 | |||
def __getattr__(self, key): | ||||
return getattr(self.parentui, key) | ||||
mason@suse.com
|
r1071 | |||
def updateopts(self, verbose=False, debug=False, quiet=False, | ||||
Vadim Gelfer
|
r2293 | interactive=True, traceback=False, config=[]): | ||
Alexis S. L. Carvalho
|
r3346 | for section, name, value in config: | ||
self.setconfig(section, name, value) | ||||
mpm@selenic.com
|
r285 | |||
Alexis S. L. Carvalho
|
r3350 | if quiet or verbose or debug: | ||
self.setconfig('ui', 'quiet', str(bool(quiet))) | ||||
self.setconfig('ui', 'verbose', str(bool(verbose))) | ||||
self.setconfig('ui', 'debug', str(bool(debug))) | ||||
self.verbosity_constraints() | ||||
if not interactive: | ||||
self.setconfig('ui', 'interactive', 'False') | ||||
self.interactive = False | ||||
self.traceback = self.traceback or traceback | ||||
def verbosity_constraints(self): | ||||
self.quiet = self.configbool('ui', 'quiet') | ||||
self.verbose = self.configbool('ui', 'verbose') | ||||
self.debugflag = self.configbool('ui', 'debug') | ||||
Alexis S. L. Carvalho
|
r3349 | if self.debugflag: | ||
self.verbose = True | ||||
self.quiet = False | ||||
elif self.verbose and self.quiet: | ||||
Alexis S. L. Carvalho
|
r3350 | self.quiet = self.verbose = False | ||
Alexis S. L. Carvalho
|
r3349 | |||
Alexis S. L. Carvalho
|
r3551 | def _is_trusted(self, fp, f, warn=True): | ||
Alexis S. L. Carvalho
|
r3676 | if not self.check_trusted: | ||
return True | ||||
Alexis S. L. Carvalho
|
r3677 | st = util.fstat(fp) | ||
if util.isowner(fp, st): | ||||
return True | ||||
Alexis S. L. Carvalho
|
r3551 | tusers = self.trusted_users | ||
tgroups = self.trusted_groups | ||||
Alexis S. L. Carvalho
|
r3678 | if not tusers: | ||
user = util.username() | ||||
if user is not None: | ||||
self.trusted_users[user] = 1 | ||||
self.fixconfig(section='trusted') | ||||
Alexis S. L. Carvalho
|
r3551 | if (tusers or tgroups) and '*' not in tusers and '*' not in tgroups: | ||
user = util.username(st.st_uid) | ||||
group = util.groupname(st.st_gid) | ||||
if user not in tusers and group not in tgroups: | ||||
Thomas Arendsen Hein
|
r3557 | if warn and self.report_untrusted: | ||
Alexis S. L. Carvalho
|
r3552 | self.warn(_('Not trusting file %s from untrusted ' | ||
Alexis S. L. Carvalho
|
r3551 | 'user %s, group %s\n') % (f, user, group)) | ||
return False | ||||
return True | ||||
Thomas Arendsen Hein
|
r1893 | def readconfig(self, fn, root=None): | ||
Soh Tk-r28629
|
r1483 | if isinstance(fn, basestring): | ||
fn = [fn] | ||||
for f in fn: | ||||
try: | ||||
Alexis S. L. Carvalho
|
r3551 | fp = open(f) | ||
except IOError: | ||||
continue | ||||
Alexis S. L. Carvalho
|
r3552 | cdata = self.cdata | ||
trusted = self._is_trusted(fp, f) | ||||
if not trusted: | ||||
if self.ucdata is None: | ||||
self.ucdata = dupconfig(self.cdata) | ||||
cdata = self.ucdata | ||||
elif self.ucdata is not None: | ||||
# use a separate configparser, so that we don't accidentally | ||||
# override ucdata settings later on. | ||||
cdata = util.configparser() | ||||
Alexis S. L. Carvalho
|
r3551 | try: | ||
Alexis S. L. Carvalho
|
r3552 | cdata.readfp(fp, f) | ||
Soh Tk-r28629
|
r1483 | except ConfigParser.ParsingError, inst: | ||
Alexis S. L. Carvalho
|
r3552 | msg = _("Failed to parse %s\n%s") % (f, inst) | ||
if trusted: | ||||
raise util.Abort(msg) | ||||
self.warn(_("Ignored: %s\n") % msg) | ||||
if trusted: | ||||
if cdata != self.cdata: | ||||
updateconfig(cdata, self.cdata) | ||||
if self.ucdata is not None: | ||||
updateconfig(cdata, self.ucdata) | ||||
Alexis S. L. Carvalho
|
r3344 | # override data from config files with data set with ui.setconfig | ||
if self.overlay: | ||||
updateconfig(self.overlay, self.cdata) | ||||
Alexis S. L. Carvalho
|
r3347 | if root is None: | ||
root = os.path.expanduser('~') | ||||
self.fixconfig(root=root) | ||||
Vadim Gelfer
|
r2944 | for hook in self.readhooks: | ||
hook(self) | ||||
mpm@selenic.com
|
r337 | |||
Alexis S. L. Carvalho
|
r3014 | def addreadhook(self, hook): | ||
self.readhooks.append(hook) | ||||
Alexis S. L. Carvalho
|
r3433 | def readsections(self, filename, *sections): | ||
Alexis S. L. Carvalho
|
r3552 | """Read filename and add only the specified sections to the config data | ||
The settings are added to the trusted config data. | ||||
""" | ||||
Alexis S. L. Carvalho
|
r3433 | if not sections: | ||
return | ||||
cdata = util.configparser() | ||||
try: | ||||
cdata.read(filename) | ||||
except ConfigParser.ParsingError, inst: | ||||
Benoit Boissinot
|
r3474 | raise util.Abort(_("failed to parse %s\n%s") % (filename, | ||
inst)) | ||||
Alexis S. L. Carvalho
|
r3433 | |||
for section in sections: | ||||
if not cdata.has_section(section): | ||||
cdata.add_section(section) | ||||
updateconfig(cdata, self.cdata, sections) | ||||
Alexis S. L. Carvalho
|
r3552 | if self.ucdata: | ||
updateconfig(cdata, self.ucdata, sections) | ||||
Alexis S. L. Carvalho
|
r3433 | |||
Alexis S. L. Carvalho
|
r3347 | def fixconfig(self, section=None, name=None, value=None, root=None): | ||
# translate paths relative to root (or home) into absolute paths | ||||
if section is None or section == 'paths': | ||||
if root is None: | ||||
root = os.getcwd() | ||||
items = section and [(name, value)] or [] | ||||
Alexis S. L. Carvalho
|
r3552 | for cdata in self.cdata, self.ucdata, self.overlay: | ||
Alexis S. L. Carvalho
|
r3347 | if not cdata: continue | ||
if not items and cdata.has_section('paths'): | ||||
pathsitems = cdata.items('paths') | ||||
else: | ||||
pathsitems = items | ||||
for n, path in pathsitems: | ||||
if path and "://" not in path and not os.path.isabs(path): | ||||
cdata.set("paths", n, os.path.join(root, path)) | ||||
Alexis S. L. Carvalho
|
r3350 | # update quiet/verbose/debug and interactive status | ||
if section is None or section == 'ui': | ||||
if name is None or name in ('quiet', 'verbose', 'debug'): | ||||
self.verbosity_constraints() | ||||
if name is None or name == 'interactive': | ||||
self.interactive = self.configbool("ui", "interactive", True) | ||||
Alexis S. L. Carvalho
|
r3551 | # update trust information | ||
Alexis S. L. Carvalho
|
r3678 | if (section is None or section == 'trusted') and self.trusted_users: | ||
for user in self.configlist('trusted', 'users'): | ||||
Alexis S. L. Carvalho
|
r3551 | self.trusted_users[user] = 1 | ||
Alexis S. L. Carvalho
|
r3678 | for group in self.configlist('trusted', 'groups'): | ||
self.trusted_groups[group] = 1 | ||||
Alexis S. L. Carvalho
|
r3551 | |||
Alexis S. L. Carvalho
|
r3344 | def setconfig(self, section, name, value): | ||
if not self.overlay: | ||||
Alexis S. L. Carvalho
|
r3425 | self.overlay = util.configparser() | ||
Alexis S. L. Carvalho
|
r3552 | for cdata in (self.overlay, self.cdata, self.ucdata): | ||
if not cdata: continue | ||||
Alexis S. L. Carvalho
|
r3344 | if not cdata.has_section(section): | ||
cdata.add_section(section) | ||||
cdata.set(section, name, value) | ||||
Alexis S. L. Carvalho
|
r3347 | self.fixconfig(section, name, value) | ||
mpm@selenic.com
|
r960 | |||
Alexis S. L. Carvalho
|
r3552 | def _get_cdata(self, untrusted): | ||
if untrusted and self.ucdata: | ||||
return self.ucdata | ||||
return self.cdata | ||||
def _config(self, section, name, default, funcname, untrusted, abort): | ||||
cdata = self._get_cdata(untrusted) | ||||
if cdata.has_option(section, name): | ||||
Thomas Arendsen Hein
|
r1876 | try: | ||
Alexis S. L. Carvalho
|
r3552 | func = getattr(cdata, funcname) | ||
Alexis S. L. Carvalho
|
r3341 | return func(section, name) | ||
Thomas Arendsen Hein
|
r1876 | except ConfigParser.InterpolationError, inst: | ||
Alexis S. L. Carvalho
|
r3552 | msg = _("Error in configuration section [%s] " | ||
"parameter '%s':\n%s") % (section, name, inst) | ||||
if abort: | ||||
raise util.Abort(msg) | ||||
self.warn(_("Ignored: %s\n") % msg) | ||||
Alexis S. L. Carvalho
|
r3343 | return default | ||
Alexis S. L. Carvalho
|
r3341 | |||
Alexis S. L. Carvalho
|
r3552 | def _configcommon(self, section, name, default, funcname, untrusted): | ||
value = self._config(section, name, default, funcname, | ||||
untrusted, abort=True) | ||||
if self.debugflag and not untrusted and self.ucdata: | ||||
uvalue = self._config(section, name, None, funcname, | ||||
untrusted=True, abort=False) | ||||
if uvalue is not None and uvalue != value: | ||||
self.warn(_("Ignoring untrusted configuration option " | ||||
"%s.%s = %s\n") % (section, name, uvalue)) | ||||
return value | ||||
Alexis S. L. Carvalho
|
r3341 | |||
Alexis S. L. Carvalho
|
r3552 | def config(self, section, name, default=None, untrusted=False): | ||
return self._configcommon(section, name, default, 'get', untrusted) | ||||
mpm@selenic.com
|
r285 | |||
Alexis S. L. Carvalho
|
r3552 | def configbool(self, section, name, default=False, untrusted=False): | ||
return self._configcommon(section, name, default, 'getboolean', | ||||
untrusted) | ||||
def configlist(self, section, name, default=None, untrusted=False): | ||||
Thomas Arendsen Hein
|
r2499 | """Return a list of comma/space separated strings""" | ||
Alexis S. L. Carvalho
|
r3552 | result = self.config(section, name, untrusted=untrusted) | ||
Thomas Arendsen Hein
|
r2499 | if result is None: | ||
Thomas Arendsen Hein
|
r2502 | result = default or [] | ||
if isinstance(result, basestring): | ||||
result = result.replace(",", " ").split() | ||||
return result | ||||
Thomas Arendsen Hein
|
r2499 | |||
Alexis S. L. Carvalho
|
r3552 | def has_config(self, section, untrusted=False): | ||
Vadim Gelfer
|
r2343 | '''tell whether section exists in config.''' | ||
Alexis S. L. Carvalho
|
r3552 | cdata = self._get_cdata(untrusted) | ||
return cdata.has_section(section) | ||||
Vadim Gelfer
|
r2343 | |||
Alexis S. L. Carvalho
|
r3552 | def _configitems(self, section, untrusted, abort): | ||
Thomas Arendsen Hein
|
r1839 | items = {} | ||
Alexis S. L. Carvalho
|
r3552 | cdata = self._get_cdata(untrusted) | ||
if cdata.has_section(section): | ||||
Thomas Arendsen Hein
|
r1876 | try: | ||
Alexis S. L. Carvalho
|
r3552 | items.update(dict(cdata.items(section))) | ||
Thomas Arendsen Hein
|
r1876 | except ConfigParser.InterpolationError, inst: | ||
Alexis S. L. Carvalho
|
r3552 | msg = _("Error in configuration section [%s]:\n" | ||
"%s") % (section, inst) | ||||
if abort: | ||||
raise util.Abort(msg) | ||||
self.warn(_("Ignored: %s\n") % msg) | ||||
return items | ||||
def configitems(self, section, untrusted=False): | ||||
items = self._configitems(section, untrusted=untrusted, abort=True) | ||||
if self.debugflag and not untrusted and self.ucdata: | ||||
uitems = self._configitems(section, untrusted=True, abort=False) | ||||
keys = uitems.keys() | ||||
keys.sort() | ||||
for k in keys: | ||||
if uitems[k] != items.get(k): | ||||
self.warn(_("Ignoring untrusted configuration option " | ||||
"%s.%s = %s\n") % (section, k, uitems[k])) | ||||
Thomas Arendsen Hein
|
r1839 | x = items.items() | ||
x.sort() | ||||
return x | ||||
mpm@selenic.com
|
r285 | |||
Alexis S. L. Carvalho
|
r3552 | def walkconfig(self, untrusted=False): | ||
cdata = self._get_cdata(untrusted) | ||||
sections = cdata.sections() | ||||
Alexis S. L. Carvalho
|
r3342 | sections.sort() | ||
for section in sections: | ||||
Alexis S. L. Carvalho
|
r3552 | for name, value in self.configitems(section, untrusted): | ||
Alexis S. L. Carvalho
|
r3342 | yield section, name, value.replace('\n', '\\n') | ||
Bryan O'Sullivan
|
r1028 | |||
mason@suse.com
|
r1071 | def extensions(self): | ||
Thomas Arendsen Hein
|
r2403 | result = self.configitems("extensions") | ||
for i, (key, value) in enumerate(result): | ||||
if value: | ||||
result[i] = (key, os.path.expanduser(value)) | ||||
return result | ||||
mason@suse.com
|
r1071 | |||
mcmillen@cs.cmu.edu
|
r2003 | def hgignorefiles(self): | ||
Thomas Arendsen Hein
|
r2403 | result = [] | ||
for key, value in self.configitems("ui"): | ||||
if key == 'ignore' or key.startswith('ignore.'): | ||||
result.append(os.path.expanduser(value)) | ||||
return result | ||||
mcmillen@cs.cmu.edu
|
r2003 | |||
mason@suse.com
|
r2072 | def configrevlog(self): | ||
Thomas Arendsen Hein
|
r2403 | result = {} | ||
for key, value in self.configitems("revlog"): | ||||
result[key.lower()] = value | ||||
return result | ||||
Markus F.X.J. Oberhumer
|
r2388 | |||
Matt Mackall
|
r608 | def username(self): | ||
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. | ||||
Benoit Boissinot
|
r3721 | If not found, use ($LOGNAME or $USER or $LNAME or | ||
$USERNAME) +"@full.hostname". | ||||
Thomas Arendsen Hein
|
r1985 | """ | ||
user = os.environ.get("HGUSER") | ||||
if user is None: | ||||
user = self.config("ui", "username") | ||||
if user is None: | ||||
user = os.environ.get("EMAIL") | ||||
Thomas Arendsen Hein
|
r3489 | if not user: | ||
Benoit Boissinot
|
r3721 | try: | ||
user = '%s@%s' % (util.getuser(), socket.getfqdn()) | ||||
except KeyError: | ||||
raise util.Abort(_("Please specify a username.")) | ||||
self.warn(_("No username found, using '%s' instead\n" % 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.""" | ||||
Vadim Gelfer
|
r1903 | 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]""" | ||
Benoit Boissinot
|
r2624 | if "://" in loc or os.path.isdir(loc): | ||
Thomas Arendsen Hein
|
r1892 | return loc | ||
Vadim Gelfer
|
r2495 | path = self.config("paths", loc) | ||
if not path and default is not None: | ||||
path = self.config("paths", default) | ||||
Thomas Arendsen Hein
|
r2498 | return path or loc | ||
mpm@selenic.com
|
r506 | |||
Matt Mackall
|
r3737 | def pushbuffer(self): | ||
self.buffers.append([]) | ||||
def popbuffer(self): | ||||
return "".join(self.buffers.pop()) | ||||
mpm@selenic.com
|
r207 | def write(self, *args): | ||
Matt Mackall
|
r3737 | if self.buffers: | ||
self.buffers[-1].extend([str(a) for a in args]) | ||||
else: | ||||
for a in args: | ||||
sys.stdout.write(str(a)) | ||||
mpm@selenic.com
|
r565 | |||
def write_err(self, *args): | ||||
Benoit Boissinot
|
r1989 | try: | ||
if not sys.stdout.closed: sys.stdout.flush() | ||||
for a in args: | ||||
sys.stderr.write(str(a)) | ||||
except IOError, inst: | ||||
if inst.errno != errno.EPIPE: | ||||
raise | ||||
mpm@selenic.com
|
r565 | |||
Vadim Gelfer
|
r1837 | def flush(self): | ||
Eung-Ju Park
|
r2013 | try: sys.stdout.flush() | ||
except: pass | ||||
try: sys.stderr.flush() | ||||
except: pass | ||||
Vadim Gelfer
|
r1837 | |||
mpm@selenic.com
|
r207 | def readline(self): | ||
return sys.stdin.readline()[:-1] | ||||
Vadim Gelfer
|
r2281 | def prompt(self, msg, pat=None, default="y"): | ||
mpm@selenic.com
|
r207 | if not self.interactive: return default | ||
while 1: | ||||
self.write(msg, " ") | ||||
r = self.readline() | ||||
Vadim Gelfer
|
r2281 | if not pat or re.match(pat, r): | ||
mpm@selenic.com
|
r207 | return r | ||
else: | ||||
Benoit Boissinot
|
r1402 | self.write(_("unrecognized response\n")) | ||
Vadim Gelfer
|
r2281 | def getpass(self, prompt=None, default=None): | ||
if not self.interactive: return default | ||||
return getpass.getpass(prompt or _('password: ')) | ||||
mpm@selenic.com
|
r207 | def status(self, *msg): | ||
if not self.quiet: self.write(*msg) | ||||
Thomas Arendsen Hein
|
r234 | def warn(self, *msg): | ||
mpm@selenic.com
|
r565 | self.write_err(*msg) | ||
mpm@selenic.com
|
r207 | def note(self, *msg): | ||
if self.verbose: self.write(*msg) | ||||
def debug(self, *msg): | ||||
if self.debugflag: self.write(*msg) | ||||
Thomas Arendsen Hein
|
r1983 | def edit(self, text, user): | ||
Stephen Darnell
|
r2206 | (fd, name) = tempfile.mkstemp(prefix="hg-editor-", suffix=".txt", | ||
text=True) | ||||
Thomas Arendsen Hein
|
r1984 | try: | ||
f = os.fdopen(fd, "w") | ||||
f.write(text) | ||||
f.close() | ||||
editor = (os.environ.get("HGEDITOR") or | ||||
self.config("ui", "editor") or | ||||
os.environ.get("EDITOR", "vi")) | ||||
mpm@selenic.com
|
r207 | |||
Thomas Arendsen Hein
|
r1984 | util.system("%s \"%s\"" % (editor, name), | ||
environ={'HGUSER': user}, | ||||
onerr=util.Abort, errprefix=_("edit failed")) | ||||
Matt Mackall
|
r608 | |||
Thomas Arendsen Hein
|
r1984 | f = open(name) | ||
t = f.read() | ||||
f.close() | ||||
t = re.sub("(?m)^HG:.*\n", "", t) | ||||
finally: | ||||
os.unlink(name) | ||||
Radoslaw "AstralStorm" Szkodzinski
|
r662 | |||
mpm@selenic.com
|
r207 | return t | ||
Vadim Gelfer
|
r2200 | |||
Vadim Gelfer
|
r2335 | def print_exc(self): | ||
'''print exception traceback if traceback printing enabled. | ||||
only to call in exception handler. returns true if traceback | ||||
printed.''' | ||||
if self.traceback: | ||||
traceback.print_exc() | ||||
return self.traceback | ||||