ui.py
363 lines
| 13.4 KiB
| text/x-python
|
PythonLexer
/ mercurial / ui.py
mpm@selenic.com
|
r207 | # ui.py - user interface bits for mercurial | ||
# | ||||
# Copyright 2005 Matt Mackall <mpm@selenic.com> | ||||
# | ||||
# 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 * | ||
Vadim Gelfer
|
r2292 | demandload(globals(), "errno getpass os re smtplib socket sys tempfile") | ||
Vadim Gelfer
|
r2470 | demandload(globals(), "ConfigParser templater traceback util") | ||
mpm@selenic.com
|
r207 | |||
Eric Hopper
|
r1559 | class ui(object): | ||
mpm@selenic.com
|
r207 | def __init__(self, verbose=False, debug=False, quiet=False, | ||
Vadim Gelfer
|
r2166 | interactive=True, traceback=False, parentui=None): | ||
mpm@selenic.com
|
r960 | self.overlay = {} | ||
Thomas Arendsen Hein
|
r1839 | if parentui is None: | ||
Thomas Arendsen Hein
|
r1874 | # this is the parent of all ui children | ||
self.parentui = None | ||||
self.cdata = ConfigParser.SafeConfigParser() | ||||
Vadim Gelfer
|
r1951 | self.readconfig(util.rcpath()) | ||
mpm@selenic.com
|
r285 | |||
Thomas Arendsen Hein
|
r1839 | self.quiet = self.configbool("ui", "quiet") | ||
self.verbose = self.configbool("ui", "verbose") | ||||
self.debugflag = self.configbool("ui", "debug") | ||||
self.interactive = self.configbool("ui", "interactive", True) | ||||
Vadim Gelfer
|
r2166 | self.traceback = traceback | ||
mpm@selenic.com
|
r285 | |||
Thomas Arendsen Hein
|
r1839 | self.updateopts(verbose, debug, quiet, interactive) | ||
self.diffcache = None | ||||
Thomas Arendsen Hein
|
r2033 | self.header = [] | ||
self.prev_header = [] | ||||
mason@suse.com
|
r2072 | self.revlogopts = self.configrevlog() | ||
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 | ||||
parent_cdata = self.parentui.cdata | ||||
self.cdata = ConfigParser.SafeConfigParser(parent_cdata.defaults()) | ||||
# make interpolation work | ||||
for section in parent_cdata.sections(): | ||||
self.cdata.add_section(section) | ||||
for name, value in parent_cdata.items(section, raw=True): | ||||
self.cdata.set(section, name, value) | ||||
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=[]): | ||
mpm@selenic.com
|
r285 | self.quiet = (self.quiet or quiet) and not verbose and not debug | ||
self.verbose = (self.verbose or verbose) or debug | ||||
self.debugflag = (self.debugflag or debug) | ||||
self.interactive = (self.interactive and interactive) | ||||
Vadim Gelfer
|
r2166 | self.traceback = self.traceback or traceback | ||
Vadim Gelfer
|
r2293 | for cfg in config: | ||
try: | ||||
name, value = cfg.split('=', 1) | ||||
section, name = name.split('.', 1) | ||||
if not self.cdata.has_section(section): | ||||
self.cdata.add_section(section) | ||||
if not section or not name: | ||||
raise IndexError | ||||
self.cdata.set(section, name, value) | ||||
except (IndexError, ValueError): | ||||
raise util.Abort(_('malformed --config option: %s') % cfg) | ||||
mpm@selenic.com
|
r285 | |||
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: | ||||
self.cdata.read(f) | ||||
except ConfigParser.ParsingError, inst: | ||||
raise util.Abort(_("Failed to parse %s\n%s") % (f, inst)) | ||||
Thomas Arendsen Hein
|
r1893 | # translate paths relative to root (or home) into absolute paths | ||
if root is None: | ||||
root = os.path.expanduser('~') | ||||
for name, path in self.configitems("paths"): | ||||
Benoit Boissinot
|
r2579 | if path and "://" not in path and not os.path.isabs(path): | ||
Thomas Arendsen Hein
|
r1893 | self.cdata.set("paths", name, os.path.join(root, path)) | ||
mpm@selenic.com
|
r337 | |||
mpm@selenic.com
|
r960 | def setconfig(self, section, name, val): | ||
self.overlay[(section, name)] = val | ||||
def config(self, section, name, default=None): | ||||
if self.overlay.has_key((section, name)): | ||||
return self.overlay[(section, name)] | ||||
if self.cdata.has_option(section, name): | ||||
Thomas Arendsen Hein
|
r1876 | try: | ||
return self.cdata.get(section, name) | ||||
except ConfigParser.InterpolationError, inst: | ||||
raise util.Abort(_("Error in configuration:\n%s") % inst) | ||||
Thomas Arendsen Hein
|
r1839 | if self.parentui is None: | ||
return default | ||||
else: | ||||
return self.parentui.config(section, name, default) | ||||
mpm@selenic.com
|
r285 | |||
Thomas Arendsen Hein
|
r2499 | def configlist(self, section, name, default=None): | ||
"""Return a list of comma/space separated strings""" | ||||
result = self.config(section, name) | ||||
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 | |||
mpm@selenic.com
|
r960 | def configbool(self, section, name, default=False): | ||
if self.overlay.has_key((section, name)): | ||||
return self.overlay[(section, name)] | ||||
if self.cdata.has_option(section, name): | ||||
Thomas Arendsen Hein
|
r1876 | try: | ||
return self.cdata.getboolean(section, name) | ||||
except ConfigParser.InterpolationError, inst: | ||||
raise util.Abort(_("Error in configuration:\n%s") % inst) | ||||
Thomas Arendsen Hein
|
r1839 | if self.parentui is None: | ||
return default | ||||
else: | ||||
return self.parentui.configbool(section, name, default) | ||||
mpm@selenic.com
|
r285 | |||
Vadim Gelfer
|
r2343 | def has_config(self, section): | ||
'''tell whether section exists in config.''' | ||||
return self.cdata.has_section(section) | ||||
mpm@selenic.com
|
r285 | def configitems(self, section): | ||
Thomas Arendsen Hein
|
r1839 | items = {} | ||
if self.parentui is not None: | ||||
items = dict(self.parentui.configitems(section)) | ||||
mpm@selenic.com
|
r285 | if self.cdata.has_section(section): | ||
Thomas Arendsen Hein
|
r1876 | try: | ||
items.update(dict(self.cdata.items(section))) | ||||
except ConfigParser.InterpolationError, inst: | ||||
raise util.Abort(_("Error in configuration:\n%s") % inst) | ||||
Thomas Arendsen Hein
|
r1839 | x = items.items() | ||
x.sort() | ||||
return x | ||||
mpm@selenic.com
|
r285 | |||
Thomas Arendsen Hein
|
r1839 | def walkconfig(self, seen=None): | ||
if seen is None: | ||||
seen = {} | ||||
Bryan O'Sullivan
|
r1028 | for (section, name), value in self.overlay.iteritems(): | ||
yield section, name, value | ||||
seen[section, name] = 1 | ||||
for section in self.cdata.sections(): | ||||
for name, value in self.cdata.items(section): | ||||
if (section, name) in seen: continue | ||||
yield section, name, value.replace('\n', '\\n') | ||||
seen[section, name] = 1 | ||||
Thomas Arendsen Hein
|
r1839 | if self.parentui is not None: | ||
for parent in self.parentui.walkconfig(seen): | ||||
yield parent | ||||
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 | |||
mason@suse.com
|
r1637 | def diffopts(self): | ||
if self.diffcache: | ||||
return self.diffcache | ||||
Haakon Riiser
|
r2580 | result = {'showfunc': True, 'ignorews': False, | ||
'ignorewsamount': False, 'ignoreblanklines': False} | ||||
Thomas Arendsen Hein
|
r2403 | for key, value in self.configitems("diff"): | ||
if value: | ||||
result[key.lower()] = (value.lower() == 'true') | ||||
self.diffcache = result | ||||
return result | ||||
mason@suse.com
|
r1637 | |||
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. | ||||
Abort if found username is an empty string to force specifying | ||||
the commit user elsewhere, e.g. with line option or repo hgrc. | ||||
Vadim Gelfer
|
r2343 | 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") | ||||
if user is None: | ||||
Vadim Gelfer
|
r2343 | try: | ||
Vadim Gelfer
|
r2652 | user = '%s@%s' % (util.getuser(), socket.getfqdn()) | ||
Vadim Gelfer
|
r2343 | except KeyError: | ||
raise util.Abort(_("Please specify a username.")) | ||||
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 | |||
Thomas Arendsen Hein
|
r2598 | def setconfig_remoteopts(self, **opts): | ||
if opts.get('ssh'): | ||||
self.setconfig("ui", "ssh", opts['ssh']) | ||||
if opts.get('remotecmd'): | ||||
self.setconfig("ui", "remotecmd", opts['remotecmd']) | ||||
mpm@selenic.com
|
r207 | def write(self, *args): | ||
Thomas Arendsen Hein
|
r2033 | if self.header: | ||
if self.header != self.prev_header: | ||||
self.prev_header = self.header | ||||
self.write(*self.header) | ||||
self.header = [] | ||||
mpm@selenic.com
|
r207 | for a in args: | ||
sys.stdout.write(str(a)) | ||||
mpm@selenic.com
|
r565 | |||
Thomas Arendsen Hein
|
r2033 | def write_header(self, *args): | ||
for a in args: | ||||
self.header.append(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 | |||
def sendmail(self): | ||||
Vadim Gelfer
|
r2292 | '''send mail message. object returned has one method, sendmail. | ||
call as sendmail(sender, list-of-recipients, msg).''' | ||||
def smtp(): | ||||
'''send mail using smtp.''' | ||||
Valentino Volonghi aka dialtone
|
r2583 | local_hostname = self.config('smtp', 'local_hostname') | ||
s = smtplib.SMTP(local_hostname=local_hostname) | ||||
Vadim Gelfer
|
r2292 | mailhost = self.config('smtp', 'host') | ||
if not mailhost: | ||||
raise util.Abort(_('no [smtp]host in hgrc - cannot send mail')) | ||||
mailport = int(self.config('smtp', 'port', 25)) | ||||
self.note(_('sending mail: smtp host %s, port %s\n') % | ||||
(mailhost, mailport)) | ||||
s.connect(host=mailhost, port=mailport) | ||||
if self.configbool('smtp', 'tls'): | ||||
self.note(_('(using tls)\n')) | ||||
s.ehlo() | ||||
s.starttls() | ||||
s.ehlo() | ||||
username = self.config('smtp', 'username') | ||||
password = self.config('smtp', 'password') | ||||
if username and password: | ||||
self.note(_('(authenticating to mail server as %s)\n') % | ||||
(username)) | ||||
s.login(username, password) | ||||
return s | ||||
class sendmail(object): | ||||
'''send mail using sendmail.''' | ||||
def __init__(self, ui, program): | ||||
self.ui = ui | ||||
self.program = program | ||||
def sendmail(self, sender, recipients, msg): | ||||
cmdline = '%s -f %s %s' % ( | ||||
self.program, templater.email(sender), | ||||
' '.join(map(templater.email, recipients))) | ||||
self.ui.note(_('sending mail: %s\n') % cmdline) | ||||
fp = os.popen(cmdline, 'w') | ||||
fp.write(msg) | ||||
ret = fp.close() | ||||
if ret: | ||||
raise util.Abort('%s %s' % ( | ||||
os.path.basename(self.program.split(None, 1)[0]), | ||||
util.explain_exit(ret)[0])) | ||||
method = self.config('email', 'method', 'smtp') | ||||
if method == 'smtp': | ||||
mail = smtp() | ||||
else: | ||||
mail = sendmail(self, method) | ||||
return mail | ||||
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 | ||||