dirstate.py
507 lines
| 16.2 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. | ||||
""" | ||||
mpm@selenic.com
|
r1094 | from node import * | ||
Matt Mackall
|
r3891 | from i18n import _ | ||
Matt Mackall
|
r4609 | import struct, os, time, bisect, stat, strutil, util, re, errno, ignore | ||
Bryan O'Sullivan
|
r4374 | import cStringIO | ||
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
|
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: | ||
Matt Mackall
|
r4615 | self._incpath(f) | ||
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 | ||
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 | ||||
Alexis S. L. Carvalho
|
r4230 | if not rootsep.endswith(os.sep): | ||
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: | ||
Matt Mackall
|
r4611 | return path.replace(os.sep, '/') | ||
Alexis S. L. Carvalho
|
r4527 | return path | ||
Alexis S. L. Carvalho
|
r4525 | |||
mpm@selenic.com
|
r1089 | def __del__(self): | ||
Matt Mackall
|
r4612 | self.write() | ||
mpm@selenic.com
|
r1089 | |||
def __getitem__(self, key): | ||||
Matt Mackall
|
r4614 | return self._map[key] | ||
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
|
r4903 | self._dirty = 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 | |||
mpm@selenic.com
|
r1089 | def state(self, key): | ||
Matt Mackall
|
r4904 | ''' current states: | ||
n normal | ||||
m needs merging | ||||
r marked for removal | ||||
a marked for addition''' | ||||
Matt Mackall
|
r4614 | return self._map.get(key, ("?",))[0] | ||
mpm@selenic.com
|
r1089 | |||
Matt Mackall
|
r4615 | def _read(self): | ||
Matt Mackall
|
r4614 | self._map = {} | ||
self._copymap = {} | ||||
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 | ||||
Matt Mackall
|
r4614 | 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 | ||
mpm@selenic.com
|
r1089 | pos = 40 | ||
Matt Mackall
|
r4610 | e_size = struct.calcsize(_format) | ||
Vadim Gelfer
|
r2427 | |||
mpm@selenic.com
|
r1089 | while pos < len(st): | ||
Vadim Gelfer
|
r2425 | newpos = pos + e_size | ||
Matt Mackall
|
r4610 | e = unpack(_format, st[pos:newpos]) | ||
mpm@selenic.com
|
r1089 | l = e[4] | ||
Vadim Gelfer
|
r2425 | pos = newpos | ||
newpos = pos + l | ||||
f = st[pos:newpos] | ||||
mpm@selenic.com
|
r1089 | if '\0' in f: | ||
f, c = f.split('\0') | ||||
Matt Mackall
|
r3154 | copymap[f] = c | ||
Matt Mackall
|
r4607 | dmap[f] = e[:4] | ||
Vadim Gelfer
|
r2425 | pos = newpos | ||
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(): | ||
Matt Mackall
|
r4606 | if hasattr(self, a): | ||
self.__delattr__(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
|
r4605 | for c in strutil.findall(path, '/'): | ||
pc = path[:c] | ||||
Matt Mackall
|
r4615 | self._dirs.setdefault(pc, 0) | ||
self._dirs[pc] += 1 | ||||
def _decpath(self, path): | ||||
for c in strutil.findall(path, '/'): | ||||
pc = path[:c] | ||||
self._dirs.setdefault(pc, 0) | ||||
self._dirs[pc] -= 1 | ||||
Vadim Gelfer
|
r2953 | |||
Matt Mackall
|
r4616 | def _incpathcheck(self, f): | ||
if '\r' in f or '\n' in f: | ||||
raise util.Abort(_("'\\n' and '\\r' disallowed in filenames")) | ||||
# shadows | ||||
if f in self._dirs: | ||||
raise util.Abort(_('directory named %r already in dirstate') % f) | ||||
for c in strutil.rfindall(f, '/'): | ||||
d = f[:c] | ||||
if d in self._dirs: | ||||
break | ||||
if d in self._map: | ||||
raise util.Abort(_('file named %r already in dirstate') % d) | ||||
self._incpath(f) | ||||
Vadim Gelfer
|
r2953 | |||
Matt Mackall
|
r4904 | def normal(self, f): | ||
'mark a file normal' | ||||
self._dirty = True | ||||
Matt Mackall
|
r4905 | s = os.lstat(self._join(f)) | ||
Matt Mackall
|
r4904 | self._map[f] = ('n', s.st_mode, s.st_size, s.st_mtime) | ||
if self._copymap.has_key(f): | ||||
del self._copymap[f] | ||||
mpm@selenic.com
|
r1089 | |||
Matt Mackall
|
r4904 | def normaldirty(self, f): | ||
'mark a file normal, but possibly dirty' | ||||
Matt Mackall
|
r4903 | self._dirty = True | ||
Matt Mackall
|
r4905 | s = os.lstat(self._join(f)) | ||
Matt Mackall
|
r4904 | self._map[f] = ('n', s.st_mode, -1, -1) | ||
if f in self._copymap: | ||||
del self._copymap[f] | ||||
def add(self, f): | ||||
'mark a file added' | ||||
self._dirty = True | ||||
self._incpathcheck(f) | ||||
Matt Mackall
|
r4905 | s = os.lstat(self._join(f)) | ||
Matt Mackall
|
r4904 | self._map[f] = ('a', s.st_mode, s.st_size, s.st_mtime) | ||
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 | ||||
self._map[f] = ('r', 0, 0, 0) | ||||
self._decpath(f) | ||||
if f in self._copymap: | ||||
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)) | ||
Matt Mackall
|
r4904 | self._map[f] = ('m', s.st_mode, s.st_size, s.st_mtime) | ||
if f in self._copymap: | ||||
del self._copymap[f] | ||||
def forget(self, f): | ||||
'forget a file' | ||||
self._dirty = True | ||||
try: | ||||
del self._map[f] | ||||
self._decpath(f) | ||||
except KeyError: | ||||
self._ui.warn(_("not in dirstate: %s!\n") % f) | ||||
mpm@selenic.com
|
r1089 | |||
Benoit Boissinot
|
r1755 | def rebuild(self, parent, files): | ||
Matt Mackall
|
r4613 | self.invalidate() | ||
Matt Mackall
|
r2832 | for f in files: | ||
if files.execf(f): | ||||
Matt Mackall
|
r4614 | self._map[f] = ('n', 0777, -1, 0) | ||
Benoit Boissinot
|
r1755 | else: | ||
Matt Mackall
|
r4614 | self._map[f] = ('n', 0666, -1, 0) | ||
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 | ||
Bryan O'Sullivan
|
r4374 | cs = cStringIO.StringIO() | ||
Matt Mackall
|
r4614 | cs.write("".join(self._pl)) | ||
for f, e in self._map.iteritems(): | ||||
mpm@selenic.com
|
r1089 | c = self.copied(f) | ||
if c: | ||||
f = f + "\0" + c | ||||
Matt Mackall
|
r4610 | e = struct.pack(_format, e[0], e[1], e[2], e[3], len(f)) | ||
Bryan O'Sullivan
|
r4374 | cs.write(e) | ||
cs.write(f) | ||||
Matt Mackall
|
r4614 | st = self._opener("dirstate", "w", atomictemp=True) | ||
Bryan O'Sullivan
|
r4374 | st.write(cs.getvalue()) | ||
Alexis S. L. Carvalho
|
r4507 | st.rename() | ||
Matt Mackall
|
r4903 | self._dirty = False | ||
mpm@selenic.com
|
r1089 | |||
def filterfiles(self, files): | ||||
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
|
r4615 | def _supported(self, f, st, verbose=False): | ||
Matt Mackall
|
r4001 | if stat.S_ISREG(st.st_mode) or stat.S_ISLNK(st.st_mode): | ||
Benoit Boissinot
|
r1487 | return True | ||
if verbose: | ||||
kind = 'unknown' | ||||
if stat.S_ISCHR(st.st_mode): kind = _('character device') | ||||
elif stat.S_ISBLK(st.st_mode): kind = _('block device') | ||||
elif stat.S_ISFIFO(st.st_mode): kind = _('fifo') | ||||
elif stat.S_ISSOCK(st.st_mode): kind = _('socket') | ||||
elif stat.S_ISDIR(st.st_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 | ||
Matt Mackall
|
r3529 | def walk(self, files=None, match=util.always, badmatch=None): | ||
# filter out the stat | ||||
for src, f, st in self.statwalk(files, match, badmatch=badmatch): | ||||
yield src, f | ||||
def statwalk(self, files=None, match=util.always, ignored=False, | ||||
Emanuele Aina
|
r4146 | badmatch=None, directories=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 | ||||
Emanuele Aina
|
r4146 | 'd' the file is a directory of the tree | ||
Matt Mackall
|
r3529 | 'm' the file was only in the dirstate and not in the tree | ||
Matt Mackall
|
r3532 | 'b' file was not found and matched badmatch | ||
Matt Mackall
|
r3529 | and st is the stat result if the file was found in the directory. | ||
''' | ||||
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) | ||
mpm@selenic.com
|
r1089 | dc = self.filterfiles(files) | ||
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 | |||
Matt Mackall
|
r4609 | ignore = self._ignore | ||
Alexis S. L. Carvalho
|
r4193 | if ignored: | ||
imatch = match | ||||
ignore = 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) | ||||
if not self._root.endswith(os.sep): | ||||
Benoit Boissinot
|
r2671 | common_prefix_len += 1 | ||
mason@suse.com
|
r1183 | # recursion free walker, faster than os.walk. | ||
def findfiles(s): | ||||
work = [s] | ||||
Emanuele Aina
|
r4146 | if directories: | ||
yield 'd', util.normpath(s[common_prefix_len:]), os.lstat(s) | ||||
mason@suse.com
|
r1183 | while work: | ||
top = work.pop() | ||||
names = os.listdir(top) | ||||
names.sort() | ||||
# nd is the top of the repository dir tree | ||||
Benoit Boissinot
|
r2671 | nd = util.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. | ||||
Vadim Gelfer
|
r2061 | hg = bisect.bisect_left(names, '.hg') | ||
if hg < len(names) and names[hg] == '.hg': | ||||
if os.path.isdir(os.path.join(top, '.hg')): | ||||
continue | ||||
mason@suse.com
|
r1183 | for f in names: | ||
Christian Boos
|
r1562 | np = util.pconvert(os.path.join(nd, f)) | ||
mason@suse.com
|
r1183 | if seen(np): | ||
continue | ||||
p = os.path.join(top, f) | ||||
mpm@selenic.com
|
r1228 | # don't trip over symlinks | ||
st = os.lstat(p) | ||||
mason@suse.com
|
r1183 | if stat.S_ISDIR(st.st_mode): | ||
Alexis S. L. Carvalho
|
r4254 | if not ignore(np): | ||
mason@suse.com
|
r1183 | work.append(p) | ||
Emanuele Aina
|
r4146 | if directories: | ||
yield 'd', np, st | ||||
Matt Mackall
|
r3529 | if imatch(np) and np in dc: | ||
Christian Boos
|
r1562 | yield 'm', np, st | ||
Matt Mackall
|
r3529 | elif imatch(np): | ||
Matt Mackall
|
r4615 | if self._supported(np, st): | ||
Christian Boos
|
r1562 | yield 'f', np, st | ||
Benoit Boissinot
|
r1487 | elif np in dc: | ||
Christian Boos
|
r1562 | yield 'm', np, st | ||
Benoit Boissinot
|
r1392 | |||
mpm@selenic.com
|
r1089 | known = {'.hg': 1} | ||
def seen(fn): | ||||
if fn in known: return True | ||||
known[fn] = 1 | ||||
mason@suse.com
|
r1183 | |||
# step one, find all files that match our criteria | ||||
files.sort() | ||||
Matt Mackall
|
r3536 | for ff in files: | ||
nf = util.normpath(ff) | ||||
Matt Mackall
|
r4905 | f = self._join(ff) | ||
mason@suse.com
|
r1183 | try: | ||
mpm@selenic.com
|
r1230 | st = os.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: | ||||
Vadim Gelfer
|
r2042 | if inst.errno != errno.ENOENT or not badmatch: | ||
Thomas Arendsen Hein
|
r4633 | self._ui.warn('%s: %s\n' % | ||
(self.pathto(ff), inst.strerror)) | ||||
Matt Mackall
|
r3534 | elif badmatch and badmatch(ff) and imatch(nf): | ||
Vadim Gelfer
|
r2042 | yield 'b', ff, None | ||
mason@suse.com
|
r1183 | continue | ||
if stat.S_ISDIR(st.st_mode): | ||||
Benoit Boissinot
|
r1487 | cmp1 = (lambda x, y: cmp(x[1], y[1])) | ||
Benoit Boissinot
|
r1749 | sorted_ = [ x for x in findfiles(f) ] | ||
sorted_.sort(cmp1) | ||||
for e in sorted_: | ||||
Benoit Boissinot
|
r1487 | yield e | ||
Benoit Boissinot
|
r1392 | else: | ||
Matt Mackall
|
r3536 | if not seen(nf) and match(nf): | ||
Matt Mackall
|
r4615 | if self._supported(ff, st, 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
|
r3529 | if not seen(k) and imatch(k): | ||
Benoit Boissinot
|
r1471 | yield 'm', k, None | ||
mpm@selenic.com
|
r1089 | |||
Vadim Gelfer
|
r2661 | def status(self, files=None, match=util.always, list_ignored=False, | ||
list_clean=False): | ||||
Thomas Arendsen Hein
|
r2022 | lookup, modified, added, unknown, ignored = [], [], [], [], [] | ||
Vadim Gelfer
|
r2661 | removed, deleted, clean = [], [], [] | ||
mpm@selenic.com
|
r1089 | |||
Vadim Gelfer
|
r2661 | for src, fn, st in self.statwalk(files, match, ignored=list_ignored): | ||
Benoit Boissinot
|
r1471 | try: | ||
Benoit Boissinot
|
r1749 | type_, mode, size, time = self[fn] | ||
Benoit Boissinot
|
r1471 | except KeyError: | ||
Matt Mackall
|
r4609 | if list_ignored and self._ignore(fn): | ||
Thomas Arendsen Hein
|
r2022 | ignored.append(fn) | ||
else: | ||||
unknown.append(fn) | ||||
Benoit Boissinot
|
r1471 | continue | ||
Benoit Boissinot
|
r1476 | if src == 'm': | ||
Benoit Boissinot
|
r1487 | nonexistent = True | ||
if not st: | ||||
try: | ||||
Matt Mackall
|
r4905 | st = os.lstat(self._join(fn)) | ||
Benoit Boissinot
|
r1487 | except OSError, inst: | ||
if inst.errno != errno.ENOENT: | ||||
raise | ||||
st = None | ||||
# We need to re-check that it is a valid file | ||||
Matt Mackall
|
r4615 | if st and self._supported(fn, st): | ||
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 ? | ||||
Benoit Boissinot
|
r1749 | if nonexistent and type_ in "nm": | ||
Benoit Boissinot
|
r1476 | deleted.append(fn) | ||
continue | ||||
Benoit Boissinot
|
r1471 | # check the common case first | ||
Benoit Boissinot
|
r1749 | if type_ == 'n': | ||
Benoit Boissinot
|
r1471 | if not st: | ||
Matt Mackall
|
r4905 | st = os.lstat(self._join(fn)) | ||
Alexis S. L. Carvalho
|
r4677 | if (size >= 0 and (size != st.st_size | ||
or (mode ^ st.st_mode) & 0100) | ||||
or fn in self._copymap): | ||||
Benoit Boissinot
|
r1471 | modified.append(fn) | ||
Alexis S. L. Carvalho
|
r2962 | elif time != int(st.st_mtime): | ||
Benoit Boissinot
|
r1471 | lookup.append(fn) | ||
Vadim Gelfer
|
r2661 | elif list_clean: | ||
clean.append(fn) | ||||
Benoit Boissinot
|
r1749 | elif type_ == 'm': | ||
Benoit Boissinot
|
r1471 | modified.append(fn) | ||
Benoit Boissinot
|
r1749 | elif type_ == 'a': | ||
Benoit Boissinot
|
r1471 | added.append(fn) | ||
Benoit Boissinot
|
r1749 | elif type_ == 'r': | ||
Benoit Boissinot
|
r1471 | removed.append(fn) | ||
mason@suse.com
|
r1183 | |||
Vadim Gelfer
|
r2661 | return (lookup, modified, added, removed, deleted, unknown, ignored, | ||
clean) | ||||