sparse.py
823 lines
| 24.6 KiB
| text/x-python
|
PythonLexer
/ mercurial / sparse.py
Gregory Szorc
|
r33297 | # sparse.py - functionality for sparse checkouts | ||
# | ||||
# Copyright 2014 Facebook, Inc. | ||||
# | ||||
# This software may be used and distributed according to the terms of the | ||||
# GNU General Public License version 2 or any later version. | ||||
from __future__ import absolute_import | ||||
Gregory Szorc
|
r33317 | import hashlib | ||
Gregory Szorc
|
r33320 | import os | ||
Gregory Szorc
|
r33317 | |||
Gregory Szorc
|
r33297 | from .i18n import _ | ||
Pulkit Goyal
|
r35600 | from .node import ( | ||
hex, | ||||
nullid, | ||||
) | ||||
Gregory Szorc
|
r33297 | from . import ( | ||
error, | ||||
Gregory Szorc
|
r33320 | match as matchmod, | ||
Gregory Szorc
|
r33321 | merge as mergemod, | ||
Kostia Balytskyi
|
r33648 | pathutil, | ||
Gregory Szorc
|
r33320 | pycompat, | ||
Gregory Szorc
|
r33556 | scmutil, | ||
Gregory Szorc
|
r33371 | util, | ||
Gregory Szorc
|
r33297 | ) | ||
Gregory Szorc
|
r33299 | # Whether sparse features are enabled. This variable is intended to be | ||
# temporary to facilitate porting sparse to core. It should eventually be | ||||
# a per-repo option, possibly a repo requirement. | ||||
enabled = False | ||||
Augie Fackler
|
r43346 | |||
Pulkit Goyal
|
r38874 | def parseconfig(ui, raw, action): | ||
Gregory Szorc
|
r33297 | """Parse sparse config file content. | ||
Pulkit Goyal
|
r38874 | action is the command which is trigerring this read, can be narrow, sparse | ||
Gregory Szorc
|
r33297 | Returns a tuple of includes, excludes, and profiles. | ||
""" | ||||
includes = set() | ||||
excludes = set() | ||||
Gregory Szorc
|
r33550 | profiles = set() | ||
Gregory Szorc
|
r33551 | current = None | ||
havesection = False | ||||
Augie Fackler
|
r43347 | for line in raw.split(b'\n'): | ||
Gregory Szorc
|
r33297 | line = line.strip() | ||
Augie Fackler
|
r43347 | if not line or line.startswith(b'#'): | ||
Gregory Szorc
|
r33297 | # empty or comment line, skip | ||
continue | ||||
Augie Fackler
|
r43347 | elif line.startswith(b'%include '): | ||
Gregory Szorc
|
r33297 | line = line[9:].strip() | ||
if line: | ||||
Gregory Szorc
|
r33550 | profiles.add(line) | ||
Augie Fackler
|
r43347 | elif line == b'[include]': | ||
Gregory Szorc
|
r33551 | if havesection and current != includes: | ||
Gregory Szorc
|
r33297 | # TODO pass filename into this API so we can report it. | ||
Augie Fackler
|
r43346 | raise error.Abort( | ||
_( | ||||
Augie Fackler
|
r43347 | b'%(action)s config cannot have includes ' | ||
b'after excludes' | ||||
Augie Fackler
|
r43346 | ) | ||
Augie Fackler
|
r43347 | % {b'action': action} | ||
Augie Fackler
|
r43346 | ) | ||
Gregory Szorc
|
r33551 | havesection = True | ||
current = includes | ||||
Gregory Szorc
|
r33297 | continue | ||
Augie Fackler
|
r43347 | elif line == b'[exclude]': | ||
Gregory Szorc
|
r33551 | havesection = True | ||
Gregory Szorc
|
r33297 | current = excludes | ||
elif line: | ||||
Gregory Szorc
|
r33551 | if current is None: | ||
Augie Fackler
|
r43346 | raise error.Abort( | ||
Augie Fackler
|
r43347 | _( | ||
b'%(action)s config entry outside of ' | ||||
b'section: %(line)s' | ||||
) | ||||
% {b'action': action, b'line': line}, | ||||
Augie Fackler
|
r43346 | hint=_( | ||
Augie Fackler
|
r43347 | b'add an [include] or [exclude] line ' | ||
b'to declare the entry type' | ||||
Augie Fackler
|
r43346 | ), | ||
) | ||||
Gregory Szorc
|
r33551 | |||
Augie Fackler
|
r43347 | if line.strip().startswith(b'/'): | ||
Augie Fackler
|
r43346 | ui.warn( | ||
_( | ||||
Augie Fackler
|
r43347 | b'warning: %(action)s profile cannot use' | ||
b' paths starting with /, ignoring %(line)s\n' | ||||
Augie Fackler
|
r43346 | ) | ||
Augie Fackler
|
r43347 | % {b'action': action, b'line': line} | ||
Augie Fackler
|
r43346 | ) | ||
Gregory Szorc
|
r33297 | continue | ||
current.add(line) | ||||
return includes, excludes, profiles | ||||
Gregory Szorc
|
r33298 | |||
Augie Fackler
|
r43346 | |||
Gregory Szorc
|
r33298 | # Exists as separate function to facilitate monkeypatching. | ||
def readprofile(repo, profile, changeid): | ||||
"""Resolve the raw content of a sparse profile file.""" | ||||
# TODO add some kind of cache here because this incurs a manifest | ||||
# resolve and can be slow. | ||||
return repo.filectx(profile, changeid=changeid).data() | ||||
Gregory Szorc
|
r33300 | |||
Augie Fackler
|
r43346 | |||
Gregory Szorc
|
r33300 | def patternsforrev(repo, rev): | ||
"""Obtain sparse checkout patterns for the given rev. | ||||
Returns a tuple of iterables representing includes, excludes, and | ||||
patterns. | ||||
""" | ||||
# Feature isn't enabled. No-op. | ||||
if not enabled: | ||||
Gregory Szorc
|
r33550 | return set(), set(), set() | ||
Gregory Szorc
|
r33300 | |||
Augie Fackler
|
r43347 | raw = repo.vfs.tryread(b'sparse') | ||
Gregory Szorc
|
r33300 | if not raw: | ||
Gregory Szorc
|
r33550 | return set(), set(), set() | ||
Gregory Szorc
|
r33300 | |||
if rev is None: | ||||
Augie Fackler
|
r43346 | raise error.Abort( | ||
Martin von Zweigbergk
|
r43387 | _(b'cannot parse sparse patterns from working directory') | ||
Augie Fackler
|
r43346 | ) | ||
Gregory Szorc
|
r33300 | |||
Augie Fackler
|
r43347 | includes, excludes, profiles = parseconfig(repo.ui, raw, b'sparse') | ||
Gregory Szorc
|
r33300 | ctx = repo[rev] | ||
if profiles: | ||||
visited = set() | ||||
while profiles: | ||||
profile = profiles.pop() | ||||
if profile in visited: | ||||
continue | ||||
visited.add(profile) | ||||
try: | ||||
raw = readprofile(repo, profile, rev) | ||||
except error.ManifestLookupError: | ||||
msg = ( | ||||
Augie Fackler
|
r43347 | b"warning: sparse profile '%s' not found " | ||
b"in rev %s - ignoring it\n" % (profile, ctx) | ||||
Augie Fackler
|
r43346 | ) | ||
Gregory Szorc
|
r33300 | # experimental config: sparse.missingwarning | ||
Augie Fackler
|
r43347 | if repo.ui.configbool(b'sparse', b'missingwarning'): | ||
Gregory Szorc
|
r33300 | repo.ui.warn(msg) | ||
else: | ||||
repo.ui.debug(msg) | ||||
continue | ||||
Augie Fackler
|
r43347 | pincludes, pexcludes, subprofs = parseconfig( | ||
repo.ui, raw, b'sparse' | ||||
) | ||||
Gregory Szorc
|
r33300 | includes.update(pincludes) | ||
excludes.update(pexcludes) | ||||
Gregory Szorc
|
r33550 | profiles.update(subprofs) | ||
Gregory Szorc
|
r33300 | |||
profiles = visited | ||||
if includes: | ||||
Augie Fackler
|
r43347 | includes.add(b'.hg*') | ||
Gregory Szorc
|
r33300 | |||
return includes, excludes, profiles | ||||
Gregory Szorc
|
r33301 | |||
Augie Fackler
|
r43346 | |||
Gregory Szorc
|
r33370 | def activeconfig(repo): | ||
"""Determine the active sparse config rules. | ||||
Rules are constructed by reading the current sparse config and bringing in | ||||
referenced profiles from parents of the working directory. | ||||
""" | ||||
Augie Fackler
|
r43346 | revs = [ | ||
repo.changelog.rev(node) | ||||
for node in repo.dirstate.parents() | ||||
if node != nullid | ||||
] | ||||
Gregory Szorc
|
r33301 | |||
Gregory Szorc
|
r33370 | allincludes = set() | ||
allexcludes = set() | ||||
allprofiles = set() | ||||
Gregory Szorc
|
r33301 | for rev in revs: | ||
Gregory Szorc
|
r33370 | includes, excludes, profiles = patternsforrev(repo, rev) | ||
allincludes |= includes | ||||
allexcludes |= excludes | ||||
Gregory Szorc
|
r33550 | allprofiles |= profiles | ||
Gregory Szorc
|
r33301 | |||
Gregory Szorc
|
r33370 | return allincludes, allexcludes, allprofiles | ||
Gregory Szorc
|
r33302 | |||
Augie Fackler
|
r43346 | |||
Gregory Szorc
|
r33317 | def configsignature(repo, includetemp=True): | ||
"""Obtain the signature string for the current sparse configuration. | ||||
This is used to construct a cache key for matchers. | ||||
""" | ||||
cache = repo._sparsesignaturecache | ||||
Augie Fackler
|
r43347 | signature = cache.get(b'signature') | ||
Gregory Szorc
|
r33317 | |||
if includetemp: | ||||
Augie Fackler
|
r43347 | tempsignature = cache.get(b'tempsignature') | ||
Gregory Szorc
|
r33317 | else: | ||
Augie Fackler
|
r43347 | tempsignature = b'0' | ||
Gregory Szorc
|
r33317 | |||
if signature is None or (includetemp and tempsignature is None): | ||||
Augie Fackler
|
r43347 | signature = hex(hashlib.sha1(repo.vfs.tryread(b'sparse')).digest()) | ||
cache[b'signature'] = signature | ||||
Gregory Szorc
|
r33317 | |||
if includetemp: | ||||
Augie Fackler
|
r43347 | raw = repo.vfs.tryread(b'tempsparse') | ||
Pulkit Goyal
|
r35600 | tempsignature = hex(hashlib.sha1(raw).digest()) | ||
Augie Fackler
|
r43347 | cache[b'tempsignature'] = tempsignature | ||
Gregory Szorc
|
r33317 | |||
Augie Fackler
|
r43347 | return b'%s %s' % (signature, tempsignature) | ||
Gregory Szorc
|
r33317 | |||
Augie Fackler
|
r43346 | |||
Gregory Szorc
|
r33303 | def writeconfig(repo, includes, excludes, profiles): | ||
"""Write the sparse config file given a sparse configuration.""" | ||||
Augie Fackler
|
r43347 | with repo.vfs(b'sparse', b'wb') as fh: | ||
Gregory Szorc
|
r33303 | for p in sorted(profiles): | ||
Augie Fackler
|
r43347 | fh.write(b'%%include %s\n' % p) | ||
Gregory Szorc
|
r33303 | |||
if includes: | ||||
Augie Fackler
|
r43347 | fh.write(b'[include]\n') | ||
Gregory Szorc
|
r33303 | for i in sorted(includes): | ||
fh.write(i) | ||||
Augie Fackler
|
r43347 | fh.write(b'\n') | ||
Gregory Szorc
|
r33303 | |||
if excludes: | ||||
Augie Fackler
|
r43347 | fh.write(b'[exclude]\n') | ||
Gregory Szorc
|
r33303 | for e in sorted(excludes): | ||
fh.write(e) | ||||
Augie Fackler
|
r43347 | fh.write(b'\n') | ||
Gregory Szorc
|
r33303 | |||
Gregory Szorc
|
r33325 | repo._sparsesignaturecache.clear() | ||
Gregory Szorc
|
r33304 | |||
Augie Fackler
|
r43346 | |||
Gregory Szorc
|
r33304 | def readtemporaryincludes(repo): | ||
Augie Fackler
|
r43347 | raw = repo.vfs.tryread(b'tempsparse') | ||
Gregory Szorc
|
r33304 | if not raw: | ||
return set() | ||||
Augie Fackler
|
r43347 | return set(raw.split(b'\n')) | ||
Gregory Szorc
|
r33304 | |||
Augie Fackler
|
r43346 | |||
Gregory Szorc
|
r33304 | def writetemporaryincludes(repo, includes): | ||
Augie Fackler
|
r43347 | repo.vfs.write(b'tempsparse', b'\n'.join(sorted(includes))) | ||
Gregory Szorc
|
r33325 | repo._sparsesignaturecache.clear() | ||
Gregory Szorc
|
r33304 | |||
Augie Fackler
|
r43346 | |||
Gregory Szorc
|
r33304 | def addtemporaryincludes(repo, additional): | ||
includes = readtemporaryincludes(repo) | ||||
for i in additional: | ||||
includes.add(i) | ||||
writetemporaryincludes(repo, includes) | ||||
Gregory Szorc
|
r33320 | |||
Augie Fackler
|
r43346 | |||
Gregory Szorc
|
r33321 | def prunetemporaryincludes(repo): | ||
Augie Fackler
|
r43347 | if not enabled or not repo.vfs.exists(b'tempsparse'): | ||
Gregory Szorc
|
r33321 | return | ||
Martin von Zweigbergk
|
r33356 | s = repo.status() | ||
if s.modified or s.added or s.removed or s.deleted: | ||||
Gregory Szorc
|
r33321 | # Still have pending changes. Don't bother trying to prune. | ||
return | ||||
sparsematch = matcher(repo, includetemp=False) | ||||
dirstate = repo.dirstate | ||||
actions = [] | ||||
dropped = [] | ||||
tempincludes = readtemporaryincludes(repo) | ||||
for file in tempincludes: | ||||
if file in dirstate and not sparsematch(file): | ||||
Augie Fackler
|
r43347 | message = _(b'dropping temporarily included sparse files') | ||
Gregory Szorc
|
r33321 | actions.append((file, None, message)) | ||
dropped.append(file) | ||||
Martin von Zweigbergk
|
r41068 | typeactions = mergemod.emptyactions() | ||
Augie Fackler
|
r43347 | typeactions[b'r'] = actions | ||
Augie Fackler
|
r43346 | mergemod.applyupdates( | ||
Augie Fackler
|
r43347 | repo, typeactions, repo[None], repo[b'.'], False, wantfiledata=False | ||
Augie Fackler
|
r43346 | ) | ||
Gregory Szorc
|
r33321 | |||
# Fix dirstate | ||||
for file in dropped: | ||||
dirstate.drop(file) | ||||
Augie Fackler
|
r43347 | repo.vfs.unlink(b'tempsparse') | ||
Gregory Szorc
|
r33325 | repo._sparsesignaturecache.clear() | ||
Augie Fackler
|
r43346 | msg = _( | ||
Augie Fackler
|
r43347 | b'cleaned up %d temporarily added file(s) from the ' | ||
b'sparse checkout\n' | ||||
Augie Fackler
|
r43346 | ) | ||
Gregory Szorc
|
r33321 | repo.ui.status(msg % len(tempincludes)) | ||
Augie Fackler
|
r43346 | |||
Martin von Zweigbergk
|
r33447 | def forceincludematcher(matcher, includes): | ||
"""Returns a matcher that returns true for any of the forced includes | ||||
before testing against the actual matcher.""" | ||||
Augie Fackler
|
r43347 | kindpats = [(b'path', include, b'') for include in includes] | ||
includematcher = matchmod.includematcher(b'', kindpats) | ||||
Martin von Zweigbergk
|
r33447 | return matchmod.unionmatcher([includematcher, matcher]) | ||
Augie Fackler
|
r43346 | |||
Gregory Szorc
|
r33320 | def matcher(repo, revs=None, includetemp=True): | ||
"""Obtain a matcher for sparse working directories for the given revs. | ||||
If multiple revisions are specified, the matcher is the union of all | ||||
revs. | ||||
``includetemp`` indicates whether to use the temporary sparse profile. | ||||
""" | ||||
# If sparse isn't enabled, sparse matcher matches everything. | ||||
if not enabled: | ||||
Martin von Zweigbergk
|
r41825 | return matchmod.always() | ||
Gregory Szorc
|
r33320 | |||
if not revs or revs == [None]: | ||||
Augie Fackler
|
r43346 | revs = [ | ||
repo.changelog.rev(node) | ||||
for node in repo.dirstate.parents() | ||||
if node != nullid | ||||
] | ||||
Gregory Szorc
|
r33320 | |||
signature = configsignature(repo, includetemp=includetemp) | ||||
Augie Fackler
|
r43347 | key = b'%s %s' % (signature, b' '.join(map(pycompat.bytestr, revs))) | ||
Gregory Szorc
|
r33320 | |||
result = repo._sparsematchercache.get(key) | ||||
if result: | ||||
return result | ||||
matchers = [] | ||||
for rev in revs: | ||||
try: | ||||
includes, excludes, profiles = patternsforrev(repo, rev) | ||||
if includes or excludes: | ||||
Augie Fackler
|
r43346 | matcher = matchmod.match( | ||
repo.root, | ||||
Augie Fackler
|
r43347 | b'', | ||
Augie Fackler
|
r43346 | [], | ||
include=includes, | ||||
exclude=excludes, | ||||
Augie Fackler
|
r43347 | default=b'relpath', | ||
Augie Fackler
|
r43346 | ) | ||
Gregory Szorc
|
r33320 | matchers.append(matcher) | ||
except IOError: | ||||
pass | ||||
if not matchers: | ||||
Martin von Zweigbergk
|
r41825 | result = matchmod.always() | ||
Gregory Szorc
|
r33320 | elif len(matchers) == 1: | ||
result = matchers[0] | ||||
else: | ||||
result = matchmod.unionmatcher(matchers) | ||||
if includetemp: | ||||
tempincludes = readtemporaryincludes(repo) | ||||
Martin von Zweigbergk
|
r33447 | result = forceincludematcher(result, tempincludes) | ||
Gregory Szorc
|
r33320 | |||
repo._sparsematchercache[key] = result | ||||
return result | ||||
Gregory Szorc
|
r33322 | |||
Augie Fackler
|
r43346 | |||
Gregory Szorc
|
r33323 | def filterupdatesactions(repo, wctx, mctx, branchmerge, actions): | ||
"""Filter updates to only lay out files that match the sparse rules.""" | ||||
if not enabled: | ||||
return actions | ||||
Gregory Szorc
|
r33322 | |||
oldrevs = [pctx.rev() for pctx in wctx.parents()] | ||||
oldsparsematch = matcher(repo, oldrevs) | ||||
if oldsparsematch.always(): | ||||
Gregory Szorc
|
r33323 | return actions | ||
Gregory Szorc
|
r33322 | |||
files = set() | ||||
prunedactions = {} | ||||
if branchmerge: | ||||
# If we're merging, use the wctx filter, since we're merging into | ||||
# the wctx. | ||||
Martin von Zweigbergk
|
r41442 | sparsematch = matcher(repo, [wctx.p1().rev()]) | ||
Gregory Szorc
|
r33322 | else: | ||
# If we're updating, use the target context's filter, since we're | ||||
# moving to the target context. | ||||
sparsematch = matcher(repo, [mctx.rev()]) | ||||
temporaryfiles = [] | ||||
Gregory Szorc
|
r43376 | for file, action in pycompat.iteritems(actions): | ||
Gregory Szorc
|
r33322 | type, args, msg = action | ||
files.add(file) | ||||
if sparsematch(file): | ||||
prunedactions[file] = action | ||||
Augie Fackler
|
r43347 | elif type == b'm': | ||
Gregory Szorc
|
r33322 | temporaryfiles.append(file) | ||
prunedactions[file] = action | ||||
elif branchmerge: | ||||
Augie Fackler
|
r43347 | if type != b'k': | ||
Gregory Szorc
|
r33322 | temporaryfiles.append(file) | ||
prunedactions[file] = action | ||||
Augie Fackler
|
r43347 | elif type == b'f': | ||
Gregory Szorc
|
r33322 | prunedactions[file] = action | ||
elif file in wctx: | ||||
Augie Fackler
|
r43347 | prunedactions[file] = (b'r', args, msg) | ||
Gregory Szorc
|
r33322 | |||
Pulkit Goyal
|
r39563 | if branchmerge and type == mergemod.ACTION_MERGE: | ||
f1, f2, fa, move, anc = args | ||||
if not sparsematch(f1): | ||||
temporaryfiles.append(f1) | ||||
Gregory Szorc
|
r33322 | if len(temporaryfiles) > 0: | ||
Augie Fackler
|
r43346 | repo.ui.status( | ||
_( | ||||
Augie Fackler
|
r43347 | b'temporarily included %d file(s) in the sparse ' | ||
b'checkout for merging\n' | ||||
Augie Fackler
|
r43346 | ) | ||
% len(temporaryfiles) | ||||
) | ||||
Gregory Szorc
|
r33322 | addtemporaryincludes(repo, temporaryfiles) | ||
# Add the new files to the working copy so they can be merged, etc | ||||
actions = [] | ||||
Augie Fackler
|
r43347 | message = b'temporarily adding to sparse checkout' | ||
Gregory Szorc
|
r33322 | wctxmanifest = repo[None].manifest() | ||
for file in temporaryfiles: | ||||
if file in wctxmanifest: | ||||
fctx = repo[None][file] | ||||
actions.append((file, (fctx.flags(), False), message)) | ||||
Martin von Zweigbergk
|
r41068 | typeactions = mergemod.emptyactions() | ||
Augie Fackler
|
r43347 | typeactions[b'g'] = actions | ||
Augie Fackler
|
r43346 | mergemod.applyupdates( | ||
Augie Fackler
|
r43347 | repo, typeactions, repo[None], repo[b'.'], False, wantfiledata=False | ||
Augie Fackler
|
r43346 | ) | ||
Gregory Szorc
|
r33322 | |||
dirstate = repo.dirstate | ||||
for file, flags, msg in actions: | ||||
dirstate.normal(file) | ||||
Gregory Szorc
|
r33370 | profiles = activeconfig(repo)[2] | ||
Gregory Szorc
|
r33322 | changedprofiles = profiles & files | ||
# If an active profile changed during the update, refresh the checkout. | ||||
# Don't do this during a branch merge, since all incoming changes should | ||||
# have been handled by the temporary includes above. | ||||
if changedprofiles and not branchmerge: | ||||
mf = mctx.manifest() | ||||
for file in mf: | ||||
old = oldsparsematch(file) | ||||
new = sparsematch(file) | ||||
if not old and new: | ||||
flags = mf.flags(file) | ||||
Augie Fackler
|
r43347 | prunedactions[file] = (b'g', (flags, False), b'') | ||
Gregory Szorc
|
r33322 | elif old and not new: | ||
Augie Fackler
|
r43347 | prunedactions[file] = (b'r', [], b'') | ||
Gregory Szorc
|
r33322 | |||
Gregory Szorc
|
r33323 | return prunedactions | ||
Gregory Szorc
|
r33324 | |||
Augie Fackler
|
r43346 | |||
Gregory Szorc
|
r33324 | def refreshwdir(repo, origstatus, origsparsematch, force=False): | ||
"""Refreshes working directory by taking sparse config into account. | ||||
The old status and sparse matcher is compared against the current sparse | ||||
matcher. | ||||
Will abort if a file with pending changes is being excluded or included | ||||
unless ``force`` is True. | ||||
""" | ||||
# Verify there are no pending changes | ||||
pending = set() | ||||
Martin von Zweigbergk
|
r33356 | pending.update(origstatus.modified) | ||
pending.update(origstatus.added) | ||||
pending.update(origstatus.removed) | ||||
Gregory Szorc
|
r33324 | sparsematch = matcher(repo) | ||
abort = False | ||||
for f in pending: | ||||
if not sparsematch(f): | ||||
Augie Fackler
|
r43347 | repo.ui.warn(_(b"pending changes to '%s'\n") % f) | ||
Gregory Szorc
|
r33324 | abort = not force | ||
if abort: | ||||
Augie Fackler
|
r43346 | raise error.Abort( | ||
Martin von Zweigbergk
|
r43387 | _(b'could not update sparseness due to pending changes') | ||
Augie Fackler
|
r43346 | ) | ||
Gregory Szorc
|
r33324 | |||
# Calculate actions | ||||
dirstate = repo.dirstate | ||||
Augie Fackler
|
r43347 | ctx = repo[b'.'] | ||
Gregory Szorc
|
r33324 | added = [] | ||
lookup = [] | ||||
dropped = [] | ||||
mf = ctx.manifest() | ||||
files = set(mf) | ||||
actions = {} | ||||
for file in files: | ||||
old = origsparsematch(file) | ||||
new = sparsematch(file) | ||||
# Add files that are newly included, or that don't exist in | ||||
# the dirstate yet. | ||||
if (new and not old) or (old and new and not file in dirstate): | ||||
fl = mf.flags(file) | ||||
if repo.wvfs.exists(file): | ||||
Augie Fackler
|
r43347 | actions[file] = (b'e', (fl,), b'') | ||
Gregory Szorc
|
r33324 | lookup.append(file) | ||
else: | ||||
Augie Fackler
|
r43347 | actions[file] = (b'g', (fl, False), b'') | ||
Gregory Szorc
|
r33324 | added.append(file) | ||
# Drop files that are newly excluded, or that still exist in | ||||
# the dirstate. | ||||
elif (old and not new) or (not old and not new and file in dirstate): | ||||
dropped.append(file) | ||||
if file not in pending: | ||||
Augie Fackler
|
r43347 | actions[file] = (b'r', [], b'') | ||
Gregory Szorc
|
r33324 | |||
# Verify there are no pending changes in newly included files | ||||
abort = False | ||||
for file in lookup: | ||||
Augie Fackler
|
r43347 | repo.ui.warn(_(b"pending changes to '%s'\n") % file) | ||
Gregory Szorc
|
r33324 | abort = not force | ||
if abort: | ||||
Augie Fackler
|
r43346 | raise error.Abort( | ||
_( | ||||
Augie Fackler
|
r43347 | b'cannot change sparseness due to pending ' | ||
b'changes (delete the files or use ' | ||||
b'--force to bring them back dirty)' | ||||
Augie Fackler
|
r43346 | ) | ||
) | ||||
Gregory Szorc
|
r33324 | |||
# Check for files that were only in the dirstate. | ||||
Gregory Szorc
|
r43376 | for file, state in pycompat.iteritems(dirstate): | ||
Gregory Szorc
|
r33324 | if not file in files: | ||
old = origsparsematch(file) | ||||
new = sparsematch(file) | ||||
if old and not new: | ||||
dropped.append(file) | ||||
# Apply changes to disk | ||||
Martin von Zweigbergk
|
r41068 | typeactions = mergemod.emptyactions() | ||
Gregory Szorc
|
r43376 | for f, (m, args, msg) in pycompat.iteritems(actions): | ||
Gregory Szorc
|
r33324 | typeactions[m].append((f, args, msg)) | ||
Augie Fackler
|
r43346 | mergemod.applyupdates( | ||
Augie Fackler
|
r43347 | repo, typeactions, repo[None], repo[b'.'], False, wantfiledata=False | ||
Augie Fackler
|
r43346 | ) | ||
Gregory Szorc
|
r33324 | |||
# Fix dirstate | ||||
for file in added: | ||||
dirstate.normal(file) | ||||
for file in dropped: | ||||
dirstate.drop(file) | ||||
for file in lookup: | ||||
# File exists on disk, and we're bringing it back in an unknown state. | ||||
dirstate.normallookup(file) | ||||
return added, dropped, lookup | ||||
Gregory Szorc
|
r33353 | |||
Augie Fackler
|
r43346 | |||
Gregory Szorc
|
r33353 | def aftercommit(repo, node): | ||
"""Perform actions after a working directory commit.""" | ||||
# This function is called unconditionally, even if sparse isn't | ||||
# enabled. | ||||
ctx = repo[node] | ||||
profiles = patternsforrev(repo, ctx.rev())[2] | ||||
# profiles will only have data if sparse is enabled. | ||||
Gregory Szorc
|
r33550 | if profiles & set(ctx.files()): | ||
Gregory Szorc
|
r33353 | origstatus = repo.status() | ||
origsparsematch = matcher(repo) | ||||
refreshwdir(repo, origstatus, origsparsematch, force=True) | ||||
prunetemporaryincludes(repo) | ||||
Gregory Szorc
|
r33354 | |||
Augie Fackler
|
r43346 | |||
def _updateconfigandrefreshwdir( | ||||
repo, includes, excludes, profiles, force=False, removing=False | ||||
): | ||||
Gregory Szorc
|
r33555 | """Update the sparse config and working directory state.""" | ||
Augie Fackler
|
r43347 | raw = repo.vfs.tryread(b'sparse') | ||
oldincludes, oldexcludes, oldprofiles = parseconfig(repo.ui, raw, b'sparse') | ||||
Gregory Szorc
|
r33555 | |||
oldstatus = repo.status() | ||||
oldmatch = matcher(repo) | ||||
Gregory Szorc
|
r33556 | oldrequires = set(repo.requirements) | ||
Gregory Szorc
|
r33555 | |||
# TODO remove this try..except once the matcher integrates better | ||||
# with dirstate. We currently have to write the updated config | ||||
# because that will invalidate the matcher cache and force a | ||||
# re-read. We ideally want to update the cached matcher on the | ||||
# repo instance then flush the new config to disk once wdir is | ||||
# updated. But this requires massive rework to matcher() and its | ||||
# consumers. | ||||
Augie Fackler
|
r43347 | if b'exp-sparse' in oldrequires and removing: | ||
repo.requirements.discard(b'exp-sparse') | ||||
Gregory Szorc
|
r33556 | scmutil.writerequires(repo.vfs, repo.requirements) | ||
Augie Fackler
|
r43347 | elif b'exp-sparse' not in oldrequires: | ||
repo.requirements.add(b'exp-sparse') | ||||
Gregory Szorc
|
r33556 | scmutil.writerequires(repo.vfs, repo.requirements) | ||
Gregory Szorc
|
r33555 | |||
try: | ||||
Gregory Szorc
|
r33556 | writeconfig(repo, includes, excludes, profiles) | ||
Gregory Szorc
|
r33555 | return refreshwdir(repo, oldstatus, oldmatch, force=force) | ||
except Exception: | ||||
Gregory Szorc
|
r33556 | if repo.requirements != oldrequires: | ||
repo.requirements.clear() | ||||
repo.requirements |= oldrequires | ||||
scmutil.writerequires(repo.vfs, repo.requirements) | ||||
Gregory Szorc
|
r33555 | writeconfig(repo, oldincludes, oldexcludes, oldprofiles) | ||
raise | ||||
Augie Fackler
|
r43346 | |||
Gregory Szorc
|
r33354 | def clearrules(repo, force=False): | ||
"""Clears include/exclude rules from the sparse config. | ||||
The remaining sparse config only has profiles, if defined. The working | ||||
directory is refreshed, as needed. | ||||
""" | ||||
with repo.wlock(): | ||||
Augie Fackler
|
r43347 | raw = repo.vfs.tryread(b'sparse') | ||
includes, excludes, profiles = parseconfig(repo.ui, raw, b'sparse') | ||||
Gregory Szorc
|
r33354 | |||
if not includes and not excludes: | ||||
return | ||||
Gregory Szorc
|
r33555 | _updateconfigandrefreshwdir(repo, set(), set(), profiles, force=force) | ||
Gregory Szorc
|
r33355 | |||
Augie Fackler
|
r43346 | |||
Gregory Szorc
|
r33371 | def importfromfiles(repo, opts, paths, force=False): | ||
"""Import sparse config rules from files. | ||||
The updated sparse config is written out and the working directory | ||||
is refreshed, as needed. | ||||
""" | ||||
with repo.wlock(): | ||||
# read current configuration | ||||
Augie Fackler
|
r43347 | raw = repo.vfs.tryread(b'sparse') | ||
includes, excludes, profiles = parseconfig(repo.ui, raw, b'sparse') | ||||
Gregory Szorc
|
r33371 | aincludes, aexcludes, aprofiles = activeconfig(repo) | ||
# Import rules on top; only take in rules that are not yet | ||||
# part of the active rules. | ||||
changed = False | ||||
for p in paths: | ||||
Augie Fackler
|
r43347 | with util.posixfile(util.expandpath(p), mode=b'rb') as fh: | ||
Gregory Szorc
|
r33371 | raw = fh.read() | ||
Augie Fackler
|
r43346 | iincludes, iexcludes, iprofiles = parseconfig( | ||
Augie Fackler
|
r43347 | repo.ui, raw, b'sparse' | ||
Augie Fackler
|
r43346 | ) | ||
Gregory Szorc
|
r33371 | oldsize = len(includes) + len(excludes) + len(profiles) | ||
includes.update(iincludes - aincludes) | ||||
excludes.update(iexcludes - aexcludes) | ||||
Gregory Szorc
|
r33550 | profiles.update(iprofiles - aprofiles) | ||
Gregory Szorc
|
r33371 | if len(includes) + len(excludes) + len(profiles) > oldsize: | ||
changed = True | ||||
profilecount = includecount = excludecount = 0 | ||||
fcounts = (0, 0, 0) | ||||
if changed: | ||||
profilecount = len(profiles - aprofiles) | ||||
includecount = len(includes - aincludes) | ||||
excludecount = len(excludes - aexcludes) | ||||
Augie Fackler
|
r43346 | fcounts = map( | ||
len, | ||||
_updateconfigandrefreshwdir( | ||||
repo, includes, excludes, profiles, force=force | ||||
), | ||||
) | ||||
printchanges( | ||||
repo.ui, opts, profilecount, includecount, excludecount, *fcounts | ||||
) | ||||
Gregory Szorc
|
r33371 | |||
Augie Fackler
|
r43346 | def updateconfig( | ||
repo, | ||||
pats, | ||||
opts, | ||||
include=False, | ||||
exclude=False, | ||||
reset=False, | ||||
delete=False, | ||||
enableprofile=False, | ||||
disableprofile=False, | ||||
force=False, | ||||
usereporootpaths=False, | ||||
): | ||||
Gregory Szorc
|
r33374 | """Perform a sparse config update. | ||
Only one of the actions may be performed. | ||||
The new config is written out and a working directory refresh is performed. | ||||
""" | ||||
Gregory Szorc
|
r33375 | with repo.wlock(): | ||
Augie Fackler
|
r43347 | raw = repo.vfs.tryread(b'sparse') | ||
Augie Fackler
|
r43346 | oldinclude, oldexclude, oldprofiles = parseconfig( | ||
Augie Fackler
|
r43347 | repo.ui, raw, b'sparse' | ||
Augie Fackler
|
r43346 | ) | ||
Gregory Szorc
|
r33374 | |||
Gregory Szorc
|
r33376 | if reset: | ||
newinclude = set() | ||||
newexclude = set() | ||||
newprofiles = set() | ||||
else: | ||||
newinclude = set(oldinclude) | ||||
newexclude = set(oldexclude) | ||||
newprofiles = set(oldprofiles) | ||||
Gregory Szorc
|
r33374 | |||
Kostia Balytskyi
|
r33646 | if any(os.path.isabs(pat) for pat in pats): | ||
Augie Fackler
|
r43347 | raise error.Abort(_(b'paths cannot be absolute')) | ||
Kostia Balytskyi
|
r33646 | |||
Kostia Balytskyi
|
r33648 | if not usereporootpaths: | ||
# let's treat paths as relative to cwd | ||||
root, cwd = repo.root, repo.getcwd() | ||||
abspats = [] | ||||
for kindpat in pats: | ||||
kind, pat = matchmod._patsplit(kindpat, None) | ||||
if kind in matchmod.cwdrelativepatternkinds or kind is None: | ||||
Augie Fackler
|
r43347 | ap = (kind + b':' if kind else b'') + pathutil.canonpath( | ||
Augie Fackler
|
r43346 | root, cwd, pat | ||
) | ||||
Kostia Balytskyi
|
r33648 | abspats.append(ap) | ||
else: | ||||
abspats.append(kindpat) | ||||
pats = abspats | ||||
Kostia Balytskyi
|
r33646 | if include: | ||
Gregory Szorc
|
r33376 | newinclude.update(pats) | ||
elif exclude: | ||||
newexclude.update(pats) | ||||
elif enableprofile: | ||||
newprofiles.update(pats) | ||||
elif disableprofile: | ||||
newprofiles.difference_update(pats) | ||||
elif delete: | ||||
newinclude.difference_update(pats) | ||||
newexclude.difference_update(pats) | ||||
Gregory Szorc
|
r33374 | |||
Augie Fackler
|
r43346 | profilecount = len(newprofiles - oldprofiles) - len( | ||
oldprofiles - newprofiles | ||||
) | ||||
includecount = len(newinclude - oldinclude) - len( | ||||
oldinclude - newinclude | ||||
) | ||||
excludecount = len(newexclude - oldexclude) - len( | ||||
oldexclude - newexclude | ||||
) | ||||
Gregory Szorc
|
r33374 | |||
Augie Fackler
|
r43346 | fcounts = map( | ||
len, | ||||
_updateconfigandrefreshwdir( | ||||
repo, | ||||
newinclude, | ||||
newexclude, | ||||
newprofiles, | ||||
force=force, | ||||
removing=reset, | ||||
), | ||||
) | ||||
Gregory Szorc
|
r33376 | |||
Augie Fackler
|
r43346 | printchanges( | ||
repo.ui, opts, profilecount, includecount, excludecount, *fcounts | ||||
) | ||||
Gregory Szorc
|
r33374 | |||
Augie Fackler
|
r43346 | def printchanges( | ||
ui, | ||||
opts, | ||||
profilecount=0, | ||||
includecount=0, | ||||
excludecount=0, | ||||
added=0, | ||||
dropped=0, | ||||
conflicting=0, | ||||
): | ||||
Gregory Szorc
|
r33355 | """Print output summarizing sparse config changes.""" | ||
Augie Fackler
|
r43347 | with ui.formatter(b'sparse', opts) as fm: | ||
Gregory Szorc
|
r33355 | fm.startitem() | ||
Augie Fackler
|
r43346 | fm.condwrite( | ||
ui.verbose, | ||||
Augie Fackler
|
r43347 | b'profiles_added', | ||
_(b'Profiles changed: %d\n'), | ||||
Augie Fackler
|
r43346 | profilecount, | ||
) | ||||
fm.condwrite( | ||||
ui.verbose, | ||||
Augie Fackler
|
r43347 | b'include_rules_added', | ||
_(b'Include rules changed: %d\n'), | ||||
Augie Fackler
|
r43346 | includecount, | ||
) | ||||
fm.condwrite( | ||||
ui.verbose, | ||||
Augie Fackler
|
r43347 | b'exclude_rules_added', | ||
_(b'Exclude rules changed: %d\n'), | ||||
Augie Fackler
|
r43346 | excludecount, | ||
) | ||||
Gregory Szorc
|
r33355 | |||
# In 'plain' verbose mode, mergemod.applyupdates already outputs what | ||||
# files are added or removed outside of the templating formatter | ||||
# framework. No point in repeating ourselves in that case. | ||||
if not fm.isplain(): | ||||
Augie Fackler
|
r43346 | fm.condwrite( | ||
Augie Fackler
|
r43347 | ui.verbose, b'files_added', _(b'Files added: %d\n'), added | ||
Augie Fackler
|
r43346 | ) | ||
fm.condwrite( | ||||
Augie Fackler
|
r43347 | ui.verbose, b'files_dropped', _(b'Files dropped: %d\n'), dropped | ||
Augie Fackler
|
r43346 | ) | ||
fm.condwrite( | ||||
ui.verbose, | ||||
Augie Fackler
|
r43347 | b'files_conflicting', | ||
_(b'Files conflicting: %d\n'), | ||||
Augie Fackler
|
r43346 | conflicting, | ||
) | ||||