dirstate.py
1450 lines
| 49.9 KiB
| text/x-python
|
PythonLexer
/ mercurial / dirstate.py
Martin Geisler
|
r8226 | # dirstate.py - working directory tracking for mercurial | ||
# | ||||
Raphaël Gomès
|
r47575 | # Copyright 2005-2007 Olivia Mackall <olivia@selenic.com> | ||
Martin Geisler
|
r8226 | # | ||
# This software may be used and distributed according to the terms of the | ||||
Matt Mackall
|
r10263 | # GNU General Public License version 2 or any later version. | ||
mpm@selenic.com
|
r1089 | |||
Gregory Szorc
|
r27503 | from __future__ import absolute_import | ||
Laurent Charignon
|
r27670 | import collections | ||
Augie Fackler
|
r32346 | import contextlib | ||
Gregory Szorc
|
r27503 | import errno | ||
import os | ||||
import stat | ||||
from .i18n import _ | ||||
Gregory Szorc
|
r43360 | from .pycompat import delattr | ||
Augie Fackler
|
r43533 | |||
from hgdemandimport import tracing | ||||
Gregory Szorc
|
r27503 | from . import ( | ||
r48295 | dirstatemap, | |||
Gregory Szorc
|
r27503 | encoding, | ||
error, | ||||
match as matchmod, | ||||
pathutil, | ||||
Yuya Nishihara
|
r32372 | policy, | ||
Pulkit Goyal
|
r30304 | pycompat, | ||
Gregory Szorc
|
r27503 | scmutil, | ||
Raphaël Gomès
|
r45017 | sparse, | ||
Gregory Szorc
|
r27503 | util, | ||
) | ||||
mpm@selenic.com
|
r1089 | |||
Simon Sapin
|
r49079 | from .dirstateutils import ( | ||
timestamp, | ||||
) | ||||
Augie Fackler
|
r43197 | from .interfaces import ( | ||
dirstate as intdirstate, | ||||
util as interfaceutil, | ||||
) | ||||
Augie Fackler
|
r43906 | parsers = policy.importmod('parsers') | ||
rustmod = policy.importrust('dirstate') | ||||
Yuya Nishihara
|
r32372 | |||
r49049 | HAS_FAST_DIRSTATE_V2 = rustmod is not None | |||
Matt Mackall
|
r8261 | propertycache = util.propertycache | ||
Idan Kamara
|
r16201 | filecache = scmutil.filecache | ||
r48310 | _rangemask = dirstatemap.rangemask | |||
Idan Kamara
|
r16201 | |||
Simon Sapin
|
r48858 | DirstateItem = dirstatemap.DirstateItem | ||
Siddharth Agarwal
|
r21808 | |||
r48276 | ||||
Idan Kamara
|
r16201 | class repocache(filecache): | ||
"""filecache for files in .hg/""" | ||||
Augie Fackler
|
r43346 | |||
Idan Kamara
|
r16201 | def join(self, obj, fname): | ||
return obj._opener.join(fname) | ||||
Matt Mackall
|
r4610 | |||
Augie Fackler
|
r43346 | |||
Idan Kamara
|
r16202 | class rootcache(filecache): | ||
"""filecache for files in the repository root""" | ||||
Augie Fackler
|
r43346 | |||
Idan Kamara
|
r16202 | def join(self, obj, fname): | ||
return obj._join(fname) | ||||
Augie Fackler
|
r43346 | |||
r48392 | def requires_parents_change(func): | |||
def wrap(self, *args, **kwargs): | ||||
if not self.pendingparentchange(): | ||||
msg = 'calling `%s` outside of a parentchange context' | ||||
msg %= func.__name__ | ||||
raise error.ProgrammingError(msg) | ||||
return func(self, *args, **kwargs) | ||||
return wrap | ||||
r48393 | def requires_no_parents_change(func): | |||
def wrap(self, *args, **kwargs): | ||||
r48394 | if self.pendingparentchange(): | |||
r48393 | msg = 'calling `%s` inside of a parentchange context' | |||
msg %= func.__name__ | ||||
raise error.ProgrammingError(msg) | ||||
return func(self, *args, **kwargs) | ||||
return wrap | ||||
Augie Fackler
|
r43197 | @interfaceutil.implementer(intdirstate.idirstate) | ||
Eric Hopper
|
r1559 | class dirstate(object): | ||
Joerg Sonnenberger
|
r47538 | def __init__( | ||
Simon Sapin
|
r48055 | self, | ||
opener, | ||||
ui, | ||||
root, | ||||
validate, | ||||
sparsematchfn, | ||||
nodeconstants, | ||||
use_dirstate_v2, | ||||
Joerg Sonnenberger
|
r47538 | ): | ||
Augie Fackler
|
r46554 | """Create a new dirstate object. | ||
Martin Geisler
|
r10145 | |||
opener is an open()-like callable that can be used to open the | ||||
dirstate file; root is the root of the directory tracked by | ||||
the dirstate. | ||||
Augie Fackler
|
r46554 | """ | ||
Simon Sapin
|
r48055 | self._use_dirstate_v2 = use_dirstate_v2 | ||
Joerg Sonnenberger
|
r47538 | self._nodeconstants = nodeconstants | ||
Matt Mackall
|
r4614 | self._opener = opener | ||
Matt Mackall
|
r13032 | self._validate = validate | ||
Matt Mackall
|
r4614 | self._root = root | ||
Gregory Szorc
|
r33373 | self._sparsematchfn = sparsematchfn | ||
Yuya Nishihara
|
r24198 | # ntpath.join(root, '') of Python 2.7.9 does not add sep if root is | ||
# UNC path pointing to root share (issue4557) | ||||
FUJIWARA Katsunori
|
r24833 | self._rootdir = pathutil.normasprefix(root) | ||
Matt Mackall
|
r4903 | self._dirty = False | ||
Matt Mackall
|
r4614 | self._ui = ui | ||
Idan Kamara
|
r16200 | self._filecache = {} | ||
Durham Goode
|
r22404 | self._parentwriters = 0 | ||
Augie Fackler
|
r43347 | self._filename = b'dirstate' | ||
self._pendingfilename = b'%s.pending' % self._filename | ||||
Mateusz Kwapich
|
r29772 | self._plchangecallbacks = {} | ||
self._origpl = None | ||||
r48295 | self._mapcls = dirstatemap.dirstatemap | |||
Martin von Zweigbergk
|
r41823 | # Access and cache cwd early, so we don't access it for the first time | ||
# after a working-copy update caused it to not exist (accessing it then | ||||
# raises an exception). | ||||
self._cwd | ||||
Durham Goode
|
r22404 | |||
r45359 | def prefetch_parents(self): | |||
"""make sure the parents are loaded | ||||
Used to avoid a race condition. | ||||
""" | ||||
self._pl | ||||
Augie Fackler
|
r32346 | @contextlib.contextmanager | ||
def parentchange(self): | ||||
Augie Fackler
|
r46554 | """Context manager for handling dirstate parents. | ||
Augie Fackler
|
r32346 | |||
If an exception occurs in the scope of the context manager, | ||||
the incoherent dirstate won't be written when wlock is | ||||
released. | ||||
Augie Fackler
|
r46554 | """ | ||
Augie Fackler
|
r32346 | self._parentwriters += 1 | ||
yield | ||||
# Typically we want the "undo" step of a context manager in a | ||||
# finally block so it happens even when an exception | ||||
# occurs. In this case, however, we only want to decrement | ||||
# parentwriters if the code in the with statement exits | ||||
# normally, so we don't have a try/finally here on purpose. | ||||
self._parentwriters -= 1 | ||||
Durham Goode
|
r22404 | def pendingparentchange(self): | ||
Augie Fackler
|
r46554 | """Returns true if the dirstate is in the middle of a set of changes | ||
Durham Goode
|
r22404 | that modify the dirstate parent. | ||
Augie Fackler
|
r46554 | """ | ||
Durham Goode
|
r22404 | return self._parentwriters > 0 | ||
mpm@selenic.com
|
r1089 | |||
Matt Mackall
|
r8261 | @propertycache | ||
def _map(self): | ||||
Mark Thomas
|
r35077 | """Return the dirstate contents (see documentation for dirstatemap).""" | ||
Joerg Sonnenberger
|
r47538 | self._map = self._mapcls( | ||
Simon Sapin
|
r48055 | self._ui, | ||
self._opener, | ||||
self._root, | ||||
self._nodeconstants, | ||||
self._use_dirstate_v2, | ||||
Joerg Sonnenberger
|
r47538 | ) | ||
Matt Mackall
|
r8261 | return self._map | ||
Gregory Szorc
|
r33373 | @property | ||
def _sparsematcher(self): | ||||
"""The matcher for the sparse checkout. | ||||
The working directory may not include every file from a manifest. The | ||||
matcher obtained by this property will match a path if it is to be | ||||
included in the working directory. | ||||
""" | ||||
# TODO there is potential to cache this property. For now, the matcher | ||||
# is resolved on every access. (But the called function does use a | ||||
# cache to keep the lookup fast.) | ||||
return self._sparsematchfn() | ||||
Augie Fackler
|
r43347 | @repocache(b'branch') | ||
Matt Mackall
|
r8261 | def _branch(self): | ||
try: | ||||
Augie Fackler
|
r43347 | return self._opener.read(b"branch").strip() or b"default" | ||
Gregory Szorc
|
r25660 | except IOError as inst: | ||
Pierre-Yves David
|
r15799 | if inst.errno != errno.ENOENT: | ||
raise | ||||
Augie Fackler
|
r43347 | return b"default" | ||
Matt Mackall
|
r8261 | |||
Durham Goode
|
r34340 | @property | ||
Matt Mackall
|
r8261 | def _pl(self): | ||
Durham Goode
|
r34339 | return self._map.parents() | ||
Matt Mackall
|
r8261 | |||
Mark Thomas
|
r35083 | def hasdir(self, d): | ||
return self._map.hastrackeddir(d) | ||||
FUJIWARA Katsunori
|
r16143 | |||
Augie Fackler
|
r43347 | @rootcache(b'.hgignore') | ||
Matt Mackall
|
r8261 | def _ignore(self): | ||
Laurent Charignon
|
r27594 | files = self._ignorefiles() | ||
Durham Goode
|
r25216 | if not files: | ||
Martin von Zweigbergk
|
r41825 | return matchmod.never() | ||
Durham Goode
|
r25216 | |||
Augie Fackler
|
r43347 | pats = [b'include:%s' % f for f in files] | ||
return matchmod.match(self._root, b'', [], pats, warn=self._ui.warn) | ||||
Matt Mackall
|
r8261 | |||
@propertycache | ||||
def _slash(self): | ||||
Augie Fackler
|
r43347 | return self._ui.configbool(b'ui', b'slash') and pycompat.ossep != b'/' | ||
Matt Mackall
|
r8261 | |||
@propertycache | ||||
def _checklink(self): | ||||
return util.checklink(self._root) | ||||
@propertycache | ||||
def _checkexec(self): | ||||
Mitchell Plamann
|
r45310 | return bool(util.checkexec(self._root)) | ||
Matt Mackall
|
r8261 | |||
@propertycache | ||||
def _checkcase(self): | ||||
Augie Fackler
|
r43347 | return not util.fscasesensitive(self._join(b'.hg')) | ||
Matt Mackall
|
r8261 | |||
Matt Mackall
|
r4905 | def _join(self, f): | ||
Benoit Boissinot
|
r6972 | # much faster than os.path.join() | ||
Benoit Boissinot
|
r6973 | # it's safe because f is always a relative path | ||
Benoit Boissinot
|
r6972 | return self._rootdir + f | ||
mpm@selenic.com
|
r1089 | |||
Matt Mackall
|
r15337 | def flagfunc(self, buildfallback): | ||
r49111 | """build a callable that returns flags associated with a filename | |||
Augie Fackler
|
r43346 | |||
r49111 | The information is extracted from three possible layers: | |||
1. the file system if it supports the information | ||||
2. the "fallback" information stored in the dirstate if any | ||||
3. a more expensive mechanism inferring the flags from the parents. | ||||
""" | ||||
# small hack to cache the result of buildfallback() | ||||
fallback_func = [] | ||||
def get_flags(x): | ||||
entry = None | ||||
fallback_value = None | ||||
Raphaël Gomès
|
r49103 | try: | ||
st = os.lstat(self._join(x)) | ||||
r49111 | except OSError: | |||
return b'' | ||||
if self._checklink: | ||||
Raphaël Gomès
|
r49103 | if util.statislink(st): | ||
Augie Fackler
|
r43347 | return b'l' | ||
r49111 | else: | |||
entry = self.get_entry(x) | ||||
if entry.has_fallback_symlink: | ||||
if entry.fallback_symlink: | ||||
return b'l' | ||||
else: | ||||
if not fallback_func: | ||||
fallback_func.append(buildfallback()) | ||||
fallback_value = fallback_func[0](x) | ||||
if b'l' in fallback_value: | ||||
return b'l' | ||||
if self._checkexec: | ||||
Raphaël Gomès
|
r49103 | if util.statisexec(st): | ||
Augie Fackler
|
r43347 | return b'x' | ||
r49111 | else: | |||
if entry is None: | ||||
entry = self.get_entry(x) | ||||
if entry.has_fallback_exec: | ||||
if entry.fallback_exec: | ||||
return b'x' | ||||
else: | ||||
if fallback_value is None: | ||||
if not fallback_func: | ||||
fallback_func.append(buildfallback()) | ||||
fallback_value = fallback_func[0](x) | ||||
if b'x' in fallback_value: | ||||
return b'x' | ||||
Raphaël Gomès
|
r49103 | return b'' | ||
Augie Fackler
|
r43346 | |||
r49111 | return get_flags | |||
Raphaël Gomès
|
r49102 | |||
Pierre-Yves David
|
r20335 | @propertycache | ||
def _cwd(self): | ||||
FUJIWARA Katsunori
|
r33212 | # internal config: ui.forcecwd | ||
Augie Fackler
|
r43347 | forcecwd = self._ui.config(b'ui', b'forcecwd') | ||
FUJIWARA Katsunori
|
r33212 | if forcecwd: | ||
return forcecwd | ||||
Matt Harbison
|
r39843 | return encoding.getcwd() | ||
Pierre-Yves David
|
r20335 | |||
mpm@selenic.com
|
r1089 | def getcwd(self): | ||
Augie Fackler
|
r46554 | """Return the path from which a canonical path is calculated. | ||
Yuya Nishihara
|
r26293 | |||
This path should be used to resolve file patterns or to convert | ||||
canonical paths back to file paths for display. It shouldn't be | ||||
used to get real file paths. Use vfs functions instead. | ||||
Augie Fackler
|
r46554 | """ | ||
Pierre-Yves David
|
r20335 | cwd = self._cwd | ||
Matt Mackall
|
r10282 | if cwd == self._root: | ||
Augie Fackler
|
r43347 | return b'' | ||
Matt Mackall
|
r4614 | # 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): | ||
Pulkit Goyal
|
r30614 | rootsep += pycompat.ossep | ||
Alexis S. L. Carvalho
|
r4230 | if cwd.startswith(rootsep): | ||
Augie Fackler
|
r43346 | return cwd[len(rootsep) :] | ||
Alexis S. L. Carvalho
|
r4230 | 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
|
r19210 | return util.pconvert(path) | ||
Alexis S. L. Carvalho
|
r4527 | return path | ||
Alexis S. L. Carvalho
|
r4525 | |||
mpm@selenic.com
|
r1089 | def __getitem__(self, key): | ||
Augie Fackler
|
r46554 | """Return the current state of key (a filename) in the dirstate. | ||
Martin Geisler
|
r10145 | |||
Greg Ward
|
r9518 | States are: | ||
n normal | ||||
m needs merging | ||||
r marked for removal | ||||
a marked for addition | ||||
? not tracked | ||||
r48301 | ||||
XXX The "state" is a bit obscure to be in the "public" API. we should | ||||
consider migrating all user of this to going through the dirstate entry | ||||
instead. | ||||
Augie Fackler
|
r46554 | """ | ||
r48919 | msg = b"don't use dirstate[file], use dirstate.get_entry(file)" | |||
util.nouideprecwarn(msg, b'6.1', stacklevel=2) | ||||
r48301 | entry = self._map.get(key) | |||
if entry is not None: | ||||
return entry.state | ||||
return b'?' | ||||
mpm@selenic.com
|
r1089 | |||
r48897 | def get_entry(self, path): | |||
"""return a DirstateItem for the associated path""" | ||||
entry = self._map.get(path) | ||||
if entry is None: | ||||
return DirstateItem() | ||||
return entry | ||||
mpm@selenic.com
|
r1089 | def __contains__(self, key): | ||
Matt Mackall
|
r4614 | return key in self._map | ||
def __iter__(self): | ||||
Alex Gaynor
|
r33673 | return iter(sorted(self._map)) | ||
mpm@selenic.com
|
r1089 | |||
Augie Fackler
|
r32550 | def items(self): | ||
Gregory Szorc
|
r43376 | return pycompat.iteritems(self._map) | ||
Bryan O'Sullivan
|
r18792 | |||
Augie Fackler
|
r32550 | iteritems = items | ||
mpm@selenic.com
|
r1089 | def parents(self): | ||
Matt Mackall
|
r13032 | return [self._validate(p) for p in self._pl] | ||
mpm@selenic.com
|
r1089 | |||
Matt Mackall
|
r13876 | def p1(self): | ||
return self._validate(self._pl[0]) | ||||
def p2(self): | ||||
return self._validate(self._pl[1]) | ||||
r48299 | @property | |||
def in_merge(self): | ||||
"""True if a merge is in progress""" | ||||
return self._pl[1] != self._nodeconstants.nullid | ||||
Matt Mackall
|
r4179 | def branch(self): | ||
Matt Mackall
|
r13047 | return encoding.tolocal(self._branch) | ||
Matt Mackall
|
r4179 | |||
Joerg Sonnenberger
|
r47771 | def setparents(self, p1, p2=None): | ||
Patrick Mezard
|
r16551 | """Set dirstate parents to p1 and p2. | ||
r48302 | When moving from two parents to one, "merged" entries a | |||
Patrick Mezard
|
r16551 | adjusted to normal and previous copy records discarded and | ||
returned by the call. | ||||
See localrepo.setparents() | ||||
""" | ||||
Joerg Sonnenberger
|
r47771 | if p2 is None: | ||
p2 = self._nodeconstants.nullid | ||||
Durham Goode
|
r22407 | if self._parentwriters == 0: | ||
Augie Fackler
|
r43346 | raise ValueError( | ||
Augie Fackler
|
r43347 | b"cannot set dirstate parent outside of " | ||
b"dirstate.parentchange context manager" | ||||
Augie Fackler
|
r43346 | ) | ||
Durham Goode
|
r22407 | |||
Durham Goode
|
r34340 | self._dirty = True | ||
Patrick Mezard
|
r16509 | oldp2 = self._pl[1] | ||
Mateusz Kwapich
|
r29772 | if self._origpl is None: | ||
self._origpl = self._pl | ||||
r48801 | nullid = self._nodeconstants.nullid | |||
r48873 | # True if we need to fold p2 related state back to a linear case | |||
fold_p2 = oldp2 != nullid and p2 == nullid | ||||
return self._map.setparents(p1, p2, fold_p2=fold_p2) | ||||
mpm@selenic.com
|
r1089 | |||
Matt Mackall
|
r4179 | def setbranch(self, branch): | ||
Yuya Nishihara
|
r40454 | self.__class__._branch.set(self, encoding.fromlocal(branch)) | ||
Augie Fackler
|
r43347 | f = self._opener(b'branch', b'w', atomictemp=True, checkambig=True) | ||
Idan Kamara
|
r16472 | try: | ||
Augie Fackler
|
r43347 | f.write(self._branch + b'\n') | ||
Idan Kamara
|
r16472 | f.close() | ||
Idan Kamara
|
r18317 | |||
# make sure filecache has the correct stat info for _branch after | ||||
# replacing the underlying file | ||||
Augie Fackler
|
r43347 | ce = self._filecache[b'_branch'] | ||
Idan Kamara
|
r18317 | if ce: | ||
ce.refresh() | ||||
Augie Fackler
|
r43346 | except: # re-raises | ||
Idan Kamara
|
r18076 | f.discard() | ||
raise | ||||
Matt Mackall
|
r4179 | |||
Matt Mackall
|
r4613 | def invalidate(self): | ||
Augie Fackler
|
r46554 | """Causes the next access to reread the dirstate. | ||
Siddharth Agarwal
|
r32682 | |||
This is different from localrepo.invalidatedirstate() because it always | ||||
rereads the dirstate. Use localrepo.invalidatedirstate() if you want to | ||||
Augie Fackler
|
r46554 | check whether the dirstate has changed before rereading it.""" | ||
Siddharth Agarwal
|
r32682 | |||
Augie Fackler
|
r43809 | for a in ("_map", "_branch", "_ignore"): | ||
Alexis S. L. Carvalho
|
r4953 | if a in self.__dict__: | ||
delattr(self, a) | ||||
Matt Mackall
|
r4903 | self._dirty = False | ||
Durham Goode
|
r22404 | self._parentwriters = 0 | ||
Mateusz Kwapich
|
r29772 | self._origpl = None | ||
Bryan O'Sullivan
|
r4375 | |||
mpm@selenic.com
|
r1089 | def copy(self, source, dest): | ||
Martin Geisler
|
r10145 | """Mark dest as a copy of source. Unmark dest if source is None.""" | ||
Patrick Mezard
|
r6680 | if source == dest: | ||
return | ||||
Matt Mackall
|
r4903 | self._dirty = True | ||
Patrick Mezard
|
r7566 | if source is not None: | ||
Durham Goode
|
r34337 | self._map.copymap[dest] = source | ||
r48871 | else: | |||
self._map.copymap.pop(dest, None) | ||||
mpm@selenic.com
|
r1089 | |||
def copied(self, file): | ||||
Durham Goode
|
r34337 | return self._map.copymap.get(file, None) | ||
Matt Mackall
|
r3154 | |||
def copies(self): | ||||
Durham Goode
|
r34337 | return self._map.copymap | ||
mpm@selenic.com
|
r1089 | |||
r48393 | @requires_no_parents_change | |||
r49207 | def set_tracked(self, filename, reset_copy=False): | |||
r48393 | """a "public" method for generic code to mark a file as tracked | |||
This function is to be called outside of "update/merge" case. For | ||||
example by a command like `hg add X`. | ||||
r49207 | if reset_copy is set, any existing copy information will be dropped. | |||
r48393 | return True the file was previously untracked, False otherwise. | |||
""" | ||||
r48798 | self._dirty = True | |||
r48393 | entry = self._map.get(filename) | |||
r48804 | if entry is None or not entry.tracked: | |||
r48798 | self._check_new_tracked_filename(filename) | |||
r49207 | pre_tracked = self._map.set_tracked(filename) | |||
if reset_copy: | ||||
self._map.copymap.pop(filename, None) | ||||
return pre_tracked | ||||
r48393 | ||||
r48399 | @requires_no_parents_change | |||
def set_untracked(self, filename): | ||||
"""a "public" method for generic code to mark a file as untracked | ||||
This function is to be called outside of "update/merge" case. For | ||||
example by a command like `hg remove X`. | ||||
return True the file was previously tracked, False otherwise. | ||||
""" | ||||
r48786 | ret = self._map.set_untracked(filename) | |||
if ret: | ||||
r48669 | self._dirty = True | |||
r48786 | return ret | |||
r48399 | ||||
r48504 | @requires_no_parents_change | |||
r49208 | def set_clean(self, filename, parentfiledata): | |||
r48504 | """record that the current state of the file on disk is known to be clean""" | |||
self._dirty = True | ||||
r48788 | if not self._map[filename].tracked: | |||
self._check_new_tracked_filename(filename) | ||||
r49208 | (mode, size, mtime) = parentfiledata | |||
r48788 | self._map.set_clean(filename, mode, size, mtime) | |||
r48504 | ||||
r48520 | @requires_no_parents_change | |||
def set_possibly_dirty(self, filename): | ||||
"""record that the current state of the file on disk is unknown""" | ||||
self._dirty = True | ||||
self._map.set_possibly_dirty(filename) | ||||
r48392 | @requires_parents_change | |||
r48493 | def update_file_p1( | |||
r48392 | self, | |||
filename, | ||||
p1_tracked, | ||||
): | ||||
"""Set a file as tracked in the parent (or not) | ||||
This is to be called when adjust the dirstate to a new parent after an history | ||||
rewriting operation. | ||||
It should not be called during a merge (p2 != nullid) and only within | ||||
a `with dirstate.parentchange():` context. | ||||
""" | ||||
if self.in_merge: | ||||
msg = b'update_file_reference should not be called when merging' | ||||
raise error.ProgrammingError(msg) | ||||
entry = self._map.get(filename) | ||||
if entry is None: | ||||
wc_tracked = False | ||||
else: | ||||
wc_tracked = entry.tracked | ||||
r48952 | if not (p1_tracked or wc_tracked): | |||
r48392 | # the file is no longer relevant to anyone | |||
r48813 | if self._map.get(filename) is not None: | |||
self._map.reset_state(filename) | ||||
r48802 | self._dirty = True | |||
r48392 | elif (not p1_tracked) and wc_tracked: | |||
r48494 | if entry is not None and entry.added: | |||
return # avoid dropping copy information (maybe?) | ||||
r48392 | ||||
r48494 | self._map.reset_state( | |||
filename, | ||||
wc_tracked, | ||||
p1_tracked, | ||||
r48952 | # the underlying reference might have changed, we will have to | |||
# check it. | ||||
has_meaningful_mtime=False, | ||||
r48494 | ) | |||
r48411 | @requires_parents_change | |||
def update_file( | ||||
self, | ||||
filename, | ||||
wc_tracked, | ||||
p1_tracked, | ||||
r48956 | p2_info=False, | |||
r48411 | possibly_dirty=False, | |||
r48491 | parentfiledata=None, | |||
r48411 | ): | |||
"""update the information about a file in the dirstate | ||||
This is to be called when the direstates parent changes to keep track | ||||
of what is the file situation in regards to the working copy and its parent. | ||||
This function must be called within a `dirstate.parentchange` context. | ||||
Augie Fackler
|
r48570 | note: the API is at an early stage and we might need to adjust it | ||
r48411 | depending of what information ends up being relevant and useful to | |||
other processing. | ||||
""" | ||||
r48492 | ||||
# note: I do not think we need to double check name clash here since we | ||||
# are in a update/merge case that should already have taken care of | ||||
# this. The test agrees | ||||
self._dirty = True | ||||
self._map.reset_state( | ||||
filename, | ||||
wc_tracked, | ||||
p1_tracked, | ||||
r48956 | p2_info=p2_info, | |||
r48952 | has_meaningful_mtime=not possibly_dirty, | |||
r48492 | parentfiledata=parentfiledata, | |||
) | ||||
r48411 | ||||
r48787 | def _check_new_tracked_filename(self, filename): | |||
scmutil.checkfilename(filename) | ||||
if self._map.hastrackeddir(filename): | ||||
msg = _(b'directory %r already in dirstate') | ||||
msg %= pycompat.bytestr(filename) | ||||
raise error.Abort(msg) | ||||
# shadows | ||||
for d in pathutil.finddirs(filename): | ||||
if self._map.hastrackeddir(d): | ||||
break | ||||
entry = self._map.get(d) | ||||
if entry is not None and not entry.removed: | ||||
msg = _(b'file %r in dirstate clashes with %r') | ||||
msg %= (pycompat.bytestr(d), pycompat.bytestr(filename)) | ||||
raise error.Abort(msg) | ||||
Siddharth Agarwal
|
r24538 | def _discoverpath(self, path, normed, ignoremissing, exists, storemap): | ||
if exists is None: | ||||
exists = os.path.lexists(os.path.join(self._root, path)) | ||||
if not exists: | ||||
# Maybe a path component exists | ||||
Augie Fackler
|
r43347 | if not ignoremissing and b'/' in path: | ||
d, f = path.rsplit(b'/', 1) | ||||
Siddharth Agarwal
|
r24538 | d = self._normalize(d, False, ignoremissing, None) | ||
Augie Fackler
|
r43347 | folded = d + b"/" + f | ||
Siddharth Agarwal
|
r24538 | else: | ||
# No path components, preserve original case | ||||
folded = path | ||||
else: | ||||
# recursively normalize leading directory components | ||||
# against dirstate | ||||
Augie Fackler
|
r43347 | if b'/' in normed: | ||
d, f = normed.rsplit(b'/', 1) | ||||
Siddharth Agarwal
|
r24538 | d = self._normalize(d, False, ignoremissing, True) | ||
Augie Fackler
|
r43347 | r = self._root + b"/" + d | ||
folded = d + b"/" + util.fspath(f, r) | ||||
Siddharth Agarwal
|
r24538 | else: | ||
folded = util.fspath(normed, self._root) | ||||
storemap[normed] = folded | ||||
return folded | ||||
Siddharth Agarwal
|
r24539 | def _normalizefile(self, path, isknown, ignoremissing=False, exists=None): | ||
Matt Mackall
|
r15488 | normed = util.normcase(path) | ||
Durham Goode
|
r34677 | folded = self._map.filefoldmap.get(normed, None) | ||
Matt Mackall
|
r13717 | if folded is None: | ||
Patrick Mezard
|
r16542 | if isknown: | ||
Matt Mackall
|
r13717 | folded = path | ||
Petr Kodl
|
r7068 | else: | ||
Augie Fackler
|
r43346 | folded = self._discoverpath( | ||
path, normed, ignoremissing, exists, self._map.filefoldmap | ||||
) | ||||
Siddharth Agarwal
|
r24539 | return folded | ||
Matt Mackall
|
r16302 | |||
Patrick Mezard
|
r16542 | def _normalize(self, path, isknown, ignoremissing=False, exists=None): | ||
Matt Mackall
|
r15488 | normed = util.normcase(path) | ||
Durham Goode
|
r34677 | folded = self._map.filefoldmap.get(normed, None) | ||
Siddharth Agarwal
|
r24561 | if folded is None: | ||
Durham Goode
|
r34679 | folded = self._map.dirfoldmap.get(normed, None) | ||
Matt Mackall
|
r13717 | if folded is None: | ||
Patrick Mezard
|
r16542 | if isknown: | ||
Matt Mackall
|
r13717 | folded = path | ||
Petr Kodl
|
r7068 | else: | ||
Siddharth Agarwal
|
r24540 | # store discovered result in dirfoldmap so that future | ||
# normalizefile calls don't start matching directories | ||||
Augie Fackler
|
r43346 | folded = self._discoverpath( | ||
path, normed, ignoremissing, exists, self._map.dirfoldmap | ||||
) | ||||
Matt Mackall
|
r13717 | return folded | ||
Patrick Mezard
|
r16542 | def normalize(self, path, isknown=False, ignoremissing=False): | ||
Augie Fackler
|
r46554 | """ | ||
Matt Mackall
|
r13717 | normalize the case of a pathname when on a casefolding filesystem | ||
isknown specifies whether the filename came from walking the | ||||
Patrick Mezard
|
r16542 | disk, to avoid extra filesystem access. | ||
If ignoremissing is True, missing path are returned | ||||
unchanged. Otherwise, we try harder to normalize possibly | ||||
existing path components. | ||||
Matt Mackall
|
r13717 | |||
The normalized case is determined based on the following precedence: | ||||
- version of name already stored in the dirstate | ||||
- version of name stored on disk | ||||
- version provided via command arguments | ||||
Augie Fackler
|
r46554 | """ | ||
Matt Mackall
|
r13717 | |||
if self._checkcase: | ||||
Patrick Mezard
|
r16542 | return self._normalize(path, isknown, ignoremissing) | ||
Matt Mackall
|
r13717 | return path | ||
Paul Moore
|
r6677 | |||
Alexis S. L. Carvalho
|
r5065 | def clear(self): | ||
Durham Goode
|
r34934 | self._map.clear() | ||
Alexis S. L. Carvalho
|
r5123 | self._dirty = True | ||
Alexis S. L. Carvalho
|
r5065 | |||
Durham Goode
|
r18760 | def rebuild(self, parent, allfiles, changedfiles=None): | ||
Pierre-Yves David
|
r25448 | if changedfiles is None: | ||
Christian Delahousse
|
r27176 | # Rebuild entire dirstate | ||
Kyle Lippincott
|
r44343 | to_lookup = allfiles | ||
to_drop = [] | ||||
Christian Delahousse
|
r27176 | self.clear() | ||
Kyle Lippincott
|
r44343 | elif len(changedfiles) < 10: | ||
# Avoid turning allfiles into a set, which can be expensive if it's | ||||
# large. | ||||
to_lookup = [] | ||||
to_drop = [] | ||||
for f in changedfiles: | ||||
if f in allfiles: | ||||
to_lookup.append(f) | ||||
else: | ||||
to_drop.append(f) | ||||
else: | ||||
changedfilesset = set(changedfiles) | ||||
to_lookup = changedfilesset & set(allfiles) | ||||
to_drop = changedfilesset - to_lookup | ||||
Christian Delahousse
|
r27176 | |||
Mateusz Kwapich
|
r29772 | if self._origpl is None: | ||
self._origpl = self._pl | ||||
Joerg Sonnenberger
|
r47771 | self._map.setparents(parent, self._nodeconstants.nullid) | ||
Kyle Lippincott
|
r44343 | |||
for f in to_lookup: | ||||
r48806 | ||||
if self.in_merge: | ||||
self.set_tracked(f) | ||||
else: | ||||
self._map.reset_state( | ||||
f, | ||||
wc_tracked=True, | ||||
p1_tracked=True, | ||||
) | ||||
Kyle Lippincott
|
r44343 | for f in to_drop: | ||
r48814 | self._map.reset_state(f) | |||
Mateusz Kwapich
|
r30026 | |||
Matt Mackall
|
r4903 | self._dirty = True | ||
mpm@selenic.com
|
r1089 | |||
FUJIWARA Katsunori
|
r32750 | def identity(self): | ||
Augie Fackler
|
r46554 | """Return identity of dirstate itself to detect changing in storage | ||
FUJIWARA Katsunori
|
r32750 | |||
If identity of previous dirstate is equal to this, writing | ||||
changes based on the former dirstate out can keep consistency. | ||||
Augie Fackler
|
r46554 | """ | ||
Durham Goode
|
r34676 | return self._map.identity | ||
FUJIWARA Katsunori
|
r32750 | |||
Pierre-Yves David
|
r29673 | def write(self, tr): | ||
Matt Mackall
|
r4612 | if not self._dirty: | ||
Benoit Boissinot
|
r1794 | return | ||
FUJIWARA Katsunori
|
r21931 | |||
FUJIWARA Katsunori
|
r26634 | filename = self._filename | ||
Pierre-Yves David
|
r29673 | if tr: | ||
FUJIWARA Katsunori
|
r26634 | # delay writing in-memory changes out | ||
Augie Fackler
|
r43346 | tr.addfilegenerator( | ||
Augie Fackler
|
r43347 | b'dirstate', | ||
Augie Fackler
|
r43346 | (self._filename,), | ||
r49222 | lambda f: self._writedirstate(tr, f), | |||
Augie Fackler
|
r43347 | location=b'plain', | ||
Augie Fackler
|
r43346 | ) | ||
FUJIWARA Katsunori
|
r26634 | return | ||
Augie Fackler
|
r43347 | st = self._opener(filename, b"w", atomictemp=True, checkambig=True) | ||
Simon Sapin
|
r48474 | self._writedirstate(tr, st) | ||
FUJIWARA Katsunori
|
r26521 | |||
Mateusz Kwapich
|
r29772 | def addparentchangecallback(self, category, callback): | ||
"""add a callback to be called when the wd parents are changed | ||||
Callback will be called with the following arguments: | ||||
dirstate, (oldp1, oldp2), (newp1, newp2) | ||||
Category is a unique identifier to allow overwriting an old callback | ||||
with a newer callback. | ||||
""" | ||||
self._plchangecallbacks[category] = callback | ||||
r49222 | def _writedirstate(self, tr, st): | |||
Mateusz Kwapich
|
r29772 | # notify callbacks about parents change | ||
if self._origpl is not None and self._origpl != self._pl: | ||||
Gregory Szorc
|
r43376 | for c, callback in sorted( | ||
pycompat.iteritems(self._plchangecallbacks) | ||||
): | ||||
Mateusz Kwapich
|
r29772 | callback(self, self._origpl, self._pl) | ||
self._origpl = None | ||||
r48869 | ||||
r49222 | self._map.write(tr, st) | |||
Durham Goode
|
r34674 | self._dirty = False | ||
mpm@selenic.com
|
r1089 | |||
Alexis S. L. Carvalho
|
r6032 | def _dirignore(self, f): | ||
if self._ignore(f): | ||||
return True | ||||
Martin von Zweigbergk
|
r44032 | for p in pathutil.finddirs(f): | ||
Matt Mackall
|
r6767 | if self._ignore(p): | ||
Alexis S. L. Carvalho
|
r6032 | return True | ||
return False | ||||
Laurent Charignon
|
r27594 | def _ignorefiles(self): | ||
files = [] | ||||
Augie Fackler
|
r43347 | if os.path.exists(self._join(b'.hgignore')): | ||
files.append(self._join(b'.hgignore')) | ||||
for name, path in self._ui.configitems(b"ui"): | ||||
if name == b'ignore' or name.startswith(b'ignore.'): | ||||
Laurent Charignon
|
r27594 | # we need to use os.path.join here rather than self._join | ||
# because path is arbitrary and user-specified | ||||
files.append(os.path.join(self._rootdir, util.expandpath(path))) | ||||
return files | ||||
Laurent Charignon
|
r27670 | def _ignorefileandline(self, f): | ||
files = collections.deque(self._ignorefiles()) | ||||
visited = set() | ||||
while files: | ||||
i = files.popleft() | ||||
Augie Fackler
|
r43346 | patterns = matchmod.readpatternfile( | ||
i, self._ui.warn, sourceinfo=True | ||||
) | ||||
Laurent Charignon
|
r27670 | for pattern, lineno, line in patterns: | ||
Augie Fackler
|
r43347 | kind, p = matchmod._patsplit(pattern, b'glob') | ||
if kind == b"subinclude": | ||||
Laurent Charignon
|
r27670 | if p not in visited: | ||
files.append(p) | ||||
continue | ||||
Augie Fackler
|
r43346 | m = matchmod.match( | ||
Augie Fackler
|
r43347 | self._root, b'', [], [pattern], warn=self._ui.warn | ||
Augie Fackler
|
r43346 | ) | ||
Laurent Charignon
|
r27670 | if m(f): | ||
return (i, lineno, line) | ||||
visited.add(i) | ||||
Augie Fackler
|
r43347 | return (None, -1, b"") | ||
Laurent Charignon
|
r27670 | |||
Siddharth Agarwal
|
r19173 | def _walkexplicit(self, match, subrepos): | ||
Augie Fackler
|
r46554 | """Get stat data about the files explicitly specified by match. | ||
Matt Mackall
|
r3529 | |||
Siddharth Agarwal
|
r19174 | Return a triple (results, dirsfound, dirsnotfound). | ||
Siddharth Agarwal
|
r19173 | - results is a mapping from filename to stat result. It also contains | ||
listings mapping subrepos and .hg to None. | ||||
Siddharth Agarwal
|
r19174 | - dirsfound is a list of files found to be directories. | ||
Siddharth Agarwal
|
r19173 | - dirsnotfound is a list of files that the dirstate thinks are | ||
Augie Fackler
|
r46554 | directories and that were not found.""" | ||
Matt Mackall
|
r6578 | |||
Matt Mackall
|
r8681 | def badtype(mode): | ||
Augie Fackler
|
r43347 | kind = _(b'unknown') | ||
Matt Mackall
|
r10282 | if stat.S_ISCHR(mode): | ||
Augie Fackler
|
r43347 | kind = _(b'character device') | ||
Matt Mackall
|
r10282 | elif stat.S_ISBLK(mode): | ||
Augie Fackler
|
r43347 | kind = _(b'block device') | ||
Matt Mackall
|
r10282 | elif stat.S_ISFIFO(mode): | ||
Augie Fackler
|
r43347 | kind = _(b'fifo') | ||
Matt Mackall
|
r10282 | elif stat.S_ISSOCK(mode): | ||
Augie Fackler
|
r43347 | kind = _(b'socket') | ||
Matt Mackall
|
r10282 | elif stat.S_ISDIR(mode): | ||
Augie Fackler
|
r43347 | kind = _(b'directory') | ||
return _(b'unsupported file type (type is %s)') % kind | ||||
Matt Mackall
|
r6830 | |||
Matt Mackall
|
r8676 | badfn = match.bad | ||
Matt Mackall
|
r6831 | dmap = self._map | ||
Matt Mackall
|
r5000 | lstat = os.lstat | ||
Matt Mackall
|
r6830 | getkind = stat.S_IFMT | ||
Matt Mackall
|
r6828 | dirkind = stat.S_IFDIR | ||
Matt Mackall
|
r6830 | regkind = stat.S_IFREG | ||
lnkkind = stat.S_IFLNK | ||||
Matt Mackall
|
r6831 | join = self._join | ||
Siddharth Agarwal
|
r19174 | dirsfound = [] | ||
foundadd = dirsfound.append | ||||
Siddharth Agarwal
|
r19173 | dirsnotfound = [] | ||
Siddharth Agarwal
|
r19175 | notfoundadd = dirsnotfound.append | ||
Matt Mackall
|
r6820 | |||
Martin von Zweigbergk
|
r24448 | if not match.isexact() and self._checkcase: | ||
Matt Mackall
|
r12907 | normalize = self._normalize | ||
else: | ||||
Siddharth Agarwal
|
r18032 | normalize = None | ||
Matt Mackall
|
r12907 | |||
Martin Geisler
|
r12211 | files = sorted(match.files()) | ||
subrepos.sort() | ||||
i, j = 0, 0 | ||||
while i < len(files) and j < len(subrepos): | ||||
Augie Fackler
|
r43347 | subpath = subrepos[j] + b"/" | ||
Oleg Stepanov
|
r13233 | if files[i] < subpath: | ||
Martin Geisler
|
r12211 | i += 1 | ||
continue | ||||
trbs
|
r13339 | while i < len(files) and files[i].startswith(subpath): | ||
Martin Geisler
|
r12211 | del files[i] | ||
j += 1 | ||||
Augie Fackler
|
r43347 | if not files or b'' in files: | ||
files = [b''] | ||||
Martin von Zweigbergk
|
r42527 | # constructing the foldmap is expensive, so don't do it for the | ||
Martin von Zweigbergk
|
r42528 | # common case where files is [''] | ||
Martin von Zweigbergk
|
r42527 | normalize = None | ||
Augie Fackler
|
r10176 | results = dict.fromkeys(subrepos) | ||
Augie Fackler
|
r43347 | results[b'.hg'] = None | ||
Matt Mackall
|
r5000 | |||
Martin Geisler
|
r12211 | for ff in files: | ||
Martin von Zweigbergk
|
r42527 | if normalize: | ||
Siddharth Agarwal
|
r24522 | nf = normalize(ff, False, True) | ||
Siddharth Agarwal
|
r18032 | else: | ||
Siddharth Agarwal
|
r24522 | nf = ff | ||
Matt Mackall
|
r6829 | if nf in results: | ||
Matt Mackall
|
r6820 | continue | ||
try: | ||||
Matt Mackall
|
r6831 | st = lstat(join(nf)) | ||
Matt Mackall
|
r6830 | kind = getkind(st.st_mode) | ||
if kind == dirkind: | ||||
Simon Heimberg
|
r8588 | if nf in dmap: | ||
Mads Kiilerich
|
r21115 | # file replaced by dir on disk but still in dirstate | ||
Simon Heimberg
|
r8588 | results[nf] = None | ||
Matt Harbison
|
r24537 | foundadd((nf, ff)) | ||
Matt Mackall
|
r12401 | elif kind == regkind or kind == lnkkind: | ||
Matt Mackall
|
r6830 | results[nf] = st | ||
Matt Mackall
|
r6828 | else: | ||
Matt Mackall
|
r8681 | badfn(ff, badtype(kind)) | ||
Matt Mackall
|
r6830 | if nf in dmap: | ||
Matt Mackall
|
r6829 | results[nf] = None | ||
Augie Fackler
|
r43346 | except OSError as inst: # nf not found on disk - it is dirstate only | ||
if nf in dmap: # does it exactly match a missing file? | ||||
Matt Mackall
|
r8675 | results[nf] = None | ||
Augie Fackler
|
r43346 | else: # does it match a missing directory? | ||
Mark Thomas
|
r35083 | if self._map.hasdir(nf): | ||
Martin von Zweigbergk
|
r23375 | notfoundadd(nf) | ||
Matt Mackall
|
r8677 | else: | ||
Augie Fackler
|
r34024 | badfn(ff, encoding.strtolocal(inst.strerror)) | ||
Matt Mackall
|
r6820 | |||
Yuya Nishihara
|
r36218 | # match.files() may contain explicitly-specified paths that shouldn't | ||
# be taken; drop them from the list of files found. dirsfound/notfound | ||||
# aren't filtered here because they will be tested later. | ||||
if match.anypats(): | ||||
for f in list(results): | ||||
Augie Fackler
|
r43347 | if f == b'.hg' or f in subrepos: | ||
Yuya Nishihara
|
r36218 | # keep sentinel to disable further out-of-repo walks | ||
continue | ||||
if not match(f): | ||||
del results[f] | ||||
Matt Harbison
|
r25877 | # Case insensitive filesystems cannot rely on lstat() failing to detect | ||
# a case-only rename. Prune the stat object for any file that does not | ||||
# match the case in the filesystem, if there are multiple files that | ||||
# normalize to the same path. | ||||
if match.isexact() and self._checkcase: | ||||
normed = {} | ||||
Gregory Szorc
|
r43376 | for f, st in pycompat.iteritems(results): | ||
Matt Harbison
|
r25877 | if st is None: | ||
continue | ||||
nc = util.normcase(f) | ||||
paths = normed.get(nc) | ||||
if paths is None: | ||||
paths = set() | ||||
normed[nc] = paths | ||||
paths.add(f) | ||||
Gregory Szorc
|
r43376 | for norm, paths in pycompat.iteritems(normed): | ||
Matt Harbison
|
r25877 | if len(paths) > 1: | ||
for path in paths: | ||||
Augie Fackler
|
r43346 | folded = self._discoverpath( | ||
path, norm, True, None, self._map.dirfoldmap | ||||
) | ||||
Matt Harbison
|
r25877 | if path != folded: | ||
results[path] = None | ||||
Siddharth Agarwal
|
r19174 | return results, dirsfound, dirsnotfound | ||
Siddharth Agarwal
|
r19173 | |||
Siddharth Agarwal
|
r19190 | def walk(self, match, subrepos, unknown, ignored, full=True): | ||
Augie Fackler
|
r46554 | """ | ||
Siddharth Agarwal
|
r19173 | Walk recursively through the directory tree, finding all files | ||
matched by match. | ||||
Siddharth Agarwal
|
r19190 | If full is False, maybe skip some known-clean files. | ||
Siddharth Agarwal
|
r19173 | Return a dict mapping filename to stat-like object (either | ||
mercurial.osutil.stat instance or return value of os.stat()). | ||||
Siddharth Agarwal
|
r19190 | |||
Augie Fackler
|
r46554 | """ | ||
Siddharth Agarwal
|
r19190 | # full is a flag that extensions that hook into walk can use -- this | ||
# implementation doesn't use it at all. This satisfies the contract | ||||
# because we only guarantee a "maybe". | ||||
Siddharth Agarwal
|
r19173 | |||
if ignored: | ||||
ignore = util.never | ||||
dirignore = util.never | ||||
Mads Kiilerich
|
r21115 | elif unknown: | ||
ignore = self._ignore | ||||
dirignore = self._dirignore | ||||
else: | ||||
# if not unknown and not ignored, drop dir recursion and step 2 | ||||
Siddharth Agarwal
|
r19173 | ignore = util.always | ||
dirignore = util.always | ||||
matchfn = match.matchfn | ||||
matchalways = match.always() | ||||
matchtdir = match.traversedir | ||||
dmap = self._map | ||||
Yuya Nishihara
|
r32203 | listdir = util.listdir | ||
Siddharth Agarwal
|
r19173 | lstat = os.lstat | ||
dirkind = stat.S_IFDIR | ||||
regkind = stat.S_IFREG | ||||
lnkkind = stat.S_IFLNK | ||||
join = self._join | ||||
exact = skipstep3 = False | ||||
Augie Fackler
|
r43346 | if match.isexact(): # match.exact | ||
Siddharth Agarwal
|
r19173 | exact = True | ||
Augie Fackler
|
r43346 | dirignore = util.always # skip step 2 | ||
elif match.prefix(): # match.match, no patterns | ||||
Siddharth Agarwal
|
r19173 | skipstep3 = True | ||
if not exact and self._checkcase: | ||||
normalize = self._normalize | ||||
Siddharth Agarwal
|
r24541 | normalizefile = self._normalizefile | ||
Siddharth Agarwal
|
r19173 | skipstep3 = False | ||
else: | ||||
Siddharth Agarwal
|
r24560 | normalize = self._normalize | ||
Siddharth Agarwal
|
r24541 | normalizefile = None | ||
Siddharth Agarwal
|
r19173 | |||
# step 1: find all explicit files | ||||
results, work, dirsnotfound = self._walkexplicit(match, subrepos) | ||||
Martin von Zweigbergk
|
r44112 | if matchtdir: | ||
for d in work: | ||||
matchtdir(d[0]) | ||||
for d in dirsnotfound: | ||||
matchtdir(d) | ||||
Siddharth Agarwal
|
r19173 | |||
Siddharth Agarwal
|
r19172 | skipstep3 = skipstep3 and not (work or dirsnotfound) | ||
Matt Harbison
|
r24537 | work = [d for d in work if not dirignore(d[0])] | ||
Siddharth Agarwal
|
r19171 | |||
Matt Mackall
|
r6826 | # step 2: visit subdirectories | ||
Siddharth Agarwal
|
r24560 | def traverse(work, alreadynormed): | ||
Siddharth Agarwal
|
r24559 | wadd = work.append | ||
while work: | ||||
Augie Fackler
|
r43533 | tracing.counter('dirstate.walk work', len(work)) | ||
Siddharth Agarwal
|
r24560 | nd = work.pop() | ||
Kyle Lippincott
|
r38991 | visitentries = match.visitchildrenset(nd) | ||
if not visitentries: | ||||
Martin von Zweigbergk
|
r32177 | continue | ||
Augie Fackler
|
r43347 | if visitentries == b'this' or visitentries == b'all': | ||
Kyle Lippincott
|
r38991 | visitentries = None | ||
Siddharth Agarwal
|
r24559 | skip = None | ||
Augie Fackler
|
r43347 | if nd != b'': | ||
skip = b'.hg' | ||||
Siddharth Agarwal
|
r24559 | try: | ||
Augie Fackler
|
r43533 | with tracing.log('dirstate.walk.traverse listdir %s', nd): | ||
entries = listdir(join(nd), stat=True, skip=skip) | ||||
Gregory Szorc
|
r25660 | except OSError as inst: | ||
Siddharth Agarwal
|
r24559 | if inst.errno in (errno.EACCES, errno.ENOENT): | ||
Augie Fackler
|
r43346 | match.bad( | ||
self.pathto(nd), encoding.strtolocal(inst.strerror) | ||||
) | ||||
Siddharth Agarwal
|
r24559 | continue | ||
raise | ||||
for f, kind, st in entries: | ||||
Kyle Lippincott
|
r39296 | # Some matchers may return files in the visitentries set, | ||
# instead of 'this', if the matcher explicitly mentions them | ||||
# and is not an exactmatcher. This is acceptable; we do not | ||||
# make any hard assumptions about file-or-directory below | ||||
# based on the presence of `f` in visitentries. If | ||||
# visitchildrenset returned a set, we can always skip the | ||||
# entries *not* in the set it provided regardless of whether | ||||
# they're actually a file or a directory. | ||||
Kyle Lippincott
|
r38991 | if visitentries and f not in visitentries: | ||
continue | ||||
Siddharth Agarwal
|
r24559 | if normalizefile: | ||
# even though f might be a directory, we're only | ||||
# interested in comparing it to files currently in the | ||||
# dmap -- therefore normalizefile is enough | ||||
Augie Fackler
|
r43346 | nf = normalizefile( | ||
Augie Fackler
|
r43347 | nd and (nd + b"/" + f) or f, True, True | ||
Augie Fackler
|
r43346 | ) | ||
Siddharth Agarwal
|
r24559 | else: | ||
Augie Fackler
|
r43347 | nf = nd and (nd + b"/" + f) or f | ||
Siddharth Agarwal
|
r24559 | if nf not in results: | ||
if kind == dirkind: | ||||
if not ignore(nf): | ||||
if matchtdir: | ||||
matchtdir(nf) | ||||
Siddharth Agarwal
|
r24560 | wadd(nf) | ||
Siddharth Agarwal
|
r24559 | if nf in dmap and (matchalways or matchfn(nf)): | ||
results[nf] = None | ||||
elif kind == regkind or kind == lnkkind: | ||||
if nf in dmap: | ||||
if matchalways or matchfn(nf): | ||||
results[nf] = st | ||||
Augie Fackler
|
r43346 | elif (matchalways or matchfn(nf)) and not ignore( | ||
nf | ||||
): | ||||
Siddharth Agarwal
|
r24560 | # unknown file -- normalize if necessary | ||
if not alreadynormed: | ||||
nf = normalize(nf, False, True) | ||||
Siddharth Agarwal
|
r24559 | results[nf] = st | ||
elif nf in dmap and (matchalways or matchfn(nf)): | ||||
Matt Mackall
|
r6829 | results[nf] = None | ||
Siddharth Agarwal
|
r24559 | |||
Siddharth Agarwal
|
r24560 | for nd, d in work: | ||
# alreadynormed means that processwork doesn't have to do any | ||||
# expensive directory normalization | ||||
alreadynormed = not normalize or nd == d | ||||
traverse([d], alreadynormed) | ||||
mpm@selenic.com
|
r1089 | |||
Siddharth Agarwal
|
r18812 | for s in subrepos: | ||
del results[s] | ||||
Augie Fackler
|
r43347 | del results[b'.hg'] | ||
Siddharth Agarwal
|
r18812 | |||
Mads Kiilerich
|
r21115 | # step 3: visit remaining files from dmap | ||
Matt Mackall
|
r8684 | if not skipstep3 and not exact: | ||
Mads Kiilerich
|
r21115 | # If a dmap file is not in results yet, it was either | ||
# a) not matching matchfn b) ignored, c) missing, or d) under a | ||||
# symlink directory. | ||||
Siddharth Agarwal
|
r18815 | if not results and matchalways: | ||
Pulkit Goyal
|
r31422 | visit = [f for f in dmap] | ||
Siddharth Agarwal
|
r18815 | else: | ||
visit = [f for f in dmap if f not in results and matchfn(f)] | ||||
visit.sort() | ||||
Durham Goode
|
r18625 | if unknown: | ||
Mads Kiilerich
|
r21115 | # unknown == True means we walked all dirs under the roots | ||
# that wasn't ignored, and everything that matched was stat'ed | ||||
# and is already in results. | ||||
# The rest must thus be ignored or under a symlink. | ||||
Yuya Nishihara
|
r33722 | audit_path = pathutil.pathauditor(self._root, cached=True) | ||
Durham Goode
|
r18625 | |||
for nf in iter(visit): | ||||
Martin von Zweigbergk
|
r24621 | # If a stat for the same file was already added with a | ||
# different case, don't add one for this, since that would | ||||
# make it appear as if the file exists under both names | ||||
# on disk. | ||||
Augie Fackler
|
r43346 | if ( | ||
normalizefile | ||||
and normalizefile(nf, True, True) in results | ||||
): | ||||
Martin von Zweigbergk
|
r24621 | results[nf] = None | ||
Durham Goode
|
r18625 | # Report ignored items in the dmap as long as they are not | ||
# under a symlink directory. | ||||
Martin von Zweigbergk
|
r24621 | elif audit_path.check(nf): | ||
Durham Goode
|
r18663 | try: | ||
results[nf] = lstat(join(nf)) | ||||
Mads Kiilerich
|
r21115 | # file was just ignored, no links, and exists | ||
Durham Goode
|
r18663 | except OSError: | ||
# file doesn't exist | ||||
results[nf] = None | ||||
Durham Goode
|
r18625 | else: | ||
# It's either missing or under a symlink directory | ||||
Mads Kiilerich
|
r21115 | # which we in this case report as missing | ||
Durham Goode
|
r18625 | results[nf] = None | ||
else: | ||||
# We may not have walked the full directory tree above, | ||||
Mads Kiilerich
|
r21115 | # so stat and check everything we missed. | ||
Augie Fackler
|
r31507 | iv = iter(visit) | ||
Bryan O'Sullivan
|
r26984 | for st in util.statfiles([join(i) for i in visit]): | ||
Augie Fackler
|
r31507 | results[next(iv)] = st | ||
Matt Mackall
|
r6829 | return results | ||
mpm@selenic.com
|
r1089 | |||
Raphaël Gomès
|
r45017 | def _rust_status(self, matcher, list_clean, list_ignored, list_unknown): | ||
Raphaël Gomès
|
r44537 | # Force Rayon (Rust parallelism library) to respect the number of | ||
# workers. This is a temporary workaround until Rust code knows | ||||
# how to read the config file. | ||||
numcpus = self._ui.configint(b"worker", b"numcpus") | ||||
if numcpus is not None: | ||||
encoding.environ.setdefault(b'RAYON_NUM_THREADS', b'%d' % numcpus) | ||||
workers_enabled = self._ui.configbool(b"worker", b"enabled", True) | ||||
if not workers_enabled: | ||||
encoding.environ[b"RAYON_NUM_THREADS"] = b"1" | ||||
( | ||||
lookup, | ||||
modified, | ||||
added, | ||||
removed, | ||||
deleted, | ||||
Raphaël Gomès
|
r45017 | clean, | ||
ignored, | ||||
Raphaël Gomès
|
r44537 | unknown, | ||
Raphaël Gomès
|
r45017 | warnings, | ||
bad, | ||||
Raphaël Gomès
|
r45355 | traversed, | ||
Simon Sapin
|
r48139 | dirty, | ||
Raphaël Gomès
|
r44537 | ) = rustmod.status( | ||
r48933 | self._map._map, | |||
Raphaël Gomès
|
r44537 | matcher, | ||
self._rootdir, | ||||
Raphaël Gomès
|
r45017 | self._ignorefiles(), | ||
self._checkexec, | ||||
bool(list_clean), | ||||
bool(list_ignored), | ||||
bool(list_unknown), | ||||
Raphaël Gomès
|
r45355 | bool(matcher.traversedir), | ||
Raphaël Gomès
|
r44537 | ) | ||
Raphaël Gomès
|
r45355 | |||
Simon Sapin
|
r48139 | self._dirty |= dirty | ||
Raphaël Gomès
|
r45355 | if matcher.traversedir: | ||
for dir in traversed: | ||||
matcher.traversedir(dir) | ||||
Raphaël Gomès
|
r45017 | if self._ui.warn: | ||
for item in warnings: | ||||
if isinstance(item, tuple): | ||||
file_path, syntax = item | ||||
msg = _(b"%s: ignoring invalid syntax '%s'\n") % ( | ||||
file_path, | ||||
syntax, | ||||
) | ||||
self._ui.warn(msg) | ||||
else: | ||||
msg = _(b"skipping unreadable pattern file '%s': %s\n") | ||||
self._ui.warn( | ||||
msg | ||||
% ( | ||||
pathutil.canonpath( | ||||
self._rootdir, self._rootdir, item | ||||
), | ||||
b"No such file or directory", | ||||
) | ||||
) | ||||
for (fn, message) in bad: | ||||
matcher.bad(fn, encoding.strtolocal(message)) | ||||
Raphaël Gomès
|
r44537 | |||
status = scmutil.status( | ||||
modified=modified, | ||||
added=added, | ||||
removed=removed, | ||||
deleted=deleted, | ||||
unknown=unknown, | ||||
Raphaël Gomès
|
r45017 | ignored=ignored, | ||
Raphaël Gomès
|
r44537 | clean=clean, | ||
) | ||||
return (lookup, status) | ||||
Augie Fackler
|
r10176 | def status(self, match, subrepos, ignored, clean, unknown): | ||
Augie Fackler
|
r46554 | """Determine the status of the working copy relative to the | ||
Martin von Zweigbergk
|
r22915 | dirstate and return a pair of (unsure, status), where status is of type | ||
scmutil.status and: | ||||
Greg Ward
|
r9518 | |||
unsure: | ||||
files that might have been modified since the dirstate was | ||||
written, but need to be read to be sure (size is the same | ||||
but mtime differs) | ||||
Martin von Zweigbergk
|
r22915 | status.modified: | ||
Greg Ward
|
r9518 | files that have definitely been modified since the dirstate | ||
was written (different size or mode) | ||||
Martin von Zweigbergk
|
r22915 | status.clean: | ||
Greg Ward
|
r9518 | files that have definitely not been modified since the | ||
dirstate was written | ||||
Augie Fackler
|
r46554 | """ | ||
Matt Mackall
|
r6753 | listignored, listclean, listunknown = ignored, clean, unknown | ||
Thomas Arendsen Hein
|
r2022 | lookup, modified, added, unknown, ignored = [], [], [], [], [] | ||
Vadim Gelfer
|
r2661 | removed, deleted, clean = [], [], [] | ||
mpm@selenic.com
|
r1089 | |||
Matt Mackall
|
r5003 | dmap = self._map | ||
Durham Goode
|
r34936 | dmap.preload() | ||
Raphaël Gomès
|
r43568 | |||
use_rust = True | ||||
Raphaël Gomès
|
r44369 | |||
Raphaël Gomès
|
r45017 | allowed_matchers = ( | ||
matchmod.alwaysmatcher, | ||||
matchmod.exactmatcher, | ||||
matchmod.includematcher, | ||||
) | ||||
Raphaël Gomès
|
r44369 | |||
Raphaël Gomès
|
r43568 | if rustmod is None: | ||
use_rust = False | ||||
Raphaël Gomès
|
r45017 | elif self._checkcase: | ||
# Case-insensitive filesystems are not handled yet | ||||
use_rust = False | ||||
Raphaël Gomès
|
r43568 | elif subrepos: | ||
use_rust = False | ||||
Raphaël Gomès
|
r45017 | elif sparse.enabled: | ||
Raphaël Gomès
|
r43568 | use_rust = False | ||
Raphaël Gomès
|
r44369 | elif not isinstance(match, allowed_matchers): | ||
Raphaël Gomès
|
r45351 | # Some matchers have yet to be implemented | ||
Raphaël Gomès
|
r43568 | use_rust = False | ||
r49213 | # Get the time from the filesystem so we can disambiguate files that | |||
# appear modified in the present or future. | ||||
try: | ||||
mtime_boundary = timestamp.get_fs_now(self._opener) | ||||
except OSError: | ||||
# In largefiles or readonly context | ||||
mtime_boundary = None | ||||
Raphaël Gomès
|
r43568 | if use_rust: | ||
Raphaël Gomès
|
r45017 | try: | ||
r49213 | res = self._rust_status( | |||
Raphaël Gomès
|
r45017 | match, listclean, listignored, listunknown | ||
) | ||||
r49213 | return res + (mtime_boundary,) | |||
Raphaël Gomès
|
r45017 | except rustmod.FallbackError: | ||
pass | ||||
Raphaël Gomès
|
r43568 | |||
Martin von Zweigbergk
|
r44019 | def noop(f): | ||
pass | ||||
Durham Goode
|
r34936 | dcontains = dmap.__contains__ | ||
dget = dmap.__getitem__ | ||||
Augie Fackler
|
r43346 | ladd = lookup.append # aka "unsure" | ||
Matt Mackall
|
r5003 | madd = modified.append | ||
aadd = added.append | ||||
Martin von Zweigbergk
|
r44019 | uadd = unknown.append if listunknown else noop | ||
iadd = ignored.append if listignored else noop | ||||
Matt Mackall
|
r5003 | radd = removed.append | ||
dadd = deleted.append | ||||
Martin von Zweigbergk
|
r44019 | cadd = clean.append if listclean else noop | ||
Siddharth Agarwal
|
r18034 | mexact = match.exact | ||
dirignore = self._dirignore | ||||
checkexec = self._checkexec | ||||
r49112 | checklink = self._checklink | |||
Durham Goode
|
r34337 | copymap = self._map.copymap | ||
Matt Mackall
|
r5003 | |||
Siddharth Agarwal
|
r19191 | # We need to do full walks when either | ||
# - we're listing all clean files, or | ||||
# - match.traversedir does something, because match.traversedir should | ||||
# be called for every dir in the working dir | ||||
full = listclean or match.traversedir is not None | ||||
Gregory Szorc
|
r43376 | for fn, st in pycompat.iteritems( | ||
self.walk(match, subrepos, listunknown, listignored, full=full) | ||||
): | ||||
Durham Goode
|
r34936 | if not dcontains(fn): | ||
Siddharth Agarwal
|
r18034 | if (listignored or mexact(fn)) and dirignore(fn): | ||
Matt Mackall
|
r6753 | if listignored: | ||
Alexis S. L. Carvalho
|
r6033 | iadd(fn) | ||
Siddharth Agarwal
|
r19910 | else: | ||
Matt Mackall
|
r5003 | uadd(fn) | ||
Benoit Boissinot
|
r1471 | continue | ||
Matt Mackall
|
r6591 | |||
Durham Goode
|
r34936 | t = dget(fn) | ||
r48325 | mode = t.mode | |||
r48326 | size = t.size | |||
Matt Mackall
|
r6591 | |||
r48320 | if not st and t.tracked: | |||
Matt Mackall
|
r6818 | dadd(fn) | ||
r48960 | elif t.p2_info: | |||
r48322 | madd(fn) | |||
elif t.added: | ||||
aadd(fn) | ||||
elif t.removed: | ||||
radd(fn) | ||||
r48323 | elif t.tracked: | |||
r49112 | if not checklink and t.has_fallback_symlink: | |||
# If the file system does not support symlink, the mode | ||||
# might not be correctly stored in the dirstate, so do not | ||||
# trust it. | ||||
ladd(fn) | ||||
elif not checkexec and t.has_fallback_exec: | ||||
# If the file system does not support exec bits, the mode | ||||
# might not be correctly stored in the dirstate, so do not | ||||
# trust it. | ||||
ladd(fn) | ||||
elif ( | ||||
Augie Fackler
|
r43346 | size >= 0 | ||
and ( | ||||
(size != st.st_size and size != st.st_size & _rangemask) | ||||
or ((mode ^ st.st_mode) & 0o100 and checkexec) | ||||
) | ||||
or fn in copymap | ||||
): | ||||
Raphaël Gomès
|
r47530 | if stat.S_ISLNK(st.st_mode) and size != st.st_size: | ||
Corey Schuhen
|
r47436 | # issue6456: Size returned may be longer due to | ||
# encryption on EXT-4 fscrypt, undecided. | ||||
ladd(fn) | ||||
else: | ||||
madd(fn) | ||||
Simon Sapin
|
r49079 | elif not t.mtime_likely_equal_to(timestamp.mtime_of(st)): | ||
r49220 | # There might be a change in the future if for example the | |||
# internal clock is off, but this is a case where the issues | ||||
# the user would face would be a lot worse and there is | ||||
# nothing we can really do. | ||||
Greg Ward
|
r13704 | ladd(fn) | ||
Matt Mackall
|
r6753 | elif listclean: | ||
Matt Mackall
|
r5003 | cadd(fn) | ||
Raphaël Gomès
|
r45017 | status = scmutil.status( | ||
modified, added, removed, deleted, unknown, ignored, clean | ||||
Augie Fackler
|
r43346 | ) | ||
r49213 | return (lookup, status, mtime_boundary) | |||
Siddharth Agarwal
|
r21984 | |||
def matches(self, match): | ||||
Augie Fackler
|
r46554 | """ | ||
Siddharth Agarwal
|
r21984 | return files in the dirstate (in whatever state) filtered by match | ||
Augie Fackler
|
r46554 | """ | ||
Siddharth Agarwal
|
r21984 | dmap = self._map | ||
Raphaël Gomès
|
r44833 | if rustmod is not None: | ||
r48933 | dmap = self._map._map | |||
Raphaël Gomès
|
r44833 | |||
Siddharth Agarwal
|
r21984 | if match.always(): | ||
return dmap.keys() | ||||
files = match.files() | ||||
Martin von Zweigbergk
|
r24448 | if match.isexact(): | ||
Siddharth Agarwal
|
r21984 | # fast path -- filter the other way around, since typically files is | ||
# much smaller than dmap | ||||
return [f for f in files if f in dmap] | ||||
Martin von Zweigbergk
|
r25275 | if match.prefix() and all(fn in dmap for fn in files): | ||
Siddharth Agarwal
|
r21984 | # fast path -- all the values are known to be files, so just return | ||
# that | ||||
return list(files) | ||||
return [f for f in dmap if match(f)] | ||||
FUJIWARA Katsunori
|
r26632 | |||
FUJIWARA Katsunori
|
r26746 | def _actualfilename(self, tr): | ||
if tr: | ||||
FUJIWARA Katsunori
|
r26633 | return self._pendingfilename | ||
else: | ||||
return self._filename | ||||
Adam Simpkins
|
r33440 | def savebackup(self, tr, backupname): | ||
'''Save current dirstate into backup file''' | ||||
FUJIWARA Katsunori
|
r26746 | filename = self._actualfilename(tr) | ||
Adam Simpkins
|
r33440 | assert backupname != filename | ||
FUJIWARA Katsunori
|
r26633 | |||
# use '_writedirstate' instead of 'write' to write changes certainly, | ||||
# because the latter omits writing out if transaction is running. | ||||
# output file will be used to create backup of dirstate at this point. | ||||
Jun Wu
|
r31208 | if self._dirty or not self._opener.exists(filename): | ||
Augie Fackler
|
r43346 | self._writedirstate( | ||
Simon Sapin
|
r48474 | tr, | ||
self._opener(filename, b"w", atomictemp=True, checkambig=True), | ||||
Augie Fackler
|
r43346 | ) | ||
FUJIWARA Katsunori
|
r26633 | |||
if tr: | ||||
# ensure that subsequent tr.writepending returns True for | ||||
# changes written out above, even if dirstate is never | ||||
# changed after this | ||||
Augie Fackler
|
r43346 | tr.addfilegenerator( | ||
Augie Fackler
|
r43347 | b'dirstate', | ||
Augie Fackler
|
r43346 | (self._filename,), | ||
Simon Sapin
|
r48474 | lambda f: self._writedirstate(tr, f), | ||
Augie Fackler
|
r43347 | location=b'plain', | ||
Augie Fackler
|
r43346 | ) | ||
FUJIWARA Katsunori
|
r26633 | |||
# ensure that pending file written above is unlinked at | ||||
# failure, even if tr.writepending isn't invoked until the | ||||
# end of this transaction | ||||
Augie Fackler
|
r43347 | tr.registertmp(filename, location=b'plain') | ||
FUJIWARA Katsunori
|
r26633 | |||
Ryan McElroy
|
r31547 | self._opener.tryunlink(backupname) | ||
Jun Wu
|
r31207 | # hardlink backup is okay because _writedirstate is always called | ||
# with an "atomictemp=True" file. | ||||
Augie Fackler
|
r43346 | util.copyfile( | ||
self._opener.join(filename), | ||||
self._opener.join(backupname), | ||||
hardlink=True, | ||||
) | ||||
FUJIWARA Katsunori
|
r26632 | |||
Adam Simpkins
|
r33440 | def restorebackup(self, tr, backupname): | ||
'''Restore dirstate by backup file''' | ||||
FUJIWARA Katsunori
|
r26632 | # this "invalidate()" prevents "wlock.release()" from writing | ||
# changes of dirstate out after restoring from backup file | ||||
self.invalidate() | ||||
FUJIWARA Katsunori
|
r26746 | filename = self._actualfilename(tr) | ||
Mark Thomas
|
r34941 | o = self._opener | ||
if util.samefile(o.join(backupname), o.join(filename)): | ||||
o.unlink(backupname) | ||||
else: | ||||
o.rename(backupname, filename, checkambig=True) | ||||
FUJIWARA Katsunori
|
r26632 | |||
Adam Simpkins
|
r33440 | def clearbackup(self, tr, backupname): | ||
'''Clear backup file''' | ||||
self._opener.unlink(backupname) | ||||
r48901 | ||||
def verify(self, m1, m2): | ||||
"""check the dirstate content again the parent manifest and yield errors""" | ||||
missing_from_p1 = b"%s in state %s, but not in manifest1\n" | ||||
unexpected_in_p1 = b"%s in state %s, but also in manifest1\n" | ||||
missing_from_ps = b"%s in state %s, but not in either manifest\n" | ||||
missing_from_ds = b"%s in manifest1, but listed as state %s\n" | ||||
for f, entry in self.items(): | ||||
state = entry.state | ||||
if state in b"nr" and f not in m1: | ||||
yield (missing_from_p1, f, state) | ||||
if state in b"a" and f in m1: | ||||
yield (unexpected_in_p1, f, state) | ||||
if state in b"m" and f not in m1 and f not in m2: | ||||
yield (missing_from_ps, f, state) | ||||
for f in m1: | ||||
state = self.get_entry(f).state | ||||
if state not in b"nrm": | ||||
yield (missing_from_ds, f, state) | ||||