dirstate.py
640 lines
| 20.4 KiB
| text/x-python
|
PythonLexer
/ mercurial / dirstate.py
mpm@selenic.com
|
r1089 | """ | ||
dirstate.py - working directory tracking for mercurial | ||||
Thomas Arendsen Hein
|
r4635 | Copyright 2005-2007 Matt Mackall <mpm@selenic.com> | ||
mpm@selenic.com
|
r1089 | |||
This software may be used and distributed according to the terms | ||||
of the GNU General Public License, incorporated herein by reference. | ||||
""" | ||||
Joel Rosdahl
|
r6211 | from node import nullid | ||
Matt Mackall
|
r3891 | from i18n import _ | ||
Joel Rosdahl
|
r6212 | import struct, os, bisect, stat, strutil, util, errno, ignore | ||
Alexis S. L. Carvalho
|
r6326 | import cStringIO, osutil, sys | ||
mpm@selenic.com
|
r1089 | |||
Matt Mackall
|
r4610 | _unknown = ('?', 0, 0, 0) | ||
_format = ">cllll" | ||||
Eric Hopper
|
r1559 | class dirstate(object): | ||
Benoit Boissinot
|
r2393 | |||
mpm@selenic.com
|
r1089 | def __init__(self, opener, ui, root): | ||
Matt Mackall
|
r4614 | self._opener = opener | ||
self._root = root | ||||
Matt Mackall
|
r4903 | self._dirty = False | ||
Matt Mackall
|
r4965 | self._dirtypl = False | ||
Matt Mackall
|
r4614 | self._ui = ui | ||
mpm@selenic.com
|
r1089 | |||
Matt Mackall
|
r4603 | def __getattr__(self, name): | ||
Matt Mackall
|
r4614 | if name == '_map': | ||
Matt Mackall
|
r4615 | self._read() | ||
Matt Mackall
|
r4614 | return self._map | ||
elif name == '_copymap': | ||||
Matt Mackall
|
r4615 | self._read() | ||
Matt Mackall
|
r4614 | return self._copymap | ||
Matt Mackall
|
r4604 | elif name == '_branch': | ||
try: | ||||
Thomas Arendsen Hein
|
r4633 | self._branch = (self._opener("branch").read().strip() | ||
or "default") | ||||
Matt Mackall
|
r4604 | except IOError: | ||
self._branch = "default" | ||||
return self._branch | ||||
Matt Mackall
|
r4614 | elif name == '_pl': | ||
self._pl = [nullid, nullid] | ||||
Matt Mackall
|
r4604 | try: | ||
Matt Mackall
|
r4614 | st = self._opener("dirstate").read(40) | ||
Matt Mackall
|
r4604 | if len(st) == 40: | ||
Matt Mackall
|
r4614 | self._pl = st[:20], st[20:40] | ||
Matt Mackall
|
r4604 | except IOError, err: | ||
if err.errno != errno.ENOENT: raise | ||||
Matt Mackall
|
r4614 | return self._pl | ||
Matt Mackall
|
r4615 | elif name == '_dirs': | ||
self._dirs = {} | ||||
Matt Mackall
|
r4614 | for f in self._map: | ||
Maxim Dounin
|
r5487 | if self[f] != 'r': | ||
self._incpath(f) | ||||
Matt Mackall
|
r4615 | return self._dirs | ||
Matt Mackall
|
r4609 | elif name == '_ignore': | ||
Matt Mackall
|
r4905 | files = [self._join('.hgignore')] | ||
Matt Mackall
|
r4620 | for name, path in self._ui.configitems("ui"): | ||
if name == 'ignore' or name.startswith('ignore.'): | ||||
files.append(os.path.expanduser(path)) | ||||
Matt Mackall
|
r4614 | self._ignore = ignore.ignore(self._root, files, self._ui.warn) | ||
Matt Mackall
|
r4609 | return self._ignore | ||
Matt Mackall
|
r4611 | elif name == '_slash': | ||
Matt Mackall
|
r4614 | self._slash = self._ui.configbool('ui', 'slash') and os.sep != '/' | ||
Matt Mackall
|
r4611 | return self._slash | ||
Alexis S. L. Carvalho
|
r6257 | elif name == '_checkexec': | ||
self._checkexec = util.checkexec(self._root) | ||||
return self._checkexec | ||||
Matt Mackall
|
r4603 | else: | ||
raise AttributeError, name | ||||
Matt Mackall
|
r4905 | def _join(self, f): | ||
Matt Mackall
|
r4614 | return os.path.join(self._root, f) | ||
mpm@selenic.com
|
r1089 | |||
def getcwd(self): | ||||
cwd = os.getcwd() | ||||
Matt Mackall
|
r4614 | if cwd == self._root: return '' | ||
# self._root ends with a path separator if self._root is '/' or 'C:\' | ||||
rootsep = self._root | ||||
Shun-ichi GOTO
|
r5843 | if not util.endswithsep(rootsep): | ||
Alexis S. L. Carvalho
|
r4230 | rootsep += os.sep | ||
if cwd.startswith(rootsep): | ||||
return cwd[len(rootsep):] | ||||
else: | ||||
# we're outside the repo. return an absolute path. | ||||
return cwd | ||||
mpm@selenic.com
|
r1089 | |||
Alexis S. L. Carvalho
|
r4525 | def pathto(self, f, cwd=None): | ||
if cwd is None: | ||||
cwd = self.getcwd() | ||||
Matt Mackall
|
r4614 | path = util.pathto(self._root, cwd, f) | ||
Alexis S. L. Carvalho
|
r4527 | if self._slash: | ||
Shun-ichi GOTO
|
r5842 | return util.normpath(path) | ||
Alexis S. L. Carvalho
|
r4527 | return path | ||
Alexis S. L. Carvalho
|
r4525 | |||
mpm@selenic.com
|
r1089 | def __getitem__(self, key): | ||
Matt Mackall
|
r4906 | ''' current states: | ||
n normal | ||||
m needs merging | ||||
r marked for removal | ||||
a marked for addition | ||||
? not tracked''' | ||||
return self._map.get(key, ("?",))[0] | ||||
mpm@selenic.com
|
r1089 | |||
def __contains__(self, key): | ||||
Matt Mackall
|
r4614 | return key in self._map | ||
def __iter__(self): | ||||
a = self._map.keys() | ||||
a.sort() | ||||
for x in a: | ||||
yield x | ||||
mpm@selenic.com
|
r1089 | |||
def parents(self): | ||||
Matt Mackall
|
r4614 | return self._pl | ||
mpm@selenic.com
|
r1089 | |||
Matt Mackall
|
r4179 | def branch(self): | ||
return self._branch | ||||
mpm@selenic.com
|
r1089 | def setparents(self, p1, p2=nullid): | ||
Matt Mackall
|
r4965 | self._dirty = self._dirtypl = True | ||
Matt Mackall
|
r4614 | self._pl = p1, p2 | ||
mpm@selenic.com
|
r1089 | |||
Matt Mackall
|
r4179 | def setbranch(self, branch): | ||
self._branch = branch | ||||
Matt Mackall
|
r4614 | self._opener("branch", "w").write(branch + '\n') | ||
Matt Mackall
|
r4179 | |||
Matt Mackall
|
r4615 | def _read(self): | ||
Matt Mackall
|
r4614 | self._map = {} | ||
self._copymap = {} | ||||
Alexis S. L. Carvalho
|
r4952 | if not self._dirtypl: | ||
self._pl = [nullid, nullid] | ||||
Matt Mackall
|
r4607 | try: | ||
Matt Mackall
|
r4614 | st = self._opener("dirstate").read() | ||
Matt Mackall
|
r4607 | except IOError, err: | ||
if err.errno != errno.ENOENT: raise | ||||
return | ||||
if not st: | ||||
return | ||||
Alexis S. L. Carvalho
|
r4952 | if not self._dirtypl: | ||
self._pl = [st[:20], st[20: 40]] | ||||
mpm@selenic.com
|
r1089 | |||
Vadim Gelfer
|
r2427 | # deref fields so they will be local in loop | ||
Matt Mackall
|
r4614 | dmap = self._map | ||
copymap = self._copymap | ||||
Vadim Gelfer
|
r2427 | unpack = struct.unpack | ||
Matt Mackall
|
r4610 | e_size = struct.calcsize(_format) | ||
Matt Mackall
|
r5327 | pos1 = 40 | ||
l = len(st) | ||||
Vadim Gelfer
|
r2427 | |||
Matt Mackall
|
r5327 | # the inner loop | ||
while pos1 < l: | ||||
pos2 = pos1 + e_size | ||||
e = unpack(">cllll", st[pos1:pos2]) # a literal here is faster | ||||
pos1 = pos2 + e[4] | ||||
f = st[pos2:pos1] | ||||
mpm@selenic.com
|
r1089 | if '\0' in f: | ||
f, c = f.split('\0') | ||||
Matt Mackall
|
r3154 | copymap[f] = c | ||
Matt Mackall
|
r5327 | dmap[f] = e # we hold onto e[4] because making a subtuple is slow | ||
mpm@selenic.com
|
r1089 | |||
Matt Mackall
|
r4613 | def invalidate(self): | ||
Alexis S. L. Carvalho
|
r4656 | for a in "_map _copymap _branch _pl _dirs _ignore".split(): | ||
Alexis S. L. Carvalho
|
r4953 | if a in self.__dict__: | ||
delattr(self, a) | ||||
Matt Mackall
|
r4903 | self._dirty = False | ||
Bryan O'Sullivan
|
r4375 | |||
mpm@selenic.com
|
r1089 | def copy(self, source, dest): | ||
Matt Mackall
|
r4903 | self._dirty = True | ||
Matt Mackall
|
r4614 | self._copymap[dest] = source | ||
mpm@selenic.com
|
r1089 | |||
def copied(self, file): | ||||
Matt Mackall
|
r4614 | return self._copymap.get(file, None) | ||
Matt Mackall
|
r3154 | |||
def copies(self): | ||||
Matt Mackall
|
r4614 | return self._copymap | ||
mpm@selenic.com
|
r1089 | |||
Matt Mackall
|
r4615 | def _incpath(self, path): | ||
Matt Mackall
|
r5326 | c = path.rfind('/') | ||
if c >= 0: | ||||
dirs = self._dirs | ||||
base = path[:c] | ||||
if base not in dirs: | ||||
self._incpath(base) | ||||
dirs[base] = 1 | ||||
else: | ||||
dirs[base] += 1 | ||||
Matt Mackall
|
r4615 | |||
def _decpath(self, path): | ||||
Maxim Dounin
|
r5516 | c = path.rfind('/') | ||
if c >= 0: | ||||
base = path[:c] | ||||
dirs = self._dirs | ||||
if dirs[base] == 1: | ||||
del dirs[base] | ||||
self._decpath(base) | ||||
else: | ||||
dirs[base] -= 1 | ||||
Vadim Gelfer
|
r2953 | |||
Matt Mackall
|
r4616 | def _incpathcheck(self, f): | ||
if '\r' in f or '\n' in f: | ||||
Thomas Arendsen Hein
|
r6138 | raise util.Abort(_("'\\n' and '\\r' disallowed in filenames: %r") | ||
% f) | ||||
Matt Mackall
|
r4616 | # shadows | ||
if f in self._dirs: | ||||
Bryan O'Sullivan
|
r5045 | raise util.Abort(_('directory %r already in dirstate') % f) | ||
Matt Mackall
|
r4616 | for c in strutil.rfindall(f, '/'): | ||
d = f[:c] | ||||
if d in self._dirs: | ||||
break | ||||
Maxim Dounin
|
r5487 | if d in self._map and self[d] != 'r': | ||
Bryan O'Sullivan
|
r5045 | raise util.Abort(_('file %r in dirstate clashes with %r') % | ||
(d, f)) | ||||
Matt Mackall
|
r4616 | self._incpath(f) | ||
Vadim Gelfer
|
r2953 | |||
Maxim Dounin
|
r5516 | def _changepath(self, f, newstate, relaxed=False): | ||
Maxim Dounin
|
r5487 | # handle upcoming path changes | ||
oldstate = self[f] | ||||
if oldstate not in "?r" and newstate in "?r": | ||||
Maxim Dounin
|
r5516 | if "_dirs" in self.__dict__: | ||
self._decpath(f) | ||||
Maxim Dounin
|
r5487 | return | ||
if oldstate in "?r" and newstate not in "?r": | ||||
Maxim Dounin
|
r5516 | if relaxed and oldstate == '?': | ||
# XXX | ||||
# in relaxed mode we assume the caller knows | ||||
# what it is doing, workaround for updating | ||||
# dir-to-file revisions | ||||
if "_dirs" in self.__dict__: | ||||
self._incpath(f) | ||||
return | ||||
Maxim Dounin
|
r5487 | self._incpathcheck(f) | ||
return | ||||
Matt Mackall
|
r4904 | def normal(self, f): | ||
Alexis S. L. Carvalho
|
r5210 | 'mark a file normal and clean' | ||
Matt Mackall
|
r4904 | self._dirty = True | ||
Maxim Dounin
|
r5516 | self._changepath(f, 'n', True) | ||
Matt Mackall
|
r4905 | s = os.lstat(self._join(f)) | ||
Matt Mackall
|
r5327 | self._map[f] = ('n', s.st_mode, s.st_size, s.st_mtime, 0) | ||
Christian Ebert
|
r5915 | if f in self._copymap: | ||
Matt Mackall
|
r4904 | del self._copymap[f] | ||
mpm@selenic.com
|
r1089 | |||
Alexis S. L. Carvalho
|
r5210 | def normallookup(self, f): | ||
Matt Mackall
|
r4904 | 'mark a file normal, but possibly dirty' | ||
Alexis S. L. Carvalho
|
r6298 | if self._pl[1] != nullid and f in self._map: | ||
# if there is a merge going on and the file was either | ||||
# in state 'm' or dirty before being removed, restore that state. | ||||
entry = self._map[f] | ||||
if entry[0] == 'r' and entry[2] in (-1, -2): | ||||
source = self._copymap.get(f) | ||||
if entry[2] == -1: | ||||
self.merge(f) | ||||
elif entry[2] == -2: | ||||
self.normaldirty(f) | ||||
if source: | ||||
self.copy(source, f) | ||||
return | ||||
if entry[0] == 'm' or entry[0] == 'n' and entry[2] == -2: | ||||
return | ||||
Matt Mackall
|
r4903 | self._dirty = True | ||
Maxim Dounin
|
r5516 | self._changepath(f, 'n', True) | ||
Matt Mackall
|
r5327 | self._map[f] = ('n', 0, -1, -1, 0) | ||
Alexis S. L. Carvalho
|
r5210 | if f in self._copymap: | ||
del self._copymap[f] | ||||
def normaldirty(self, f): | ||||
'mark a file normal, but dirty' | ||||
self._dirty = True | ||||
Maxim Dounin
|
r5516 | self._changepath(f, 'n', True) | ||
Matt Mackall
|
r5327 | self._map[f] = ('n', 0, -2, -1, 0) | ||
Matt Mackall
|
r4904 | if f in self._copymap: | ||
del self._copymap[f] | ||||
def add(self, f): | ||||
'mark a file added' | ||||
self._dirty = True | ||||
Maxim Dounin
|
r5487 | self._changepath(f, 'a') | ||
Matt Mackall
|
r5327 | self._map[f] = ('a', 0, -1, -1, 0) | ||
Matt Mackall
|
r4904 | if f in self._copymap: | ||
del self._copymap[f] | ||||
Matt Mackall
|
r4616 | |||
Matt Mackall
|
r4904 | def remove(self, f): | ||
'mark a file removed' | ||||
self._dirty = True | ||||
Maxim Dounin
|
r5487 | self._changepath(f, 'r') | ||
Alexis S. L. Carvalho
|
r6297 | size = 0 | ||
if self._pl[1] != nullid and f in self._map: | ||||
entry = self._map[f] | ||||
if entry[0] == 'm': | ||||
size = -1 | ||||
elif entry[0] == 'n' and entry[2] == -2: | ||||
size = -2 | ||||
self._map[f] = ('r', 0, size, 0, 0) | ||||
if size == 0 and f in self._copymap: | ||||
Matt Mackall
|
r4904 | del self._copymap[f] | ||
mpm@selenic.com
|
r1089 | |||
Matt Mackall
|
r4904 | def merge(self, f): | ||
'mark a file merged' | ||||
Matt Mackall
|
r4903 | self._dirty = True | ||
Matt Mackall
|
r4905 | s = os.lstat(self._join(f)) | ||
Maxim Dounin
|
r5516 | self._changepath(f, 'm', True) | ||
Matt Mackall
|
r5327 | self._map[f] = ('m', s.st_mode, s.st_size, s.st_mtime, 0) | ||
Matt Mackall
|
r4904 | if f in self._copymap: | ||
del self._copymap[f] | ||||
def forget(self, f): | ||||
'forget a file' | ||||
self._dirty = True | ||||
try: | ||||
Maxim Dounin
|
r5487 | self._changepath(f, '?') | ||
Matt Mackall
|
r4904 | del self._map[f] | ||
except KeyError: | ||||
Thomas Arendsen Hein
|
r6057 | self._ui.warn(_("not in dirstate: %s\n") % f) | ||
mpm@selenic.com
|
r1089 | |||
Alexis S. L. Carvalho
|
r5065 | def clear(self): | ||
self._map = {} | ||||
Maxim Dounin
|
r5487 | if "_dirs" in self.__dict__: | ||
delattr(self, "_dirs"); | ||||
Alexis S. L. Carvalho
|
r5065 | self._copymap = {} | ||
self._pl = [nullid, nullid] | ||||
Alexis S. L. Carvalho
|
r5123 | self._dirty = True | ||
Alexis S. L. Carvalho
|
r5065 | |||
Benoit Boissinot
|
r1755 | def rebuild(self, parent, files): | ||
Alexis S. L. Carvalho
|
r5065 | self.clear() | ||
Matt Mackall
|
r2832 | for f in files: | ||
if files.execf(f): | ||||
Matt Mackall
|
r5327 | self._map[f] = ('n', 0777, -1, 0, 0) | ||
Benoit Boissinot
|
r1755 | else: | ||
Matt Mackall
|
r5327 | self._map[f] = ('n', 0666, -1, 0, 0) | ||
Matt Mackall
|
r4614 | self._pl = (parent, nullid) | ||
Matt Mackall
|
r4903 | self._dirty = True | ||
mpm@selenic.com
|
r1089 | |||
def write(self): | ||||
Matt Mackall
|
r4612 | if not self._dirty: | ||
Benoit Boissinot
|
r1794 | return | ||
Alexis S. L. Carvalho
|
r6326 | st = self._opener("dirstate", "w", atomictemp=True) | ||
Matt Mackall
|
r6327 | |||
try: | ||||
gran = int(self._ui.config('dirstate', 'granularity', 1)) | ||||
except ValueError: | ||||
gran = 1 | ||||
limit = sys.maxint | ||||
if gran > 0: | ||||
limit = util.fstat(st).st_mtime - gran | ||||
Bryan O'Sullivan
|
r4374 | cs = cStringIO.StringIO() | ||
Matt Mackall
|
r5327 | copymap = self._copymap | ||
pack = struct.pack | ||||
write = cs.write | ||||
write("".join(self._pl)) | ||||
Matt Mackall
|
r4614 | for f, e in self._map.iteritems(): | ||
Matt Mackall
|
r5327 | if f in copymap: | ||
f = "%s\0%s" % (f, copymap[f]) | ||||
Alexis S. L. Carvalho
|
r6326 | if e[3] > limit and e[0] == 'n': | ||
e = (e[0], 0, -1, -1, 0) | ||||
Matt Mackall
|
r5327 | e = pack(_format, e[0], e[1], e[2], e[3], len(f)) | ||
write(e) | ||||
write(f) | ||||
Bryan O'Sullivan
|
r4374 | st.write(cs.getvalue()) | ||
Alexis S. L. Carvalho
|
r4507 | st.rename() | ||
Matt Mackall
|
r4965 | self._dirty = self._dirtypl = False | ||
mpm@selenic.com
|
r1089 | |||
Matt Mackall
|
r4907 | def _filter(self, files): | ||
mpm@selenic.com
|
r1089 | ret = {} | ||
unknown = [] | ||||
for x in files: | ||||
twaldmann@thinkmo.de
|
r1541 | if x == '.': | ||
Matt Mackall
|
r4614 | return self._map.copy() | ||
if x not in self._map: | ||||
mpm@selenic.com
|
r1089 | unknown.append(x) | ||
else: | ||||
Matt Mackall
|
r4614 | ret[x] = self._map[x] | ||
mpm@selenic.com
|
r1089 | |||
if not unknown: | ||||
return ret | ||||
Matt Mackall
|
r4614 | b = self._map.keys() | ||
mpm@selenic.com
|
r1089 | b.sort() | ||
blen = len(b) | ||||
for x in unknown: | ||||
Benoit Boissinot
|
r2486 | bs = bisect.bisect(b, "%s%s" % (x, '/')) | ||
mpm@selenic.com
|
r1089 | while bs < blen: | ||
s = b[bs] | ||||
Brendan Cully
|
r2485 | if len(s) > len(x) and s.startswith(x): | ||
Matt Mackall
|
r4614 | ret[s] = self._map[s] | ||
mpm@selenic.com
|
r1089 | else: | ||
break | ||||
bs += 1 | ||||
return ret | ||||
Matt Mackall
|
r5001 | def _supported(self, f, mode, verbose=False): | ||
if stat.S_ISREG(mode) or stat.S_ISLNK(mode): | ||||
Benoit Boissinot
|
r1487 | return True | ||
if verbose: | ||||
kind = 'unknown' | ||||
Matt Mackall
|
r5001 | if stat.S_ISCHR(mode): kind = _('character device') | ||
elif stat.S_ISBLK(mode): kind = _('block device') | ||||
elif stat.S_ISFIFO(mode): kind = _('fifo') | ||||
elif stat.S_ISSOCK(mode): kind = _('socket') | ||||
elif stat.S_ISDIR(mode): kind = _('directory') | ||||
Matt Mackall
|
r4614 | self._ui.warn(_('%s: unsupported file type (type is %s)\n') | ||
Thomas Arendsen Hein
|
r4633 | % (self.pathto(f), kind)) | ||
Benoit Boissinot
|
r1487 | return False | ||
Alexis S. L. Carvalho
|
r6032 | def _dirignore(self, f): | ||
Patrick Mezard
|
r6479 | if f == '.': | ||
return False | ||||
Alexis S. L. Carvalho
|
r6032 | if self._ignore(f): | ||
return True | ||||
for c in strutil.findall(f, '/'): | ||||
if self._ignore(f[:c]): | ||||
return True | ||||
return False | ||||
Matt Mackall
|
r6578 | def walk(self, match): | ||
Matt Mackall
|
r6587 | # filter out the src and stat | ||
Matt Mackall
|
r6589 | for src, f, st in self.statwalk(match.files(), match): | ||
Matt Mackall
|
r6587 | yield f | ||
Matt Mackall
|
r3529 | |||
Matt Mackall
|
r6589 | def statwalk(self, files, match, unknown=True, ignored=False): | ||
Matt Mackall
|
r3529 | ''' | ||
walk recursively through the directory tree, finding all files | ||||
matched by the match function | ||||
results are yielded in a tuple (src, filename, st), where src | ||||
is one of: | ||||
'f' the file was found in the directory tree | ||||
'm' the file was only in the dirstate and not in the tree | ||||
Matt Mackall
|
r3532 | |||
Matt Mackall
|
r3529 | and st is the stat result if the file was found in the directory. | ||
''' | ||||
mpm@selenic.com
|
r1089 | |||
Matt Mackall
|
r6578 | def fwarn(f, msg): | ||
self._ui.warn('%s: %s\n' % (self.pathto(ff), msg)) | ||||
return False | ||||
Matt Mackall
|
r6589 | badfn = fwarn | ||
if hasattr(match, 'bad'): | ||||
badfn = match.bad | ||||
Matt Mackall
|
r6578 | |||
mpm@selenic.com
|
r1089 | # walk all files by default | ||
if not files: | ||||
Alexis S. L. Carvalho
|
r4172 | files = ['.'] | ||
Matt Mackall
|
r4614 | dc = self._map.copy() | ||
Matt Mackall
|
r3529 | else: | ||
Matt Mackall
|
r3536 | files = util.unique(files) | ||
Matt Mackall
|
r4907 | dc = self._filter(files) | ||
mpm@selenic.com
|
r1089 | |||
Matt Mackall
|
r3529 | def imatch(file_): | ||
Matt Mackall
|
r4609 | if file_ not in dc and self._ignore(file_): | ||
mason@suse.com
|
r1183 | return False | ||
Benoit Boissinot
|
r1749 | return match(file_) | ||
mpm@selenic.com
|
r1224 | |||
Thomas Arendsen Hein
|
r6201 | # TODO: don't walk unknown directories if unknown and ignored are False | ||
Matt Mackall
|
r4609 | ignore = self._ignore | ||
Alexis S. L. Carvalho
|
r6032 | dirignore = self._dirignore | ||
Alexis S. L. Carvalho
|
r4193 | if ignored: | ||
imatch = match | ||||
ignore = util.never | ||||
Alexis S. L. Carvalho
|
r6032 | dirignore = util.never | ||
Matt Mackall
|
r3534 | |||
Matt Mackall
|
r4614 | # self._root may end with a path separator when self._root == '/' | ||
common_prefix_len = len(self._root) | ||||
Shun-ichi GOTO
|
r5843 | if not util.endswithsep(self._root): | ||
Benoit Boissinot
|
r2671 | common_prefix_len += 1 | ||
Matt Mackall
|
r5000 | |||
normpath = util.normpath | ||||
Bryan O'Sullivan
|
r5396 | listdir = osutil.listdir | ||
Matt Mackall
|
r5000 | lstat = os.lstat | ||
bisect_left = bisect.bisect_left | ||||
isdir = os.path.isdir | ||||
pconvert = util.pconvert | ||||
join = os.path.join | ||||
s_isdir = stat.S_ISDIR | ||||
supported = self._supported | ||||
Matt Mackall
|
r5001 | _join = self._join | ||
known = {'.hg': 1} | ||||
Matt Mackall
|
r5000 | |||
mason@suse.com
|
r1183 | # recursion free walker, faster than os.walk. | ||
def findfiles(s): | ||||
work = [s] | ||||
Matt Mackall
|
r5002 | wadd = work.append | ||
found = [] | ||||
add = found.append | ||||
Matt Mackall
|
r6588 | if hasattr(match, 'dir'): | ||
match.dir(normpath(s[common_prefix_len:])) | ||||
mason@suse.com
|
r1183 | while work: | ||
top = work.pop() | ||||
Bryan O'Sullivan
|
r5396 | entries = listdir(top, stat=True) | ||
mason@suse.com
|
r1183 | # nd is the top of the repository dir tree | ||
Matt Mackall
|
r5000 | nd = normpath(top[common_prefix_len:]) | ||
Vadim Gelfer
|
r2061 | if nd == '.': | ||
nd = '' | ||||
else: | ||||
Vadim Gelfer
|
r2063 | # do not recurse into a repo contained in this | ||
# one. use bisect to find .hg directory so speed | ||||
# is good on big directory. | ||||
Bryan O'Sullivan
|
r5396 | names = [e[0] for e in entries] | ||
Matt Mackall
|
r5000 | hg = bisect_left(names, '.hg') | ||
Vadim Gelfer
|
r2061 | if hg < len(names) and names[hg] == '.hg': | ||
Matt Mackall
|
r5000 | if isdir(join(top, '.hg')): | ||
Vadim Gelfer
|
r2061 | continue | ||
Bryan O'Sullivan
|
r5396 | for f, kind, st in entries: | ||
Matt Mackall
|
r5001 | np = pconvert(join(nd, f)) | ||
if np in known: | ||||
mason@suse.com
|
r1183 | continue | ||
Matt Mackall
|
r5001 | known[np] = 1 | ||
Matt Mackall
|
r5000 | p = join(top, f) | ||
mpm@selenic.com
|
r1228 | # don't trip over symlinks | ||
Bryan O'Sullivan
|
r5396 | if kind == stat.S_IFDIR: | ||
Alexis S. L. Carvalho
|
r4254 | if not ignore(np): | ||
Matt Mackall
|
r5002 | wadd(p) | ||
Matt Mackall
|
r6588 | if hasattr(match, 'dir'): | ||
match.dir(np) | ||||
Matt Mackall
|
r5001 | if np in dc and match(np): | ||
Matt Mackall
|
r5002 | add((np, 'm', st)) | ||
Matt Mackall
|
r3529 | elif imatch(np): | ||
Matt Mackall
|
r5001 | if supported(np, st.st_mode): | ||
Matt Mackall
|
r5002 | add((np, 'f', st)) | ||
Benoit Boissinot
|
r1487 | elif np in dc: | ||
Matt Mackall
|
r5002 | add((np, 'm', st)) | ||
found.sort() | ||||
return found | ||||
mason@suse.com
|
r1183 | |||
# step one, find all files that match our criteria | ||||
files.sort() | ||||
Matt Mackall
|
r3536 | for ff in files: | ||
Matt Mackall
|
r5000 | nf = normpath(ff) | ||
Matt Mackall
|
r5001 | f = _join(ff) | ||
mason@suse.com
|
r1183 | try: | ||
Matt Mackall
|
r5000 | st = lstat(f) | ||
mason@suse.com
|
r1183 | except OSError, inst: | ||
Benoit Boissinot
|
r1564 | found = False | ||
for fn in dc: | ||||
if nf == fn or (fn.startswith(nf) and fn[len(nf)] == '/'): | ||||
found = True | ||||
break | ||||
if not found: | ||||
Matt Mackall
|
r6578 | if inst.errno != errno.ENOENT: | ||
fwarn(ff, inst.strerror) | ||||
elif badfn(ff, inst.strerror) and imatch(nf): | ||||
Matt Mackall
|
r6583 | yield 'f', ff, None | ||
mason@suse.com
|
r1183 | continue | ||
Matt Mackall
|
r5000 | if s_isdir(st.st_mode): | ||
Alexis S. L. Carvalho
|
r6032 | if not dirignore(nf): | ||
for f, src, st in findfiles(f): | ||||
yield src, f, st | ||||
Benoit Boissinot
|
r1392 | else: | ||
Matt Mackall
|
r5001 | if nf in known: | ||
continue | ||||
known[nf] = 1 | ||||
if match(nf): | ||||
if supported(ff, st.st_mode, verbose=True): | ||||
Matt Mackall
|
r3536 | yield 'f', nf, st | ||
Benoit Boissinot
|
r1487 | elif ff in dc: | ||
Matt Mackall
|
r3536 | yield 'm', nf, st | ||
mpm@selenic.com
|
r1089 | |||
mason@suse.com
|
r1183 | # step two run through anything left in the dc hash and yield | ||
# if we haven't already seen it | ||||
ks = dc.keys() | ||||
ks.sort() | ||||
for k in ks: | ||||
Matt Mackall
|
r5001 | if k in known: | ||
continue | ||||
known[k] = 1 | ||||
if imatch(k): | ||||
Benoit Boissinot
|
r1471 | yield 'm', k, None | ||
mpm@selenic.com
|
r1089 | |||
Thomas Arendsen Hein
|
r6201 | def status(self, files, match, list_ignored, list_clean, list_unknown=True): | ||
Thomas Arendsen Hein
|
r2022 | lookup, modified, added, unknown, ignored = [], [], [], [], [] | ||
Vadim Gelfer
|
r2661 | removed, deleted, clean = [], [], [] | ||
mpm@selenic.com
|
r1089 | |||
Alexis S. L. Carvalho
|
r6033 | files = files or [] | ||
Matt Mackall
|
r5003 | _join = self._join | ||
lstat = os.lstat | ||||
cmap = self._copymap | ||||
dmap = self._map | ||||
ladd = lookup.append | ||||
madd = modified.append | ||||
aadd = added.append | ||||
uadd = unknown.append | ||||
iadd = ignored.append | ||||
radd = removed.append | ||||
dadd = deleted.append | ||||
cadd = clean.append | ||||
Thomas Arendsen Hein
|
r6201 | for src, fn, st in self.statwalk(files, match, unknown=list_unknown, | ||
ignored=list_ignored): | ||||
Matt Mackall
|
r5003 | if fn in dmap: | ||
Matt Mackall
|
r6590 | state, mode, size, time, foo = dmap[fn] | ||
Matt Mackall
|
r5003 | else: | ||
Alexis S. L. Carvalho
|
r6033 | if (list_ignored or fn in files) and self._dirignore(fn): | ||
if list_ignored: | ||||
iadd(fn) | ||||
Thomas Arendsen Hein
|
r6201 | elif list_unknown: | ||
Matt Mackall
|
r5003 | uadd(fn) | ||
Benoit Boissinot
|
r1471 | continue | ||
Benoit Boissinot
|
r1476 | if src == 'm': | ||
Benoit Boissinot
|
r1487 | nonexistent = True | ||
if not st: | ||||
try: | ||||
Matt Mackall
|
r5003 | st = lstat(_join(fn)) | ||
Benoit Boissinot
|
r1487 | except OSError, inst: | ||
Maxim Dounin
|
r5487 | if inst.errno not in (errno.ENOENT, errno.ENOTDIR): | ||
Benoit Boissinot
|
r1487 | raise | ||
st = None | ||||
# We need to re-check that it is a valid file | ||||
Matt Mackall
|
r5001 | if st and self._supported(fn, st.st_mode): | ||
Benoit Boissinot
|
r1487 | nonexistent = False | ||
Benoit Boissinot
|
r1476 | # XXX: what to do with file no longer present in the fs | ||
# who are not removed in the dirstate ? | ||||
Matt Mackall
|
r6590 | if nonexistent and state in "nma": | ||
Matt Mackall
|
r5003 | dadd(fn) | ||
Benoit Boissinot
|
r1476 | continue | ||
Benoit Boissinot
|
r1471 | # check the common case first | ||
Matt Mackall
|
r6590 | if state == 'n': | ||
Benoit Boissinot
|
r1471 | if not st: | ||
Matt Mackall
|
r5003 | st = lstat(_join(fn)) | ||
Alexis S. L. Carvalho
|
r6257 | if (size >= 0 and | ||
(size != st.st_size | ||||
or ((mode ^ st.st_mode) & 0100 and self._checkexec)) | ||||
Alexis S. L. Carvalho
|
r5210 | or size == -2 | ||
Alexis S. L. Carvalho
|
r4677 | or fn in self._copymap): | ||
Matt Mackall
|
r5003 | madd(fn) | ||
Alexis S. L. Carvalho
|
r2962 | elif time != int(st.st_mtime): | ||
Matt Mackall
|
r5003 | ladd(fn) | ||
Vadim Gelfer
|
r2661 | elif list_clean: | ||
Matt Mackall
|
r5003 | cadd(fn) | ||
Matt Mackall
|
r6590 | elif state == 'm': | ||
Matt Mackall
|
r5003 | madd(fn) | ||
Matt Mackall
|
r6590 | elif state == 'a': | ||
Matt Mackall
|
r5003 | aadd(fn) | ||
Matt Mackall
|
r6590 | elif state == 'r': | ||
Matt Mackall
|
r5003 | radd(fn) | ||
mason@suse.com
|
r1183 | |||
Vadim Gelfer
|
r2661 | return (lookup, modified, added, removed, deleted, unknown, ignored, | ||
clean) | ||||