reposetup.py
477 lines
| 18.6 KiB
| text/x-python
|
PythonLexer
various
|
r15168 | # Copyright 2009-2010 Gregory P. Ward | ||
# Copyright 2009-2010 Intelerad Medical Systems Incorporated | ||||
# Copyright 2010-2011 Fog Creek Software | ||||
# Copyright 2010-2011 Unity Technologies | ||||
# | ||||
# This software may be used and distributed according to the terms of the | ||||
# GNU General Public License version 2 or any later version. | ||||
'''setup for largefiles repositories: reposetup''' | ||||
liscju
|
r29314 | |||
Matt Harbison
|
r52756 | from __future__ import annotations | ||
various
|
r15168 | import copy | ||
from mercurial.i18n import _ | ||||
liscju
|
r29314 | from mercurial import ( | ||
error, | ||||
Martin von Zweigbergk
|
r43986 | extensions, | ||
liscju
|
r29314 | localrepo, | ||
liscju
|
r29319 | match as matchmod, | ||
liscju
|
r29314 | scmutil, | ||
Martin von Zweigbergk
|
r43984 | util, | ||
liscju
|
r29314 | ) | ||
r49205 | from mercurial.dirstateutils import timestamp | |||
liscju
|
r29314 | from . import ( | ||
lfcommands, | ||||
lfutil, | ||||
) | ||||
various
|
r15168 | |||
Augie Fackler
|
r43346 | |||
various
|
r15168 | def reposetup(ui, repo): | ||
FUJIWARA Katsunori
|
r20858 | # wire repositories should be given new wireproto functions | ||
# by "proto.wirereposetup()" via "hg.wirepeersetupfuncs" | ||||
various
|
r15168 | if not repo.local(): | ||
FUJIWARA Katsunori
|
r20858 | return | ||
various
|
r15168 | |||
Na'Tosha Bard
|
r16247 | class lfilesrepo(repo.__class__): | ||
FUJIWARA Katsunori
|
r24158 | # the mark to examine whether "repo" object enables largefiles or not | ||
_largefilesenabled = True | ||||
various
|
r15168 | lfstatus = False | ||
Augie Fackler
|
r43346 | |||
Greg Ward
|
r15252 | # When lfstatus is set, return a context that gives the names | ||
# of largefiles instead of their corresponding standins and | ||||
# identifies the largefiles as always binary, regardless of | ||||
# their actual contents. | ||||
various
|
r15168 | def __getitem__(self, changeid): | ||
Na'Tosha Bard
|
r16247 | ctx = super(lfilesrepo, self).__getitem__(changeid) | ||
Matt Harbison
|
r23958 | if self.lfstatus: | ||
Augie Fackler
|
r43346 | |||
Martin von Zweigbergk
|
r43986 | def files(orig): | ||
filenames = orig() | ||||
return [lfutil.splitstandin(f) or f for f in filenames] | ||||
Augie Fackler
|
r43346 | |||
Martin von Zweigbergk
|
r43986 | extensions.wrapfunction(ctx, 'files', files) | ||
def manifest(orig): | ||||
man1 = orig() | ||||
Augie Fackler
|
r43346 | |||
Martin von Zweigbergk
|
r43986 | class lfilesmanifest(man1.__class__): | ||
def __contains__(self, filename): | ||||
orig = super(lfilesmanifest, self).__contains__ | ||||
return orig(filename) or orig( | ||||
lfutil.standin(filename) | ||||
) | ||||
Augie Fackler
|
r43346 | |||
Martin von Zweigbergk
|
r43986 | man1.__class__ = lfilesmanifest | ||
return man1 | ||||
Augie Fackler
|
r43346 | |||
Martin von Zweigbergk
|
r43986 | extensions.wrapfunction(ctx, 'manifest', manifest) | ||
Augie Fackler
|
r43346 | |||
Martin von Zweigbergk
|
r43986 | def filectx(orig, path, fileid=None, filelog=None): | ||
try: | ||||
if filelog is not None: | ||||
result = orig(path, fileid, filelog) | ||||
else: | ||||
result = orig(path, fileid) | ||||
except error.LookupError: | ||||
# Adding a null character will cause Mercurial to | ||||
# identify this as a binary file. | ||||
if filelog is not None: | ||||
result = orig(lfutil.standin(path), fileid, filelog) | ||||
else: | ||||
result = orig(lfutil.standin(path), fileid) | ||||
olddata = result.data | ||||
result.data = lambda: olddata() + b'\0' | ||||
return result | ||||
extensions.wrapfunction(ctx, 'filectx', filectx) | ||||
various
|
r15168 | return ctx | ||
# Figure out the status of big files and insert them into the | ||||
Greg Ward
|
r15252 | # appropriate list in the result. Also removes standin files | ||
# from the listing. Revert to the original status if | ||||
# self.lfstatus is False. | ||||
Matt Harbison
|
r23958 | # XXX large file status is buggy when used on repo proxy. | ||
# XXX this needs to be investigated. | ||||
@localrepo.unfilteredmethod | ||||
Augie Fackler
|
r43346 | def status( | ||
self, | ||||
Augie Fackler
|
r43347 | node1=b'.', | ||
Augie Fackler
|
r43346 | node2=None, | ||
match=None, | ||||
ignored=False, | ||||
clean=False, | ||||
unknown=False, | ||||
listsubrepos=False, | ||||
): | ||||
various
|
r15168 | listignored, listclean, listunknown = ignored, clean, unknown | ||
Martin von Zweigbergk
|
r22518 | orig = super(lfilesrepo, self).status | ||
Matt Harbison
|
r23958 | if not self.lfstatus: | ||
Augie Fackler
|
r43346 | return orig( | ||
node1, | ||||
node2, | ||||
match, | ||||
listignored, | ||||
listclean, | ||||
listunknown, | ||||
listsubrepos, | ||||
) | ||||
Martin von Zweigbergk
|
r22515 | |||
# some calls in this function rely on the old version of status | ||||
Matt Harbison
|
r23958 | self.lfstatus = False | ||
Martin von Zweigbergk
|
r22515 | ctx1 = self[node1] | ||
ctx2 = self[node2] | ||||
working = ctx2.rev() is None | ||||
Augie Fackler
|
r43347 | parentworking = working and ctx1 == self[b'.'] | ||
various
|
r15168 | |||
Martin von Zweigbergk
|
r22515 | if match is None: | ||
Martin von Zweigbergk
|
r41825 | match = matchmod.always() | ||
various
|
r15168 | |||
Martin von Zweigbergk
|
r22515 | try: | ||
Martin von Zweigbergk
|
r43984 | # updating the dirstate is optional | ||
# so we don't wait on the lock | ||||
wlock = self.wlock(False) | ||||
gotlock = True | ||||
except error.LockError: | ||||
wlock = util.nullcontextmanager() | ||||
gotlock = False | ||||
r51031 | with wlock, self.dirstate.running_status(self): | |||
Martin von Zweigbergk
|
r23146 | # First check if paths or patterns were specified on the | ||
# command line. If there were, and they don't match any | ||||
Martin von Zweigbergk
|
r22515 | # largefiles, we should just bail here and let super | ||
# handle it -- thus gaining a big performance boost. | ||||
lfdirstate = lfutil.openlfdirstate(ui, self) | ||||
Martin von Zweigbergk
|
r23146 | if not match.always(): | ||
Martin von Zweigbergk
|
r22515 | for f in lfdirstate: | ||
if match(f): | ||||
break | ||||
else: | ||||
Augie Fackler
|
r43346 | return orig( | ||
node1, | ||||
node2, | ||||
match, | ||||
listignored, | ||||
listclean, | ||||
listunknown, | ||||
listsubrepos, | ||||
) | ||||
various
|
r15168 | |||
Martin von Zweigbergk
|
r22515 | # Create a copy of match that matches standins instead | ||
# of largefiles. | ||||
def tostandins(files): | ||||
if not working: | ||||
return files | ||||
newfiles = [] | ||||
dirstate = self.dirstate | ||||
for f in files: | ||||
sf = lfutil.standin(f) | ||||
if sf in dirstate: | ||||
newfiles.append(sf) | ||||
Mark Thomas
|
r35083 | elif dirstate.hasdir(sf): | ||
Martin von Zweigbergk
|
r22515 | # Directory entries could be regular or | ||
# standin, check both | ||||
newfiles.extend((f, sf)) | ||||
else: | ||||
newfiles.append(f) | ||||
return newfiles | ||||
Mads Kiilerich
|
r18149 | |||
Martin von Zweigbergk
|
r22515 | m = copy.copy(match) | ||
Arseniy Alekseyev
|
r52517 | m._was_tampered_with = True | ||
Martin von Zweigbergk
|
r22515 | m._files = tostandins(m._files) | ||
Augie Fackler
|
r43346 | result = orig( | ||
node1, node2, m, ignored, clean, unknown, listsubrepos | ||||
) | ||||
Martin von Zweigbergk
|
r22515 | if working: | ||
Na'Tosha Bard
|
r15617 | |||
Martin von Zweigbergk
|
r22515 | def sfindirstate(f): | ||
sf = lfutil.standin(f) | ||||
dirstate = self.dirstate | ||||
Mark Thomas
|
r35083 | return sf in dirstate or dirstate.hasdir(sf) | ||
Mads Kiilerich
|
r18149 | |||
Arseniy Alekseyev
|
r52517 | match._was_tampered_with = True | ||
Augie Fackler
|
r43346 | match._files = [f for f in match._files if sfindirstate(f)] | ||
Martin von Zweigbergk
|
r22515 | # Don't waste time getting the ignored and unknown | ||
# files from lfdirstate | ||||
r49213 | unsure, s, mtime_boundary = lfdirstate.status( | |||
Augie Fackler
|
r43346 | match, | ||
subrepos=[], | ||||
ignored=False, | ||||
clean=listclean, | ||||
unknown=False, | ||||
) | ||||
Mads Kiilerich
|
r30191 | (modified, added, removed, deleted, clean) = ( | ||
Augie Fackler
|
r43346 | s.modified, | ||
s.added, | ||||
s.removed, | ||||
s.deleted, | ||||
s.clean, | ||||
) | ||||
Martin von Zweigbergk
|
r22515 | if parentworking: | ||
r49205 | wctx = repo[None] | |||
Martin von Zweigbergk
|
r22515 | for lfile in unsure: | ||
standin = lfutil.standin(lfile) | ||||
if standin not in ctx1: | ||||
# from second parent | ||||
modified.append(lfile) | ||||
Augie Fackler
|
r43346 | elif lfutil.readasstandin( | ||
ctx1[standin] | ||||
) != lfutil.hashfile(self.wjoin(lfile)): | ||||
Martin von Zweigbergk
|
r22515 | modified.append(lfile) | ||
else: | ||||
Martin von Zweigbergk
|
r22523 | if listclean: | ||
clean.append(lfile) | ||||
r49205 | s = wctx[lfile].lstat() | |||
mode = s.st_mode | ||||
size = s.st_size | ||||
r49225 | mtime = timestamp.reliable_mtime_of( | |||
s, mtime_boundary | ||||
) | ||||
if mtime is not None: | ||||
cache_data = (mode, size, mtime) | ||||
lfdirstate.set_clean(lfile, cache_data) | ||||
Martin von Zweigbergk
|
r22515 | else: | ||
tocheck = unsure + modified + added + clean | ||||
modified, added, clean = [], [], [] | ||||
FUJIWARA Katsunori
|
r23383 | checkexec = self.dirstate._checkexec | ||
Martin von Zweigbergk
|
r22515 | |||
for lfile in tocheck: | ||||
standin = lfutil.standin(lfile) | ||||
Mads Kiilerich
|
r23043 | if standin in ctx1: | ||
FUJIWARA Katsunori
|
r23090 | abslfile = self.wjoin(lfile) | ||
Augie Fackler
|
r43346 | if ( | ||
lfutil.readasstandin(ctx1[standin]) | ||||
!= lfutil.hashfile(abslfile) | ||||
) or ( | ||||
checkexec | ||||
Augie Fackler
|
r43347 | and (b'x' in ctx1.flags(standin)) | ||
Augie Fackler
|
r43346 | != bool(lfutil.getexecutable(abslfile)) | ||
): | ||||
various
|
r15168 | modified.append(lfile) | ||
Martin von Zweigbergk
|
r22523 | elif listclean: | ||
various
|
r15168 | clean.append(lfile) | ||
Martin von Zweigbergk
|
r22515 | else: | ||
added.append(lfile) | ||||
various
|
r15168 | |||
FUJIWARA Katsunori
|
r23089 | # at this point, 'removed' contains largefiles | ||
# marked as 'R' in the working context. | ||||
# then, largefiles not managed also in the target | ||||
# context should be excluded from 'removed'. | ||||
Augie Fackler
|
r43346 | removed = [ | ||
lfile | ||||
for lfile in removed | ||||
if lfutil.standin(lfile) in ctx1 | ||||
] | ||||
FUJIWARA Katsunori
|
r23089 | |||
Mads Kiilerich
|
r30191 | # Standins no longer found in lfdirstate have been deleted | ||
Martin von Zweigbergk
|
r22525 | for standin in ctx1.walk(lfutil.getstandinmatcher(self)): | ||
Martin von Zweigbergk
|
r22515 | lfile = lfutil.splitstandin(standin) | ||
if not match(lfile): | ||||
continue | ||||
if lfile not in lfdirstate: | ||||
Mads Kiilerich
|
r30191 | deleted.append(lfile) | ||
# Sync "largefile has been removed" back to the | ||||
# standin. Removing a file as a side effect of | ||||
# running status is gross, but the alternatives (if | ||||
# any) are worse. | ||||
Mads Kiilerich
|
r30233 | self.wvfs.unlinkpath(standin, ignoremissing=True) | ||
Martin Geisler
|
r15663 | |||
Martin von Zweigbergk
|
r22515 | # Filter result lists | ||
result = list(result) | ||||
Mads Kiilerich
|
r19056 | |||
Martin von Zweigbergk
|
r22515 | # Largefiles are not really removed when they're | ||
# still in the normal dirstate. Likewise, normal | ||||
# files are not really removed if they are still in | ||||
# lfdirstate. This happens in merges where files | ||||
# change type. | ||||
Augie Fackler
|
r43346 | removed = [f for f in removed if f not in self.dirstate] | ||
result[2] = [f for f in result[2] if f not in lfdirstate] | ||||
Martin Geisler
|
r15663 | |||
Augie Fackler
|
r43196 | lfiles = set(lfdirstate) | ||
Martin von Zweigbergk
|
r22515 | # Unknown files | ||
result[4] = set(result[4]).difference(lfiles) | ||||
# Ignored files | ||||
result[5] = set(result[5]).difference(lfiles) | ||||
# combine normal files and largefiles | ||||
Augie Fackler
|
r43346 | normals = [ | ||
[fn for fn in filelist if not lfutil.isstandin(fn)] | ||||
for filelist in result | ||||
] | ||||
lfstatus = ( | ||||
modified, | ||||
added, | ||||
removed, | ||||
deleted, | ||||
[], | ||||
[], | ||||
clean, | ||||
) | ||||
result = [ | ||||
sorted(list1 + list2) | ||||
for (list1, list2) in zip(normals, lfstatus) | ||||
] | ||||
else: # not against working directory | ||||
result = [ | ||||
[lfutil.splitstandin(f) or f for f in items] | ||||
for items in result | ||||
] | ||||
Martin Geisler
|
r15663 | |||
Martin von Zweigbergk
|
r43984 | if gotlock: | ||
Pulkit Goyal
|
r48982 | lfdirstate.write(self.currenttransaction()) | ||
r51031 | else: | |||
lfdirstate.invalidate() | ||||
Martin von Zweigbergk
|
r22515 | |||
Matt Harbison
|
r23958 | self.lfstatus = True | ||
Martin von Zweigbergk
|
r22914 | return scmutil.status(*result) | ||
various
|
r15168 | |||
FUJIWARA Katsunori
|
r23184 | def commitctx(self, ctx, *args, **kwargs): | ||
node = super(lfilesrepo, self).commitctx(ctx, *args, **kwargs) | ||||
Augie Fackler
|
r43346 | |||
FUJIWARA Katsunori
|
r23184 | class lfilesctx(ctx.__class__): | ||
def markcommitted(self, node): | ||||
orig = super(lfilesctx, self).markcommitted | ||||
return lfutil.markcommitted(orig, self, node) | ||||
Augie Fackler
|
r43346 | |||
FUJIWARA Katsunori
|
r23184 | ctx.__class__ = lfilesctx | ||
various
|
r15168 | return node | ||
Greg Ward
|
r15254 | # Before commit, largefile standins have not had their | ||
# contents updated to reflect the hash of their largefile. | ||||
# Do that here. | ||||
Augie Fackler
|
r43346 | def commit( | ||
self, | ||||
Augie Fackler
|
r43347 | text=b"", | ||
Augie Fackler
|
r43346 | user=None, | ||
date=None, | ||||
match=None, | ||||
force=False, | ||||
editor=False, | ||||
extra=None, | ||||
): | ||||
Pierre-Yves David
|
r31410 | if extra is None: | ||
extra = {} | ||||
Na'Tosha Bard
|
r16247 | orig = super(lfilesrepo, self).commit | ||
various
|
r15168 | |||
Bryan O'Sullivan
|
r27842 | with self.wlock(): | ||
FUJIWARA Katsunori
|
r23186 | lfcommithook = self._lfcommithooks[-1] | ||
match = lfcommithook(self, match) | ||||
Augie Fackler
|
r43346 | result = orig( | ||
text=text, | ||||
user=user, | ||||
date=date, | ||||
match=match, | ||||
force=force, | ||||
editor=editor, | ||||
extra=extra, | ||||
) | ||||
Levi Bard
|
r15794 | return result | ||
various
|
r15168 | |||
FUJIWARA Katsunori
|
r23185 | # TODO: _subdirlfs should be moved into "lfutil.py", because | ||
# it is referred only from "lfutil.updatestandinsbymatch" | ||||
Levi Bard
|
r18064 | def _subdirlfs(self, files, lfiles): | ||
Augie Fackler
|
r46554 | """ | ||
Levi Bard
|
r18064 | Adjust matched file list | ||
Mads Kiilerich
|
r23543 | If we pass a directory to commit whose only committable files | ||
Levi Bard
|
r18064 | are largefiles, the core commit code aborts before finding | ||
the largefiles. | ||||
So we do the following: | ||||
For directories that only have largefiles as matches, | ||||
Mads Kiilerich
|
r18644 | we explicitly add the largefiles to the match list and remove | ||
Levi Bard
|
r18064 | the directory. | ||
In other cases, we leave the match list unmodified. | ||||
Augie Fackler
|
r46554 | """ | ||
Levi Bard
|
r18064 | actualfiles = [] | ||
dirs = [] | ||||
regulars = [] | ||||
for f in files: | ||||
Augie Fackler
|
r43347 | if lfutil.isstandin(f + b'/'): | ||
Pierre-Yves David
|
r26587 | raise error.Abort( | ||
Augie Fackler
|
r43347 | _(b'file "%s" is a largefile standin') % f, | ||
hint=b'commit the largefile itself instead', | ||||
Augie Fackler
|
r43346 | ) | ||
Levi Bard
|
r18064 | # Scan directories | ||
liscju
|
r28716 | if self.wvfs.isdir(f): | ||
Levi Bard
|
r18064 | dirs.append(f) | ||
else: | ||||
regulars.append(f) | ||||
for f in dirs: | ||||
matcheddir = False | ||||
Augie Fackler
|
r43347 | d = self.dirstate.normalize(f) + b'/' | ||
Levi Bard
|
r18064 | # Check for matched normal files | ||
for mf in regulars: | ||||
if self.dirstate.normalize(mf).startswith(d): | ||||
actualfiles.append(f) | ||||
matcheddir = True | ||||
break | ||||
if not matcheddir: | ||||
# If no normal match, manually append | ||||
# any matching largefiles | ||||
for lf in lfiles: | ||||
if self.dirstate.normalize(lf).startswith(d): | ||||
actualfiles.append(lf) | ||||
if not matcheddir: | ||||
Matt Harbison
|
r23923 | # There may still be normal files in the dir, so | ||
Mads Kiilerich
|
r24180 | # add a directory to the list, which | ||
Matt Harbison
|
r24007 | # forces status/dirstate to walk all files and | ||
# call the match function on the matcher, even | ||||
Mads Kiilerich
|
r24180 | # on case sensitive filesystems. | ||
Augie Fackler
|
r43347 | actualfiles.append(b'.') | ||
Levi Bard
|
r18064 | matcheddir = True | ||
# Nothing in dir, so readd it | ||||
# and let commit reject it | ||||
if not matcheddir: | ||||
actualfiles.append(f) | ||||
# Always add normal files | ||||
actualfiles += regulars | ||||
return actualfiles | ||||
Na'Tosha Bard
|
r16247 | repo.__class__ = lfilesrepo | ||
various
|
r15168 | |||
FUJIWARA Katsunori
|
r23186 | # stack of hooks being executed before committing. | ||
# only last element ("_lfcommithooks[-1]") is used for each committing. | ||||
repo._lfcommithooks = [lfutil.updatestandinsbymatch] | ||||
FUJIWARA Katsunori
|
r23188 | # Stack of status writer functions taking "*msg, **opts" arguments | ||
Mads Kiilerich
|
r23543 | # like "ui.status()". Only last element ("_lfstatuswriters[-1]") | ||
FUJIWARA Katsunori
|
r23188 | # is used to write status out. | ||
repo._lfstatuswriters = [ui.status] | ||||
Mads Kiilerich
|
r28876 | def prepushoutgoinghook(pushop): | ||
Mads Kiilerich
|
r28878 | """Push largefiles for pushop before pushing revisions.""" | ||
lfrevs = pushop.lfrevs | ||||
if lfrevs is None: | ||||
lfrevs = pushop.outgoing.missing | ||||
if lfrevs: | ||||
FUJIWARA Katsunori
|
r21044 | toupload = set() | ||
addfunc = lambda fn, lfhash: toupload.add(lfhash) | ||||
Augie Fackler
|
r43346 | lfutil.getlfilestoupload(pushop.repo, lfrevs, addfunc) | ||
Mads Kiilerich
|
r28876 | lfcommands.uploadlfiles(ui, pushop.repo, pushop.remote, toupload) | ||
Augie Fackler
|
r43346 | |||
Augie Fackler
|
r43347 | repo.prepushoutgoinghooks.add(b"largefiles", prepushoutgoinghook) | ||
FUJIWARA Katsunori
|
r21044 | |||
various
|
r15168 | def checkrequireslfiles(ui, repo, **kwargs): | ||
r49510 | with repo.lock(): | |||
r51381 | if b'largefiles' in repo.requirements: | |||
return | ||||
marker = lfutil.shortnameslash | ||||
r51397 | for entry in repo.store.data_entries(): | |||
r51381 | # XXX note that this match is not rooted and can wrongly match | |||
# directory ending with ".hglf" | ||||
if entry.is_revlog and marker in entry.target_id: | ||||
repo.requirements.add(b'largefiles') | ||||
scmutil.writereporequirements(repo) | ||||
break | ||||
various
|
r15168 | |||
Augie Fackler
|
r43346 | ui.setconfig( | ||
Augie Fackler
|
r43347 | b'hooks', b'changegroup.lfiles', checkrequireslfiles, b'largefiles' | ||
Augie Fackler
|
r43346 | ) | ||
Augie Fackler
|
r43347 | ui.setconfig(b'hooks', b'commit.lfiles', checkrequireslfiles, b'largefiles') | ||