util.py
281 lines
| 8.0 KiB
| text/x-python
|
PythonLexer
/ mercurial / util.py
mpm@selenic.com
|
r419 | # util.py - utility functions and platform specfic implementations | ||
# | ||||
# Copyright 2005 K. Thananchayan <thananck@yahoo.com> | ||||
# | ||||
# This software may be used and distributed according to the terms | ||||
# of the GNU General Public License, incorporated herein by reference. | ||||
Thomas Arendsen Hein
|
r704 | import os, errno | ||
Bryan O'Sullivan
|
r724 | from demandload import * | ||
demandload(globals(), "re") | ||||
mpm@selenic.com
|
r419 | |||
mpm@selenic.com
|
r556 | def unique(g): | ||
seen = {} | ||||
for f in g: | ||||
if f not in seen: | ||||
seen[f] = 1 | ||||
yield f | ||||
Bryan O'Sullivan
|
r870 | class Abort(Exception): | ||
"""Raised if a command needs to print an error and exit.""" | ||||
mpm@selenic.com
|
r508 | |||
Bryan O'Sullivan
|
r724 | def always(fn): return True | ||
def never(fn): return False | ||||
def globre(pat, head = '^', tail = '$'): | ||||
"convert a glob pattern into a regexp" | ||||
i, n = 0, len(pat) | ||||
res = '' | ||||
group = False | ||||
def peek(): return i < n and pat[i] | ||||
while i < n: | ||||
c = pat[i] | ||||
i = i+1 | ||||
if c == '*': | ||||
if peek() == '*': | ||||
i += 1 | ||||
res += '.*' | ||||
else: | ||||
res += '[^/]*' | ||||
elif c == '?': | ||||
res += '.' | ||||
elif c == '[': | ||||
j = i | ||||
if j < n and pat[j] in '!]': | ||||
j += 1 | ||||
while j < n and pat[j] != ']': | ||||
j += 1 | ||||
if j >= n: | ||||
res += '\\[' | ||||
else: | ||||
stuff = pat[i:j].replace('\\','\\\\') | ||||
i = j + 1 | ||||
if stuff[0] == '!': | ||||
stuff = '^' + stuff[1:] | ||||
elif stuff[0] == '^': | ||||
stuff = '\\' + stuff | ||||
res = '%s[%s]' % (res, stuff) | ||||
elif c == '{': | ||||
group = True | ||||
res += '(?:' | ||||
elif c == '}' and group: | ||||
res += ')' | ||||
group = False | ||||
elif c == ',' and group: | ||||
res += '|' | ||||
else: | ||||
res += re.escape(c) | ||||
return head + res + tail | ||||
Bryan O'Sullivan
|
r812 | _globchars = {'[': 1, '{': 1, '*': 1, '?': 1} | ||
Bryan O'Sullivan
|
r884 | def pathto(n1, n2): | ||
Bryan O'Sullivan
|
r886 | '''return the relative path from one place to another. | ||
this returns a path in the form used by the local filesystem, not hg.''' | ||||
if not n1: return localpath(n2) | ||||
a, b = n1.split('/'), n2.split('/') | ||||
Bryan O'Sullivan
|
r884 | a.reverse(), b.reverse() | ||
while a and b and a[-1] == b[-1]: | ||||
a.pop(), b.pop() | ||||
b.reverse() | ||||
return os.sep.join((['..'] * len(a)) + b) | ||||
Bryan O'Sullivan
|
r870 | def canonpath(repo, cwd, myname): | ||
rootsep = repo.root + os.sep | ||||
name = myname | ||||
if not name.startswith(os.sep): | ||||
name = os.path.join(repo.root, cwd, name) | ||||
name = os.path.normpath(name) | ||||
if name.startswith(rootsep): | ||||
Bryan O'Sullivan
|
r886 | return pconvert(name[len(rootsep):]) | ||
Bryan O'Sullivan
|
r870 | elif name == repo.root: | ||
return '' | ||||
else: | ||||
raise Abort('%s not under repository root' % myname) | ||||
mpm@selenic.com
|
r897 | |||
Bryan O'Sullivan
|
r870 | def matcher(repo, cwd, names, inc, exc, head = ''): | ||
Bryan O'Sullivan
|
r820 | def patkind(name): | ||
Bryan O'Sullivan
|
r888 | for prefix in 're:', 'glob:', 'path:', 'relpath:': | ||
Bryan O'Sullivan
|
r820 | if name.startswith(prefix): return name.split(':', 1) | ||
Bryan O'Sullivan
|
r812 | for c in name: | ||
Bryan O'Sullivan
|
r820 | if c in _globchars: return 'glob', name | ||
return 'relpath', name | ||||
Bryan O'Sullivan
|
r888 | def regex(kind, name, tail): | ||
mpm@selenic.com
|
r742 | '''convert a pattern into a regular expression''' | ||
Bryan O'Sullivan
|
r820 | if kind == 're': | ||
return name | ||||
elif kind == 'path': | ||||
Bryan O'Sullivan
|
r888 | return '^' + re.escape(name) + '(?:/|$)' | ||
elif kind == 'relpath': | ||||
return head + re.escape(name) + tail | ||||
mpm@selenic.com
|
r742 | return head + globre(name, '', tail) | ||
def matchfn(pats, tail): | ||||
"""build a matching function from a set of patterns""" | ||||
if pats: | ||||
Bryan O'Sullivan
|
r888 | pat = '(?:%s)' % '|'.join([regex(k, p, tail) for (k, p) in pats]) | ||
mpm@selenic.com
|
r742 | return re.compile(pat).match | ||
Bryan O'Sullivan
|
r820 | def globprefix(pat): | ||
'''return the non-glob prefix of a path, e.g. foo/* -> foo''' | ||||
root = [] | ||||
for p in pat.split(os.sep): | ||||
if patkind(p)[0] == 'glob': break | ||||
root.append(p) | ||||
Bryan O'Sullivan
|
r886 | return '/'.join(root) | ||
Bryan O'Sullivan
|
r820 | |||
Bryan O'Sullivan
|
r870 | pats = [] | ||
files = [] | ||||
roots = [] | ||||
for kind, name in map(patkind, names): | ||||
if kind in ('glob', 'relpath'): | ||||
name = canonpath(repo, cwd, name) | ||||
if name == '': | ||||
kind, name = 'glob', '**' | ||||
Bryan O'Sullivan
|
r888 | if kind in ('glob', 'path', 're'): | ||
pats.append((kind, name)) | ||||
Bryan O'Sullivan
|
r870 | if kind == 'glob': | ||
root = globprefix(name) | ||||
if root: roots.append(root) | ||||
elif kind == 'relpath': | ||||
Bryan O'Sullivan
|
r888 | files.append((kind, name)) | ||
Bryan O'Sullivan
|
r870 | roots.append(name) | ||
mpm@selenic.com
|
r897 | |||
Bryan O'Sullivan
|
r820 | patmatch = matchfn(pats, '$') or always | ||
filematch = matchfn(files, '(?:/|$)') or always | ||||
mpm@selenic.com
|
r897 | incmatch = always | ||
if inc: | ||||
incmatch = matchfn(map(patkind, inc), '(?:/|$)') | ||||
excmatch = lambda fn: False | ||||
if exc: | ||||
excmatch = matchfn(map(patkind, exc), '(?:/|$)') | ||||
mpm@selenic.com
|
r742 | |||
Bryan O'Sullivan
|
r820 | return roots, lambda fn: (incmatch(fn) and not excmatch(fn) and | ||
Bryan O'Sullivan
|
r812 | (fn.endswith('/') or | ||
(not pats and not files) or | ||||
(pats and patmatch(fn)) or | ||||
(files and filematch(fn)))) | ||||
mpm@selenic.com
|
r742 | |||
mpm@selenic.com
|
r521 | def system(cmd, errprefix=None): | ||
mpm@selenic.com
|
r508 | """execute a shell command that must succeed""" | ||
rc = os.system(cmd) | ||||
if rc: | ||||
mpm@selenic.com
|
r521 | errmsg = "%s %s" % (os.path.basename(cmd.split(None, 1)[0]), | ||
explain_exit(rc)[0]) | ||||
if errprefix: | ||||
errmsg = "%s: %s" % (errprefix, errmsg) | ||||
Bryan O'Sullivan
|
r870 | raise Abort(errmsg) | ||
mpm@selenic.com
|
r508 | |||
mpm@selenic.com
|
r421 | def rename(src, dst): | ||
try: | ||||
os.rename(src, dst) | ||||
except: | ||||
os.unlink(dst) | ||||
os.rename(src, dst) | ||||
Thomas Arendsen Hein
|
r698 | def copytree(src, dst, copyfile): | ||
"""Copy a directory tree, files are copied using 'copyfile'.""" | ||||
names = os.listdir(src) | ||||
os.mkdir(dst) | ||||
for name in names: | ||||
srcname = os.path.join(src, name) | ||||
dstname = os.path.join(dst, name) | ||||
if os.path.isdir(srcname): | ||||
copytree(srcname, dstname, copyfile) | ||||
elif os.path.isfile(srcname): | ||||
copyfile(srcname, dstname) | ||||
else: | ||||
mpm@selenic.com
|
r917 | pass | ||
Thomas Arendsen Hein
|
r698 | |||
Thomas Arendsen Hein
|
r704 | def _makelock_file(info, pathname): | ||
ld = os.open(pathname, os.O_CREAT | os.O_WRONLY | os.O_EXCL) | ||||
os.write(ld, info) | ||||
os.close(ld) | ||||
def _readlock_file(pathname): | ||||
return file(pathname).read() | ||||
mpm@selenic.com
|
r421 | # Platfor specific varients | ||
mpm@selenic.com
|
r419 | if os.name == 'nt': | ||
mpm@selenic.com
|
r461 | nulldev = 'NUL:' | ||
mpm@selenic.com
|
r441 | def is_exec(f, last): | ||
return last | ||||
def set_exec(f, mode): | ||||
pass | ||||
mpm@selenic.com
|
r515 | |||
mpm@selenic.com
|
r419 | def pconvert(path): | ||
return path.replace("\\", "/") | ||||
mpm@selenic.com
|
r422 | |||
Bryan O'Sullivan
|
r886 | def localpath(path): | ||
return path.replace('/', '\\') | ||||
def normpath(path): | ||||
return pconvert(os.path.normpath(path)) | ||||
Thomas Arendsen Hein
|
r704 | makelock = _makelock_file | ||
readlock = _readlock_file | ||||
mpm@selenic.com
|
r461 | |||
thananck@yahoo.com
|
r782 | def explain_exit(code): | ||
return "exited with status %d" % code, code | ||||
mpm@selenic.com
|
r419 | else: | ||
mpm@selenic.com
|
r461 | nulldev = '/dev/null' | ||
mpm@selenic.com
|
r441 | def is_exec(f, last): | ||
return (os.stat(f).st_mode & 0100 != 0) | ||||
def set_exec(f, mode): | ||||
s = os.stat(f).st_mode | ||||
if (s & 0100 != 0) == mode: | ||||
return | ||||
if mode: | ||||
# Turn on +x for every +r bit when making a file executable | ||||
# and obey umask. | ||||
umask = os.umask(0) | ||||
os.umask(umask) | ||||
os.chmod(f, s | (s & 0444) >> 2 & ~umask) | ||||
else: | ||||
os.chmod(f, s & 0666) | ||||
mpm@selenic.com
|
r419 | def pconvert(path): | ||
return path | ||||
Bryan O'Sullivan
|
r886 | def localpath(path): | ||
return path | ||||
normpath = os.path.normpath | ||||
mpm@selenic.com
|
r422 | def makelock(info, pathname): | ||
Thomas Arendsen Hein
|
r704 | try: | ||
os.symlink(info, pathname) | ||||
except OSError, why: | ||||
if why.errno == errno.EEXIST: | ||||
raise | ||||
else: | ||||
_makelock_file(info, pathname) | ||||
mpm@selenic.com
|
r422 | |||
def readlock(pathname): | ||||
Thomas Arendsen Hein
|
r704 | try: | ||
return os.readlink(pathname) | ||||
except OSError, why: | ||||
if why.errno == errno.EINVAL: | ||||
return _readlock_file(pathname) | ||||
else: | ||||
raise | ||||
thananck@yahoo.com
|
r782 | |||
def explain_exit(code): | ||||
"""return a 2-tuple (desc, code) describing a process's status""" | ||||
if os.WIFEXITED(code): | ||||
val = os.WEXITSTATUS(code) | ||||
return "exited with status %d" % val, val | ||||
elif os.WIFSIGNALED(code): | ||||
val = os.WTERMSIG(code) | ||||
return "killed by signal %d" % val, val | ||||
elif os.WIFSTOPPED(code): | ||||
mark.williamson@cl.cam.ac.uk
|
r912 | val = os.WSTOPSIG(code) | ||
thananck@yahoo.com
|
r782 | return "stopped by signal %d" % val, val | ||
raise ValueError("invalid exit code") | ||||