hg.py
1690 lines
| 54.5 KiB
| text/x-python
|
PythonLexer
/ mercurial / hg.py
mpm@selenic.com
|
r0 | # hg.py - repository classes for mercurial | ||
# | ||||
Raphaël Gomès
|
r47575 | # Copyright 2005-2007 Olivia Mackall <olivia@selenic.com> | ||
Vadim Gelfer
|
r2859 | # Copyright 2006 Vadim Gelfer <vadim.gelfer@gmail.com> | ||
mpm@selenic.com
|
r0 | # | ||
Martin Geisler
|
r8225 | # 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
|
r0 | |||
Matt Harbison
|
r52756 | from __future__ import annotations | ||
Gregory Szorc
|
r25939 | |||
import os | ||||
Matt Harbison
|
r49400 | import posixpath | ||
Gregory Szorc
|
r25939 | import shutil | ||
Augie Fackler
|
r36799 | import stat | ||
Matt Harbison
|
r52567 | import typing | ||
Arseniy Alekseyev
|
r49326 | import weakref | ||
Gregory Szorc
|
r25939 | |||
from .i18n import _ | ||||
Joerg Sonnenberger
|
r46729 | from .node import ( | ||
hex, | ||||
Joerg Sonnenberger
|
r47771 | sha1nodeconstants, | ||
Joerg Sonnenberger
|
r46729 | short, | ||
) | ||||
Jordi Gutiérrez Hermoso
|
r22837 | |||
Gregory Szorc
|
r25939 | from . import ( | ||
bookmarks, | ||||
bundlerepo, | ||||
cmdutil, | ||||
FUJIWARA Katsunori
|
r28501 | destutil, | ||
Gregory Szorc
|
r25939 | discovery, | ||
error, | ||||
exchange, | ||||
extensions, | ||||
r47677 | graphmod, | |||
Gregory Szorc
|
r25939 | httppeer, | ||
localrepo, | ||||
lock, | ||||
Yuya Nishihara
|
r35906 | logcmdutil, | ||
Pulkit Goyal
|
r35348 | logexchange, | ||
Gregory Szorc
|
r25939 | merge as mergemod, | ||
Augie Fackler
|
r45383 | mergestate as mergestatemod, | ||
Gregory Szorc
|
r39586 | narrowspec, | ||
Gregory Szorc
|
r25939 | phases, | ||
Pulkit Goyal
|
r45932 | requirements, | ||
Gregory Szorc
|
r25939 | scmutil, | ||
sshpeer, | ||||
statichttprepo, | ||||
ui as uimod, | ||||
unionrepo, | ||||
url, | ||||
util, | ||||
verify as verifymod, | ||||
Pierre-Yves David
|
r31218 | vfs as vfsmod, | ||
Gregory Szorc
|
r25939 | ) | ||
r48079 | from .interfaces import repository as repositorymod | |||
Matt Harbison
|
r47518 | from .utils import ( | ||
hashutil, | ||||
stringutil, | ||||
r47669 | urlutil, | |||
Matt Harbison
|
r47518 | ) | ||
Matt Harbison
|
r52567 | if typing.TYPE_CHECKING: | ||
from typing import ( | ||||
List, | ||||
Tuple, | ||||
) | ||||
Pulkit Goyal
|
r43078 | |||
Gregory Szorc
|
r25939 | release = lock.release | ||
mpm@selenic.com
|
r0 | |||
Martijn Pieters
|
r29424 | # shared features | ||
Augie Fackler
|
r43347 | sharedbookmarks = b'bookmarks' | ||
Martijn Pieters
|
r29424 | |||
Augie Fackler
|
r43346 | |||
Manuel Jacob
|
r51309 | def addbranchrevs(lrepo, other, branches, revs, remotehidden=False): | ||
r51821 | if hasattr(other, 'peer'): | |||
r50641 | # a courtesy to callers using a localrepo for other | |||
Manuel Jacob
|
r51309 | peer = other.peer(remotehidden=remotehidden) | ||
r50641 | else: | |||
peer = other | ||||
Sune Foldager
|
r11322 | hashbranch, branches = branches | ||
if not hashbranch and not branches: | ||||
Pierre-Yves David
|
r22818 | x = revs or None | ||
Martin von Zweigbergk
|
r37499 | if revs: | ||
Pierre-Yves David
|
r22818 | y = revs[0] | ||
else: | ||||
y = None | ||||
return x, y | ||||
Jordi Gutiérrez Hermoso
|
r24306 | if revs: | ||
revs = list(revs) | ||||
else: | ||||
revs = [] | ||||
Augie Fackler
|
r43347 | if not peer.capable(b'branchmap'): | ||
Sune Foldager
|
r11322 | if branches: | ||
Augie Fackler
|
r43347 | raise error.Abort(_(b"remote branch lookup not supported")) | ||
Sune Foldager
|
r11322 | revs.append(hashbranch) | ||
Sune Foldager
|
r10380 | return revs, revs[0] | ||
Gregory Szorc
|
r37658 | |||
with peer.commandexecutor() as e: | ||||
Augie Fackler
|
r43347 | branchmap = e.callcommand(b'branchmap', {}).result() | ||
Sune Foldager
|
r11322 | |||
Matt Mackall
|
r13047 | def primary(branch): | ||
Augie Fackler
|
r43347 | if branch == b'.': | ||
Sune Foldager
|
r17191 | if not lrepo: | ||
Augie Fackler
|
r43347 | raise error.Abort(_(b"dirstate branch not accessible")) | ||
Matt Mackall
|
r13047 | branch = lrepo.dirstate.branch() | ||
if branch in branchmap: | ||||
Joerg Sonnenberger
|
r46729 | revs.extend(hex(r) for r in reversed(branchmap[branch])) | ||
Sune Foldager
|
r11322 | return True | ||
Sune Foldager
|
r10365 | else: | ||
Sune Foldager
|
r11322 | return False | ||
for branch in branches: | ||||
Matt Mackall
|
r13047 | if not primary(branch): | ||
Augie Fackler
|
r43347 | raise error.RepoLookupError(_(b"unknown branch '%s'") % branch) | ||
Sune Foldager
|
r11322 | if hashbranch: | ||
Matt Mackall
|
r13047 | if not primary(hashbranch): | ||
Sune Foldager
|
r11322 | revs.append(hashbranch) | ||
Sune Foldager
|
r10365 | return revs, revs[0] | ||
Augie Fackler
|
r43346 | |||
r50580 | def _isfile(path): | |||
try: | ||||
# we use os.stat() directly here instead of os.path.isfile() | ||||
# because the latter started returning `False` on invalid path | ||||
# exceptions starting in 3.8 and we care about handling | ||||
# invalid paths specially here. | ||||
st = os.stat(path) | ||||
except ValueError as e: | ||||
msg = stringutil.forcebytestr(e) | ||||
raise error.Abort(_(b'invalid path %s: %s') % (path, msg)) | ||||
except OSError: | ||||
return False | ||||
else: | ||||
return stat.S_ISREG(st.st_mode) | ||||
class LocalFactory: | ||||
"""thin wrapper to dispatch between localrepo and bundle repo""" | ||||
@staticmethod | ||||
def islocal(path: bytes) -> bool: | ||||
path = util.expandpath(urlutil.urllocalpath(path)) | ||||
return not _isfile(path) | ||||
@staticmethod | ||||
def instance(ui, path, *args, **kwargs): | ||||
path = util.expandpath(urlutil.urllocalpath(path)) | ||||
if _isfile(path): | ||||
cls = bundlerepo | ||||
else: | ||||
cls = localrepo | ||||
return cls.instance(ui, path, *args, **kwargs) | ||||
r50584 | repo_schemes = { | |||
Augie Fackler
|
r43347 | b'bundle': bundlerepo, | ||
b'union': unionrepo, | ||||
r50580 | b'file': LocalFactory, | |||
r50584 | } | |||
peer_schemes = { | ||||
Augie Fackler
|
r43347 | b'http': httppeer, | ||
b'https': httppeer, | ||||
b'ssh': sshpeer, | ||||
b'static-http': statichttprepo, | ||||
Matt Mackall
|
r14568 | } | ||
Augie Fackler
|
r43346 | |||
Matt Mackall
|
r14605 | def islocal(repo): | ||
Siddharth Agarwal
|
r20355 | '''return true if repo (or path pointing to repo) is local''' | ||
Pulkit Goyal
|
r33018 | if isinstance(repo, bytes): | ||
r50644 | u = urlutil.url(repo) | |||
scheme = u.scheme or b'file' | ||||
if scheme in peer_schemes: | ||||
cls = peer_schemes[scheme] | ||||
r50645 | cls.make_peer # make sure we load the module | |||
r50644 | elif scheme in repo_schemes: | |||
cls = repo_schemes[scheme] | ||||
r50645 | cls.instance # make sure we load the module | |||
r50644 | else: | |||
cls = LocalFactory | ||||
r51821 | if hasattr(cls, 'islocal'): | |||
r50583 | return cls.islocal(repo) # pytype: disable=module-attr | |||
return False | ||||
r50582 | repo.ui.deprecwarn(b"use obj.local() instead of islocal(obj)", b"6.4") | |||
Matt Mackall
|
r14605 | return repo.local() | ||
Augie Fackler
|
r43346 | |||
timeless
|
r42278 | def openpath(ui, path, sendaccept=True): | ||
Siddharth Agarwal
|
r17887 | '''open path with open if local, url.open if remote''' | ||
r47669 | pathurl = urlutil.url(path, parsequery=False, parsefragment=False) | |||
Siddharth Agarwal
|
r20354 | if pathurl.islocal(): | ||
Augie Fackler
|
r43347 | return util.posixfile(pathurl.localpath(), b'rb') | ||
Siddharth Agarwal
|
r17887 | else: | ||
timeless
|
r42278 | return url.open(ui, path, sendaccept=sendaccept) | ||
Siddharth Agarwal
|
r17887 | |||
Augie Fackler
|
r43346 | |||
FUJIWARA Katsunori
|
r20858 | # a list of (ui, repo) functions called for wire peer initialization | ||
wirepeersetupfuncs = [] | ||||
Augie Fackler
|
r43346 | |||
r50579 | def _setup_repo_or_peer(ui, obj, presetupfuncs=None): | |||
Sune Foldager
|
r17191 | ui = getattr(obj, "ui", ui) | ||
Jun Wu
|
r32379 | for f in presetupfuncs or []: | ||
f(ui, obj) | ||||
Yuya Nishihara
|
r41032 | ui.log(b'extension', b'- executing reposetup hooks\n') | ||
Augie Fackler
|
r43532 | with util.timedcm('all reposetup') as allreposetupstats: | ||
Boris Feld
|
r39546 | for name, module in extensions.extensions(ui): | ||
Yuya Nishihara
|
r41032 | ui.log(b'extension', b' - running reposetup for %s\n', name) | ||
Boris Feld
|
r39546 | hook = getattr(module, 'reposetup', None) | ||
if hook: | ||||
Augie Fackler
|
r43532 | with util.timedcm('reposetup %r', name) as stats: | ||
Boris Feld
|
r39546 | hook(ui, obj) | ||
r50579 | msg = b' > reposetup for %s took %s\n' | |||
ui.log(b'extension', msg, name, stats) | ||||
Yuya Nishihara
|
r41032 | ui.log(b'extension', b'> all reposetup took %s\n', allreposetupstats) | ||
FUJIWARA Katsunori
|
r20858 | if not obj.local(): | ||
for f in wirepeersetupfuncs: | ||||
f(ui, obj) | ||||
Sune Foldager
|
r17191 | |||
Augie Fackler
|
r43346 | |||
def repository( | ||||
Augie Fackler
|
r43347 | ui, | ||
path=b'', | ||||
create=False, | ||||
presetupfuncs=None, | ||||
intents=None, | ||||
createopts=None, | ||||
Augie Fackler
|
r43346 | ): | ||
Sune Foldager
|
r17191 | """return a repository object for the specified path""" | ||
r50588 | scheme = urlutil.url(path).scheme | |||
if scheme is None: | ||||
scheme = b'file' | ||||
cls = repo_schemes.get(scheme) | ||||
if cls is None: | ||||
if scheme in peer_schemes: | ||||
raise error.Abort(_(b"repository '%s' is not local") % path) | ||||
cls = LocalFactory | ||||
repo = cls.instance( | ||||
Augie Fackler
|
r43346 | ui, | ||
path, | ||||
create, | ||||
intents=intents, | ||||
createopts=createopts, | ||||
) | ||||
r50588 | _setup_repo_or_peer(ui, repo, presetupfuncs=presetupfuncs) | |||
Augie Fackler
|
r43347 | return repo.filtered(b'visible') | ||
Matt Mackall
|
r14605 | |||
Augie Fackler
|
r43346 | |||
Manuel Jacob
|
r51309 | def peer( | ||
uiorrepo, | ||||
opts, | ||||
path, | ||||
create=False, | ||||
intents=None, | ||||
createopts=None, | ||||
remotehidden=False, | ||||
): | ||||
Matt Mackall
|
r14554 | '''return a repository peer for the specified path''' | ||
r50649 | ui = getattr(uiorrepo, 'ui', uiorrepo) | |||
Idan Kamara
|
r14839 | rui = remoteui(uiorrepo, opts) | ||
r51821 | if hasattr(path, 'url'): | |||
r50649 | # this is already a urlutil.path object | |||
peer_path = path | ||||
r50602 | else: | |||
r50649 | peer_path = urlutil.path(ui, None, rawloc=path, validate_path=False) | |||
scheme = peer_path.url.scheme # pytype: disable=attribute-error | ||||
r50587 | if scheme in peer_schemes: | |||
cls = peer_schemes[scheme] | ||||
r50645 | peer = cls.make_peer( | |||
r50587 | rui, | |||
r50652 | peer_path, | |||
r50587 | create, | |||
intents=intents, | ||||
createopts=createopts, | ||||
Manuel Jacob
|
r51309 | remotehidden=remotehidden, | ||
r50587 | ) | |||
_setup_repo_or_peer(rui, peer) | ||||
else: | ||||
# this is a repository | ||||
r50649 | repo_path = peer_path.loc # pytype: disable=attribute-error | |||
if not repo_path: | ||||
repo_path = peer_path.rawloc # pytype: disable=attribute-error | ||||
r50587 | repo = repository( | |||
rui, | ||||
r50649 | repo_path, | |||
r50587 | create, | |||
intents=intents, | ||||
createopts=createopts, | ||||
) | ||||
Manuel Jacob
|
r51309 | peer = repo.peer(path=peer_path, remotehidden=remotehidden) | ||
r50587 | return peer | |||
Augie Fackler
|
r43346 | |||
Matt Mackall
|
r14554 | |||
Vadim Gelfer
|
r2719 | def defaultdest(source): | ||
Augie Fackler
|
r46554 | """return default destination of clone if none is given | ||
Yuya Nishihara
|
r20799 | |||
Yuya Nishihara
|
r34133 | >>> defaultdest(b'foo') | ||
Yuya Nishihara
|
r20799 | 'foo' | ||
Yuya Nishihara
|
r34133 | >>> defaultdest(b'/foo/bar') | ||
Yuya Nishihara
|
r20799 | 'bar' | ||
Yuya Nishihara
|
r34133 | >>> defaultdest(b'/') | ||
Yuya Nishihara
|
r20799 | '' | ||
Yuya Nishihara
|
r34133 | >>> defaultdest(b'') | ||
Yuya Nishihara
|
r20800 | '' | ||
Yuya Nishihara
|
r34133 | >>> defaultdest(b'http://example.org/') | ||
Yuya Nishihara
|
r20800 | '' | ||
Yuya Nishihara
|
r34133 | >>> defaultdest(b'http://example.org/foo/') | ||
Yuya Nishihara
|
r20799 | 'foo' | ||
Augie Fackler
|
r46554 | """ | ||
r47669 | path = urlutil.url(source).path | |||
Yuya Nishihara
|
r20800 | if not path: | ||
Augie Fackler
|
r43347 | return b'' | ||
Yuya Nishihara
|
r20800 | return os.path.basename(os.path.normpath(path)) | ||
Matt Mackall
|
r2774 | |||
Augie Fackler
|
r43346 | |||
Gregory Szorc
|
r36177 | def sharedreposource(repo): | ||
"""Returns repository object for source repository of a shared repo. | ||||
If repo is not a shared repository, returns None. | ||||
""" | ||||
if repo.sharedpath == repo.path: | ||||
return None | ||||
r51821 | if hasattr(repo, 'srcrepo') and repo.srcrepo: | |||
Gregory Szorc
|
r36177 | return repo.srcrepo | ||
# the sharedpath always ends in the .hg; we want the path to the repo | ||||
source = repo.vfs.split(repo.sharedpath)[0] | ||||
r47670 | srcurl, branches = urlutil.parseurl(source) | |||
Gregory Szorc
|
r36177 | srcrepo = repository(repo.ui, srcurl) | ||
repo.srcrepo = srcrepo | ||||
return srcrepo | ||||
Augie Fackler
|
r43346 | |||
def share( | ||||
ui, | ||||
source, | ||||
dest=None, | ||||
update=True, | ||||
bookmarks=True, | ||||
defaultpath=None, | ||||
relative=False, | ||||
): | ||||
Matt Mackall
|
r8800 | '''create a shared repository''' | ||
r50581 | not_local_msg = _(b'can only share local repositories') | |||
r51821 | if hasattr(source, 'local'): | |||
r50581 | if source.local() is None: | |||
raise error.Abort(not_local_msg) | ||||
elif not islocal(source): | ||||
# XXX why are we getting bytes here ? | ||||
raise error.Abort(not_local_msg) | ||||
Matt Mackall
|
r8800 | |||
Matt Mackall
|
r8807 | if not dest: | ||
Brendan Cully
|
r10099 | dest = defaultdest(source) | ||
Matt Mackall
|
r9344 | else: | ||
r50639 | dest = urlutil.get_clone_path_obj(ui, dest).loc | |||
Matt Mackall
|
r8807 | |||
Gregory Szorc
|
r36066 | if isinstance(source, bytes): | ||
r50639 | source_path = urlutil.get_clone_path_obj(ui, source) | |||
srcrepo = repository(ui, source_path.loc) | ||||
branches = (source_path.branch, []) | ||||
Sune Foldager
|
r10365 | rev, checkout = addbranchrevs(srcrepo, srcrepo, branches, None) | ||
Matt Mackall
|
r8800 | else: | ||
Sune Foldager
|
r17191 | srcrepo = source.local() | ||
Matt Mackall
|
r8800 | checkout = None | ||
Gregory Szorc
|
r39885 | shareditems = set() | ||
if bookmarks: | ||||
shareditems.add(sharedbookmarks) | ||||
Augie Fackler
|
r43346 | r = repository( | ||
ui, | ||||
dest, | ||||
create=True, | ||||
createopts={ | ||||
Augie Fackler
|
r43347 | b'sharedrepo': srcrepo, | ||
b'sharedrelative': relative, | ||||
b'shareditems': shareditems, | ||||
Augie Fackler
|
r43346 | }, | ||
) | ||||
Matt Mackall
|
r8800 | |||
Gregory Szorc
|
r39885 | postshare(srcrepo, r, defaultpath=defaultpath) | ||
Martin von Zweigbergk
|
r40609 | r = repository(ui, dest) | ||
Gregory Szorc
|
r28632 | _postshareupdate(r, update, checkout=checkout) | ||
Matt Harbison
|
r34816 | return r | ||
Matt Mackall
|
r8800 | |||
Augie Fackler
|
r43346 | |||
Pulkit Goyal
|
r46057 | def _prependsourcehgrc(repo): | ||
Augie Fackler
|
r46554 | """copies the source repo config and prepend it in current repo .hg/hgrc | ||
Pulkit Goyal
|
r46057 | on unshare. This is only done if the share was perfomed using share safe | ||
method where we share config of source in shares""" | ||||
srcvfs = vfsmod.vfs(repo.sharedpath) | ||||
dstvfs = vfsmod.vfs(repo.path) | ||||
if not srcvfs.exists(b'hgrc'): | ||||
return | ||||
currentconfig = b'' | ||||
if dstvfs.exists(b'hgrc'): | ||||
currentconfig = dstvfs.read(b'hgrc') | ||||
with dstvfs(b'hgrc', b'wb') as fp: | ||||
sourceconfig = srcvfs.read(b'hgrc') | ||||
fp.write(b"# Config copied from shared source\n") | ||||
fp.write(sourceconfig) | ||||
fp.write(b'\n') | ||||
fp.write(currentconfig) | ||||
Matt Harbison
|
r34879 | def unshare(ui, repo): | ||
"""convert a shared repository to a normal one | ||||
Copy the store data to the repo and remove the sharedpath data. | ||||
Gregory Szorc
|
r39642 | |||
Returns a new repository object representing the unshared repository. | ||||
The passed repository object is not usable after this function is | ||||
called. | ||||
Matt Harbison
|
r34879 | """ | ||
Martin von Zweigbergk
|
r41436 | with repo.lock(): | ||
Matt Harbison
|
r34879 | # we use locks here because if we race with commit, we | ||
# can end up with extra data in the cloned revlogs that's | ||||
# not pointed to by changesets, thus causing verify to | ||||
# fail | ||||
destlock = copystore(ui, repo, repo.path) | ||||
Martin von Zweigbergk
|
r41436 | with destlock or util.nullcontextmanager(): | ||
Pulkit Goyal
|
r46057 | if requirements.SHARESAFE_REQUIREMENT in repo.requirements: | ||
# we were sharing .hg/hgrc of the share source with the current | ||||
# repo. We need to copy that while unsharing otherwise it can | ||||
# disable hooks and other checks | ||||
_prependsourcehgrc(repo) | ||||
Matt Harbison
|
r34879 | |||
Augie Fackler
|
r43347 | sharefile = repo.vfs.join(b'sharedpath') | ||
util.rename(sharefile, sharefile + b'.old') | ||||
Martin von Zweigbergk
|
r41436 | |||
Pulkit Goyal
|
r45946 | repo.requirements.discard(requirements.SHARED_REQUIREMENT) | ||
repo.requirements.discard(requirements.RELATIVE_SHARED_REQUIREMENT) | ||||
Pulkit Goyal
|
r45666 | scmutil.writereporequirements(repo) | ||
Matt Harbison
|
r34879 | |||
Gregory Szorc
|
r39642 | # Removing share changes some fundamental properties of the repo instance. | ||
# So we instantiate a new repo object and operate on it rather than | ||||
# try to keep the existing repo usable. | ||||
newrepo = repository(repo.baseui, repo.root, create=False) | ||||
Matt Harbison
|
r34879 | |||
Matt Harbison
|
r34880 | # TODO: figure out how to access subrepos that exist, but were previously | ||
# removed from .hgsub | ||||
Augie Fackler
|
r43347 | c = newrepo[b'.'] | ||
Matt Harbison
|
r34880 | subs = c.substate | ||
for s in sorted(subs): | ||||
c.sub(s).unshare() | ||||
Gregory Szorc
|
r39642 | localrepo.poisonrepository(repo) | ||
return newrepo | ||||
Augie Fackler
|
r43346 | |||
Gregory Szorc
|
r39885 | def postshare(sourcerepo, destrepo, defaultpath=None): | ||
Gregory Szorc
|
r27354 | """Called after a new shared repo is created. | ||
The new repo only has a requirements file and pointer to the source. | ||||
This function configures additional shared data. | ||||
Extensions can wrap this function and write additional entries to | ||||
destrepo/.hg/shared to indicate additional pieces of data to be shared. | ||||
""" | ||||
Augie Fackler
|
r43347 | default = defaultpath or sourcerepo.ui.config(b'paths', b'default') | ||
Gregory Szorc
|
r27354 | if default: | ||
Martin von Zweigbergk
|
r43387 | template = b'[paths]\ndefault = %s\n' | ||
Augie Fackler
|
r43347 | destrepo.vfs.write(b'hgrc', util.tonativeeol(template % default)) | ||
Pulkit Goyal
|
r45932 | if requirements.NARROW_REQUIREMENT in sourcerepo.requirements: | ||
r51086 | with destrepo.wlock(), destrepo.lock(), destrepo.transaction( | |||
b"narrow-share" | ||||
): | ||||
Martin von Zweigbergk
|
r41265 | narrowspec.copytoworkingcopy(destrepo) | ||
Gregory Szorc
|
r27354 | |||
Augie Fackler
|
r43346 | |||
Gregory Szorc
|
r28632 | def _postshareupdate(repo, update, checkout=None): | ||
"""Maybe perform a working directory update after a shared repo is created. | ||||
``update`` can be a boolean or a revision to update to. | ||||
""" | ||||
if not update: | ||||
return | ||||
Augie Fackler
|
r43347 | repo.ui.status(_(b"updating working directory\n")) | ||
Gregory Szorc
|
r28632 | if update is not True: | ||
checkout = update | ||||
Augie Fackler
|
r43347 | for test in (checkout, b'default', b'tip'): | ||
Gregory Szorc
|
r28632 | if test is None: | ||
continue | ||||
try: | ||||
uprev = repo.lookup(test) | ||||
break | ||||
except error.RepoLookupError: | ||||
continue | ||||
_update(repo, uprev) | ||||
Augie Fackler
|
r43346 | |||
Simon Heimberg
|
r15078 | def copystore(ui, srcrepo, destpath): | ||
Augie Fackler
|
r46554 | """copy files from store of srcrepo in destpath | ||
Simon Heimberg
|
r15078 | |||
returns destlock | ||||
Augie Fackler
|
r46554 | """ | ||
Simon Heimberg
|
r15078 | destlock = None | ||
try: | ||||
hardlink = None | ||||
Augie Fackler
|
r43347 | topic = _(b'linking') if hardlink else _(b'copying') | ||
with ui.makeprogress(topic, unit=_(b'files')) as progress: | ||||
Matt Harbison
|
r39425 | num = 0 | ||
srcpublishing = srcrepo.publishing() | ||||
srcvfs = vfsmod.vfs(srcrepo.sharedpath) | ||||
dstvfs = vfsmod.vfs(destpath) | ||||
for f in srcrepo.store.copylist(): | ||||
Augie Fackler
|
r43347 | if srcpublishing and f.endswith(b'phaseroots'): | ||
Matt Harbison
|
r39425 | continue | ||
dstbase = os.path.dirname(f) | ||||
if dstbase and not dstvfs.exists(dstbase): | ||||
dstvfs.mkdir(dstbase) | ||||
if srcvfs.exists(f): | ||||
Augie Fackler
|
r43347 | if f.endswith(b'data'): | ||
Matt Harbison
|
r39425 | # 'dstbase' may be empty (e.g. revlog format 0) | ||
Augie Fackler
|
r43347 | lockfile = os.path.join(dstbase, b"lock") | ||
Matt Harbison
|
r39425 | # lock to avoid premature writing to the target | ||
destlock = lock.lock(dstvfs, lockfile) | ||||
Augie Fackler
|
r43346 | hardlink, n = util.copyfiles( | ||
srcvfs.join(f), dstvfs.join(f), hardlink, progress | ||||
) | ||||
Matt Harbison
|
r39425 | num += n | ||
if hardlink: | ||||
Augie Fackler
|
r43347 | ui.debug(b"linked %d files\n" % num) | ||
Matt Harbison
|
r39425 | else: | ||
Augie Fackler
|
r43347 | ui.debug(b"copied %d files\n" % num) | ||
Simon Heimberg
|
r15078 | return destlock | ||
Augie Fackler
|
r43346 | except: # re-raises | ||
Simon Heimberg
|
r15078 | release(destlock) | ||
raise | ||||
Augie Fackler
|
r43346 | |||
def clonewithshare( | ||||
ui, | ||||
peeropts, | ||||
sharepath, | ||||
source, | ||||
srcpeer, | ||||
dest, | ||||
pull=False, | ||||
rev=None, | ||||
update=True, | ||||
stream=False, | ||||
): | ||||
Gregory Szorc
|
r25761 | """Perform a clone using a shared repo. | ||
The store for the repository will be located at <sharepath>/.hg. The | ||||
specified revisions will be cloned or pulled from "source". A shared repo | ||||
will be created at "dest" and a working copy will be created if "update" is | ||||
True. | ||||
""" | ||||
revs = None | ||||
if rev: | ||||
Augie Fackler
|
r43347 | if not srcpeer.capable(b'lookup'): | ||
Augie Fackler
|
r43346 | raise error.Abort( | ||
_( | ||||
Augie Fackler
|
r43347 | b"src repository does not support " | ||
b"revision lookup and so doesn't " | ||||
b"support clone by revision" | ||||
Augie Fackler
|
r43346 | ) | ||
) | ||||
Gregory Szorc
|
r37658 | |||
# TODO this is batchable. | ||||
remoterevs = [] | ||||
for r in rev: | ||||
with srcpeer.commandexecutor() as e: | ||||
Augie Fackler
|
r43347 | remoterevs.append( | ||
Augie Fackler
|
r46554 | e.callcommand( | ||
b'lookup', | ||||
{ | ||||
b'key': r, | ||||
}, | ||||
).result() | ||||
Augie Fackler
|
r43347 | ) | ||
Gregory Szorc
|
r37658 | revs = remoterevs | ||
Gregory Szorc
|
r25761 | |||
Gregory Szorc
|
r28289 | # Obtain a lock before checking for or cloning the pooled repo otherwise | ||
# 2 clients may race creating or populating it. | ||||
pooldir = os.path.dirname(sharepath) | ||||
# lock class requires the directory to exist. | ||||
try: | ||||
util.makedir(pooldir, False) | ||||
Manuel Jacob
|
r50200 | except FileExistsError: | ||
pass | ||||
Gregory Szorc
|
r28289 | |||
Pierre-Yves David
|
r31218 | poolvfs = vfsmod.vfs(pooldir) | ||
Gregory Szorc
|
r25761 | basename = os.path.basename(sharepath) | ||
Augie Fackler
|
r43347 | with lock.lock(poolvfs, b'%s.lock' % basename): | ||
Gregory Szorc
|
r28289 | if os.path.exists(sharepath): | ||
Augie Fackler
|
r43346 | ui.status( | ||
Augie Fackler
|
r43347 | _(b'(sharing from existing pooled repository %s)\n') % basename | ||
Augie Fackler
|
r43346 | ) | ||
Gregory Szorc
|
r28289 | else: | ||
Augie Fackler
|
r43347 | ui.status( | ||
_(b'(sharing from new pooled repository %s)\n') % basename | ||||
) | ||||
Gregory Szorc
|
r28289 | # Always use pull mode because hardlinks in share mode don't work | ||
# well. Never update because working copies aren't necessary in | ||||
# share mode. | ||||
Augie Fackler
|
r43346 | clone( | ||
ui, | ||||
peeropts, | ||||
source, | ||||
dest=sharepath, | ||||
pull=True, | ||||
revs=rev, | ||||
update=False, | ||||
stream=stream, | ||||
) | ||||
Gregory Szorc
|
r25761 | |||
Gregory Szorc
|
r30041 | # Resolve the value to put in [paths] section for the source. | ||
if islocal(source): | ||||
r48426 | defaultpath = util.abspath(urlutil.urllocalpath(source)) | |||
Gregory Szorc
|
r30041 | else: | ||
defaultpath = source | ||||
Gregory Szorc
|
r25761 | sharerepo = repository(ui, path=sharepath) | ||
Augie Fackler
|
r43346 | destrepo = share( | ||
ui, | ||||
sharerepo, | ||||
dest=dest, | ||||
update=False, | ||||
bookmarks=False, | ||||
defaultpath=defaultpath, | ||||
) | ||||
Gregory Szorc
|
r25761 | |||
# We need to perform a pull against the dest repo to fetch bookmarks | ||||
# and other non-store data that isn't shared by default. In the case of | ||||
# non-existing shared repo, this means we pull from the remote twice. This | ||||
# is a bit weird. But at the time it was implemented, there wasn't an easy | ||||
# way to pull just non-changegroup data. | ||||
exchange.pull(destrepo, srcpeer, heads=revs) | ||||
Gregory Szorc
|
r28632 | _postshareupdate(destrepo, update) | ||
Gregory Szorc
|
r25761 | return srcpeer, peer(ui, peeropts, dest) | ||
Augie Fackler
|
r43346 | |||
Joerg Sonnenberger
|
r46739 | # Recomputing caches is often slow on big repos, so copy them. | ||
r32492 | def _copycache(srcrepo, dstcachedir, fname): | |||
"""copy a cache from srcrepo to destcachedir (if it exists)""" | ||||
Joerg Sonnenberger
|
r46739 | srcfname = srcrepo.cachevfs.join(fname) | ||
dstfname = os.path.join(dstcachedir, fname) | ||||
if os.path.exists(srcfname): | ||||
r32492 | if not os.path.exists(dstcachedir): | |||
os.mkdir(dstcachedir) | ||||
Joerg Sonnenberger
|
r46739 | util.copyfile(srcfname, dstfname) | ||
r32492 | ||||
Augie Fackler
|
r43346 | |||
def clone( | ||||
ui, | ||||
peeropts, | ||||
source, | ||||
dest=None, | ||||
pull=False, | ||||
revs=None, | ||||
update=True, | ||||
stream=False, | ||||
branch=None, | ||||
shareopts=None, | ||||
storeincludepats=None, | ||||
storeexcludepats=None, | ||||
depth=None, | ||||
): | ||||
Vadim Gelfer
|
r2597 | """Make a copy of an existing repository. | ||
Create a copy of an existing repository in a new directory. The | ||||
source and destination are URLs, as passed to the repository | ||||
Sune Foldager
|
r17191 | function. Returns a pair of repository peers, the source and | ||
Vadim Gelfer
|
r2597 | newly created destination. | ||
The location of the source is added to the new repository's | ||||
.hg/hgrc file, as the default to be used for future pulls and | ||||
pushes. | ||||
If an exception is raised, the partly cloned/updated destination | ||||
repository will be deleted. | ||||
Vadim Gelfer
|
r2600 | |||
Vadim Gelfer
|
r2719 | Arguments: | ||
source: repository object or URL | ||||
Vadim Gelfer
|
r2597 | |||
dest: URL of destination repository to create (defaults to base | ||||
name of source repository) | ||||
Siddharth Agarwal
|
r23545 | pull: always pull from source repository, even in local case or if the | ||
server prefers streaming | ||||
Vadim Gelfer
|
r2597 | |||
Vadim Gelfer
|
r2621 | stream: stream raw data uncompressed from repository (fast over | ||
LAN, slow over WAN) | ||||
Vadim Gelfer
|
r2613 | |||
Martin von Zweigbergk
|
r37279 | revs: revision to clone up to (implies pull=True) | ||
Vadim Gelfer
|
r2597 | |||
update: update working directory after clone completes, if | ||||
Bryan O'Sullivan
|
r6526 | destination is local repository (True means update to default rev, | ||
anything else is treated as a revision) | ||||
Sune Foldager
|
r10379 | |||
branch: branches to clone | ||||
Gregory Szorc
|
r25761 | |||
shareopts: dict of options to control auto sharing behavior. The "pool" key | ||||
activates auto sharing mode and defines the directory for stores. The | ||||
"mode" key determines how to construct the directory name of the shared | ||||
repository. "identity" means the name is derived from the node of the first | ||||
changeset in the repository. "remote" means the name is derived from the | ||||
remote's path/URL. Defaults to "identity." | ||||
Gregory Szorc
|
r39586 | |||
storeincludepats and storeexcludepats: sets of file patterns to include and | ||||
exclude in the repository copy, respectively. If not defined, all files | ||||
will be included (a "full" clone). Otherwise a "narrow" clone containing | ||||
only the requested files will be performed. If ``storeincludepats`` is not | ||||
defined but ``storeexcludepats`` is, ``storeincludepats`` is assumed to be | ||||
``path:.``. If both are empty sets, no files will be cloned. | ||||
Vadim Gelfer
|
r2597 | """ | ||
Matt Mackall
|
r4478 | |||
Pulkit Goyal
|
r32970 | if isinstance(source, bytes): | ||
r50640 | src_path = urlutil.get_clone_path_obj(ui, source) | |||
if src_path is None: | ||||
srcpeer = peer(ui, peeropts, b'') | ||||
origsource = source = b'' | ||||
branches = (None, branch or []) | ||||
else: | ||||
srcpeer = peer(ui, peeropts, src_path) | ||||
origsource = src_path.rawloc | ||||
branches = (src_path.branch, branch or []) | ||||
source = src_path.loc | ||||
Vadim Gelfer
|
r2719 | else: | ||
r51821 | if hasattr(source, 'peer'): | |||
r50642 | srcpeer = source.peer() # in case we were called with a localrepo | |||
else: | ||||
srcpeer = source | ||||
Martin von Zweigbergk
|
r37278 | branches = (None, branch or []) | ||
r50640 | # XXX path: simply use the peer `path` object when this become available | |||
Sune Foldager
|
r17191 | origsource = source = srcpeer.url() | ||
Arseniy Alekseyev
|
r49326 | srclock = destlock = destwlock = cleandir = None | ||
Valentin Gatien-Baron
|
r47419 | destpeer = None | ||
try: | ||||
revs, checkout = addbranchrevs(srcpeer, srcpeer, branches, revs) | ||||
Vadim Gelfer
|
r2719 | |||
Valentin Gatien-Baron
|
r47419 | if dest is None: | ||
dest = defaultdest(source) | ||||
if dest: | ||||
ui.status(_(b"destination directory: %s\n") % dest) | ||||
else: | ||||
r50640 | dest_path = urlutil.get_clone_path_obj(ui, dest) | |||
if dest_path is not None: | ||||
dest = dest_path.rawloc | ||||
else: | ||||
dest = b'' | ||||
Vadim Gelfer
|
r2719 | |||
r47669 | dest = urlutil.urllocalpath(dest) | |||
source = urlutil.urllocalpath(source) | ||||
Vadim Gelfer
|
r2597 | |||
Valentin Gatien-Baron
|
r47419 | if not dest: | ||
raise error.InputError(_(b"empty destination path is not valid")) | ||||
Chinmay Joshi
|
r21803 | |||
Valentin Gatien-Baron
|
r47419 | destvfs = vfsmod.vfs(dest, expandpath=True) | ||
if destvfs.lexists(): | ||||
if not destvfs.isdir(): | ||||
raise error.InputError( | ||||
_(b"destination '%s' already exists") % dest | ||||
) | ||||
elif destvfs.listdir(): | ||||
raise error.InputError( | ||||
_(b"destination '%s' is not empty") % dest | ||||
) | ||||
Vadim Gelfer
|
r2597 | |||
Valentin Gatien-Baron
|
r47419 | createopts = {} | ||
narrow = False | ||||
Gregory Szorc
|
r39586 | |||
Valentin Gatien-Baron
|
r47419 | if storeincludepats is not None: | ||
narrowspec.validatepatterns(storeincludepats) | ||||
narrow = True | ||||
if storeexcludepats is not None: | ||||
narrowspec.validatepatterns(storeexcludepats) | ||||
narrow = True | ||||
Gregory Szorc
|
r39586 | |||
Valentin Gatien-Baron
|
r47419 | if narrow: | ||
# Include everything by default if only exclusion patterns defined. | ||||
if storeexcludepats and not storeincludepats: | ||||
storeincludepats = {b'path:.'} | ||||
Gregory Szorc
|
r39586 | |||
Valentin Gatien-Baron
|
r47419 | createopts[b'narrowfiles'] = True | ||
Gregory Szorc
|
r39586 | |||
Valentin Gatien-Baron
|
r47419 | if depth: | ||
createopts[b'shallowfilestore'] = True | ||||
Gregory Szorc
|
r40426 | |||
Valentin Gatien-Baron
|
r47419 | if srcpeer.capable(b'lfs-serve'): | ||
# Repository creation honors the config if it disabled the extension, so | ||||
# we can't just announce that lfs will be enabled. This check avoids | ||||
# saying that lfs will be enabled, and then saying it's an unknown | ||||
# feature. The lfs creation option is set in either case so that a | ||||
# requirement is added. If the extension is explicitly disabled but the | ||||
# requirement is set, the clone aborts early, before transferring any | ||||
# data. | ||||
createopts[b'lfs'] = True | ||||
Matt Harbison
|
r40360 | |||
Matt Harbison
|
r50670 | if b'lfs' in extensions.disabled(): | ||
Valentin Gatien-Baron
|
r47419 | ui.status( | ||
_( | ||||
b'(remote is using large file support (lfs), but it is ' | ||||
b'explicitly disabled in the local configuration)\n' | ||||
) | ||||
Augie Fackler
|
r43346 | ) | ||
Valentin Gatien-Baron
|
r47419 | else: | ||
ui.status( | ||||
_( | ||||
b'(remote is using large file support (lfs); lfs will ' | ||||
b'be enabled for this repository)\n' | ||||
) | ||||
Augie Fackler
|
r43346 | ) | ||
Matt Harbison
|
r40360 | |||
Valentin Gatien-Baron
|
r47419 | shareopts = shareopts or {} | ||
sharepool = shareopts.get(b'pool') | ||||
sharenamemode = shareopts.get(b'mode') | ||||
if sharepool and islocal(dest): | ||||
sharepath = None | ||||
if sharenamemode == b'identity': | ||||
# Resolve the name from the initial changeset in the remote | ||||
# repository. This returns nullid when the remote is empty. It | ||||
# raises RepoLookupError if revision 0 is filtered or otherwise | ||||
# not available. If we fail to resolve, sharing is not enabled. | ||||
try: | ||||
with srcpeer.commandexecutor() as e: | ||||
rootnode = e.callcommand( | ||||
b'lookup', | ||||
{ | ||||
b'key': b'0', | ||||
}, | ||||
).result() | ||||
Gregory Szorc
|
r37658 | |||
Joerg Sonnenberger
|
r47771 | if rootnode != sha1nodeconstants.nullid: | ||
Valentin Gatien-Baron
|
r47419 | sharepath = os.path.join(sharepool, hex(rootnode)) | ||
else: | ||||
ui.status( | ||||
_( | ||||
b'(not using pooled storage: ' | ||||
b'remote appears to be empty)\n' | ||||
) | ||||
) | ||||
except error.RepoLookupError: | ||||
Augie Fackler
|
r43346 | ui.status( | ||
_( | ||||
Augie Fackler
|
r43347 | b'(not using pooled storage: ' | ||
Valentin Gatien-Baron
|
r47419 | b'unable to resolve identity of remote)\n' | ||
Augie Fackler
|
r43346 | ) | ||
) | ||||
Valentin Gatien-Baron
|
r47419 | elif sharenamemode == b'remote': | ||
sharepath = os.path.join( | ||||
sharepool, hex(hashutil.sha1(source).digest()) | ||||
) | ||||
else: | ||||
raise error.Abort( | ||||
_(b'unknown share naming mode: %s') % sharenamemode | ||||
Augie Fackler
|
r43346 | ) | ||
Valentin Gatien-Baron
|
r47419 | |||
# TODO this is a somewhat arbitrary restriction. | ||||
if narrow: | ||||
ui.status( | ||||
_(b'(pooled storage not supported for narrow clones)\n') | ||||
) | ||||
sharepath = None | ||||
Gregory Szorc
|
r25761 | |||
Valentin Gatien-Baron
|
r47419 | if sharepath: | ||
return clonewithshare( | ||||
ui, | ||||
peeropts, | ||||
sharepath, | ||||
source, | ||||
srcpeer, | ||||
dest, | ||||
pull=pull, | ||||
rev=revs, | ||||
update=update, | ||||
stream=stream, | ||||
) | ||||
Gregory Szorc
|
r39586 | |||
Valentin Gatien-Baron
|
r47419 | srcrepo = srcpeer.local() | ||
Gregory Szorc
|
r25761 | |||
Brendan Cully
|
r14377 | abspath = origsource | ||
if islocal(origsource): | ||||
r48426 | abspath = util.abspath(urlutil.urllocalpath(origsource)) | |||
Brendan Cully
|
r14377 | |||
Matt Mackall
|
r4915 | if islocal(dest): | ||
r48209 | if os.path.exists(dest): | |||
# only clean up directories we create ourselves | ||||
hgdir = os.path.realpath(os.path.join(dest, b".hg")) | ||||
cleandir = hgdir | ||||
else: | ||||
cleandir = dest | ||||
Vadim Gelfer
|
r2597 | |||
Matt Mackall
|
r4915 | copy = False | ||
Augie Fackler
|
r43346 | if ( | ||
srcrepo | ||||
and srcrepo.cancopy() | ||||
and islocal(dest) | ||||
and not phases.hassecret(srcrepo) | ||||
): | ||||
Martin von Zweigbergk
|
r37279 | copy = not pull and not revs | ||
Vadim Gelfer
|
r2597 | |||
Gregory Szorc
|
r39586 | # TODO this is a somewhat arbitrary restriction. | ||
if narrow: | ||||
copy = False | ||||
Matt Mackall
|
r4915 | if copy: | ||
try: | ||||
# we use a lock here because if we race with commit, we | ||||
# can end up with extra data in the cloned revlogs that's | ||||
# not pointed to by changesets, thus causing verify to | ||||
# fail | ||||
Martin Geisler
|
r14463 | srclock = srcrepo.lock(wait=False) | ||
Matt Mackall
|
r7640 | except error.LockError: | ||
Matt Mackall
|
r4915 | copy = False | ||
Vadim Gelfer
|
r2597 | |||
Matt Mackall
|
r4915 | if copy: | ||
Augie Fackler
|
r43347 | srcrepo.hook(b'preoutgoing', throw=True, source=b'clone') | ||
Vadim Gelfer
|
r2597 | |||
r48235 | destrootpath = urlutil.urllocalpath(dest) | |||
dest_reqs = localrepo.clone_requirements(ui, createopts, srcrepo) | ||||
localrepo.createrepository( | ||||
ui, | ||||
destrootpath, | ||||
requirements=dest_reqs, | ||||
) | ||||
destrepo = localrepo.makelocalrepository(ui, destrootpath) | ||||
Arseniy Alekseyev
|
r49326 | |||
destwlock = destrepo.wlock() | ||||
r48240 | destlock = destrepo.lock() | |||
from . import streamclone # avoid cycle | ||||
r48235 | ||||
r48240 | streamclone.local_copy(srcrepo, destrepo) | |||
Tomasz Kleczek
|
r17740 | |||
Matt Mackall
|
r4915 | # we need to re-init the repo after manually copying the data | ||
# into it | ||||
Simon Heimberg
|
r17874 | destpeer = peer(srcrepo, peeropts, dest) | ||
Arseniy Alekseyev
|
r49326 | |||
# make the peer aware that is it already locked | ||||
# | ||||
# important: | ||||
# | ||||
# We still need to release that lock at the end of the function | ||||
destpeer.local()._lockref = weakref.ref(destlock) | ||||
destpeer.local()._wlockref = weakref.ref(destwlock) | ||||
# dirstate also needs to be copied because `_wlockref` has a reference | ||||
# to it: this dirstate is saved to disk when the wlock is released | ||||
destpeer.local().dirstate = destrepo.dirstate | ||||
Joerg Sonnenberger
|
r47771 | srcrepo.hook( | ||
b'outgoing', source=b'clone', node=srcrepo.nodeconstants.nullhex | ||||
) | ||||
Matt Mackall
|
r4915 | else: | ||
Matt Mackall
|
r5569 | try: | ||
Gregory Szorc
|
r39586 | # only pass ui when no srcrepo | ||
Augie Fackler
|
r43346 | destpeer = peer( | ||
srcrepo or ui, | ||||
peeropts, | ||||
dest, | ||||
create=True, | ||||
createopts=createopts, | ||||
) | ||||
Manuel Jacob
|
r50200 | except FileExistsError: | ||
cleandir = None | ||||
raise error.Abort(_(b"destination '%s' already exists") % dest) | ||||
Vadim Gelfer
|
r2597 | |||
Martin von Zweigbergk
|
r37279 | if revs: | ||
Augie Fackler
|
r43347 | if not srcpeer.capable(b'lookup'): | ||
Augie Fackler
|
r43346 | raise error.Abort( | ||
_( | ||||
Augie Fackler
|
r43347 | b"src repository does not support " | ||
b"revision lookup and so doesn't " | ||||
b"support clone by revision" | ||||
Augie Fackler
|
r43346 | ) | ||
) | ||||
Gregory Szorc
|
r37658 | |||
# TODO this is batchable. | ||||
remoterevs = [] | ||||
for rev in revs: | ||||
with srcpeer.commandexecutor() as e: | ||||
Augie Fackler
|
r43346 | remoterevs.append( | ||
Augie Fackler
|
r46554 | e.callcommand( | ||
b'lookup', | ||||
{ | ||||
b'key': rev, | ||||
}, | ||||
).result() | ||||
Augie Fackler
|
r43346 | ) | ||
Gregory Szorc
|
r37658 | revs = remoterevs | ||
Brett Carter
|
r8417 | checkout = revs[0] | ||
Martin von Zweigbergk
|
r37279 | else: | ||
revs = None | ||||
Augie Fackler
|
r27165 | local = destpeer.local() | ||
if local: | ||||
Gregory Szorc
|
r39591 | if narrow: | ||
r51084 | with local.wlock(), local.lock(), local.transaction( | |||
b'narrow-clone' | ||||
): | ||||
Gregory Szorc
|
r39591 | local.setnarrowpats(storeincludepats, storeexcludepats) | ||
Martin von Zweigbergk
|
r41272 | narrowspec.copytoworkingcopy(local) | ||
Gregory Szorc
|
r39591 | |||
r47669 | u = urlutil.url(abspath) | |||
Boris Feld
|
r35581 | defaulturl = bytes(u) | ||
Augie Fackler
|
r43347 | local.ui.setconfig(b'paths', b'default', defaulturl, b'clone') | ||
Siddharth Agarwal
|
r23545 | if not stream: | ||
if pull: | ||||
stream = False | ||||
else: | ||||
stream = None | ||||
Augie Fackler
|
r27165 | # internal config: ui.quietbookmarkmove | ||
Augie Fackler
|
r43347 | overrides = {(b'ui', b'quietbookmarkmove'): True} | ||
with local.ui.configoverride(overrides, b'clone'): | ||||
Augie Fackler
|
r43346 | exchange.pull( | ||
local, | ||||
srcpeer, | ||||
r49055 | heads=revs, | |||
Augie Fackler
|
r43346 | streamclonerequested=stream, | ||
includepats=storeincludepats, | ||||
excludepats=storeexcludepats, | ||||
depth=depth, | ||||
) | ||||
Sune Foldager
|
r17191 | elif srcrepo: | ||
Gregory Szorc
|
r39586 | # TODO lift restriction once exchange.push() accepts narrow | ||
# push. | ||||
if narrow: | ||||
Augie Fackler
|
r43346 | raise error.Abort( | ||
_( | ||||
Augie Fackler
|
r43347 | b'narrow clone not available for ' | ||
b'remote destinations' | ||||
Augie Fackler
|
r43346 | ) | ||
) | ||||
Gregory Szorc
|
r39586 | |||
Augie Fackler
|
r43346 | exchange.push( | ||
srcrepo, | ||||
destpeer, | ||||
revs=revs, | ||||
bookmarks=srcrepo._bookmarks.keys(), | ||||
) | ||||
Matt Mackall
|
r4915 | else: | ||
Augie Fackler
|
r43346 | raise error.Abort( | ||
Augie Fackler
|
r43347 | _(b"clone from remote to remote not supported") | ||
Augie Fackler
|
r43346 | ) | ||
Vadim Gelfer
|
r2597 | |||
Augie Fackler
|
r18441 | cleandir = None | ||
Vadim Gelfer
|
r2597 | |||
Sune Foldager
|
r17191 | destrepo = destpeer.local() | ||
if destrepo: | ||||
Augie Fackler
|
r43347 | template = uimod.samplehgrcs[b'cloned'] | ||
r47669 | u = urlutil.url(abspath) | |||
Augie Fackler
|
r15552 | u.passwd = None | ||
Yuya Nishihara
|
r33650 | defaulturl = bytes(u) | ||
Augie Fackler
|
r43347 | destrepo.vfs.write(b'hgrc', util.tonativeeol(template % defaulturl)) | ||
destrepo.ui.setconfig(b'paths', b'default', defaulturl, b'clone') | ||||
Matt Mackall
|
r8814 | |||
Augie Fackler
|
r43347 | if ui.configbool(b'experimental', b'remotenames'): | ||
Pulkit Goyal
|
r35348 | logexchange.pullremotenames(destrepo, srcpeer) | ||
Pulkit Goyal
|
r35332 | |||
Matt Mackall
|
r4915 | if update: | ||
Bryan O'Sullivan
|
r6526 | if update is not True: | ||
Gregory Szorc
|
r37658 | with srcpeer.commandexecutor() as e: | ||
Augie Fackler
|
r43346 | checkout = e.callcommand( | ||
Augie Fackler
|
r46554 | b'lookup', | ||
{ | ||||
b'key': update, | ||||
}, | ||||
Augie Fackler
|
r43346 | ).result() | ||
Gregory Szorc
|
r37658 | |||
Thomas Arendsen Hein
|
r17867 | uprev = None | ||
Adrian Buehlmann
|
r17882 | status = None | ||
Thomas Arendsen Hein
|
r17867 | if checkout is not None: | ||
Boris Feld
|
r38776 | # Some extensions (at least hg-git and hg-subversion) have | ||
# a peer.lookup() implementation that returns a name instead | ||||
# of a nodeid. We work around it here until we've figured | ||||
# out a better solution. | ||||
if len(checkout) == 20 and checkout in destrepo: | ||||
Martin von Zweigbergk
|
r37498 | uprev = checkout | ||
Boris Feld
|
r38776 | elif scmutil.isrevsymbol(destrepo, checkout): | ||
uprev = scmutil.revsymbol(destrepo, checkout).node() | ||||
Martin von Zweigbergk
|
r37498 | else: | ||
Sean Farley
|
r26354 | if update is not True: | ||
try: | ||||
uprev = destrepo.lookup(update) | ||||
except error.RepoLookupError: | ||||
pass | ||||
Thomas Arendsen Hein
|
r17867 | if uprev is None: | ||
try: | ||||
Dan Villiom Podlaski Christiansen
|
r46814 | if destrepo._activebookmark: | ||
uprev = destrepo.lookup(destrepo._activebookmark) | ||||
update = destrepo._activebookmark | ||||
else: | ||||
uprev = destrepo._bookmarks[b'@'] | ||||
update = b'@' | ||||
Adrian Buehlmann
|
r17882 | bn = destrepo[uprev].branch() | ||
Augie Fackler
|
r43347 | if bn == b'default': | ||
Dan Villiom Podlaski Christiansen
|
r46814 | status = _(b"updating to bookmark %s\n" % update) | ||
Adrian Buehlmann
|
r17882 | else: | ||
Augie Fackler
|
r43346 | status = ( | ||
Dan Villiom Podlaski Christiansen
|
r46814 | _(b"updating to bookmark %s on branch %s\n") | ||
) % (update, bn) | ||||
Thomas Arendsen Hein
|
r17867 | except KeyError: | ||
try: | ||||
Augie Fackler
|
r43347 | uprev = destrepo.branchtip(b'default') | ||
Thomas Arendsen Hein
|
r17867 | except error.RepoLookupError: | ||
Augie Fackler
|
r43347 | uprev = destrepo.lookup(b'tip') | ||
Adrian Buehlmann
|
r17882 | if not status: | ||
bn = destrepo[uprev].branch() | ||||
Augie Fackler
|
r43347 | status = _(b"updating to branch %s\n") % bn | ||
Adrian Buehlmann
|
r17882 | destrepo.ui.status(status) | ||
Martin Geisler
|
r14463 | _update(destrepo, uprev) | ||
Thomas Arendsen Hein
|
r17703 | if update in destrepo._bookmarks: | ||
Ryan McElroy
|
r24945 | bookmarks.activate(destrepo, update) | ||
r47032 | if destlock is not None: | |||
release(destlock) | ||||
Arseniy Alekseyev
|
r49326 | if destwlock is not None: | ||
release(destlock) | ||||
r47032 | # here is a tiny windows were someone could end up writing the | |||
# repository before the cache are sure to be warm. This is "fine" | ||||
# as the only "bad" outcome would be some slowness. That potential | ||||
# slowness already affect reader. | ||||
with destrepo.lock(): | ||||
r48079 | destrepo.updatecaches(caches=repositorymod.CACHES_POST_CLONE) | |||
Matt Mackall
|
r4915 | finally: | ||
Arseniy Alekseyev
|
r49326 | release(srclock, destlock, destwlock) | ||
Augie Fackler
|
r18441 | if cleandir is not None: | ||
shutil.rmtree(cleandir, True) | ||||
Sune Foldager
|
r17191 | if srcpeer is not None: | ||
srcpeer.close() | ||||
Valentin Gatien-Baron
|
r47419 | if destpeer and destpeer.local() is None: | ||
destpeer.close() | ||||
simon@laptop-tosh
|
r19313 | return srcpeer, destpeer | ||
Matt Mackall
|
r2775 | |||
Augie Fackler
|
r43346 | |||
timeless
|
r27402 | def _showstats(repo, stats, quietempty=False): | ||
Gregory Szorc
|
r37143 | if quietempty and stats.isempty(): | ||
timeless
|
r27402 | return | ||
Augie Fackler
|
r43346 | repo.ui.status( | ||
_( | ||||
Augie Fackler
|
r43347 | b"%d files updated, %d files merged, " | ||
b"%d files removed, %d files unresolved\n" | ||||
Augie Fackler
|
r43346 | ) | ||
% ( | ||||
stats.updatedcount, | ||||
stats.mergedcount, | ||||
stats.removedcount, | ||||
stats.unresolvedcount, | ||||
) | ||||
) | ||||
Matt Mackall
|
r3316 | |||
Martin von Zweigbergk
|
r31166 | def updaterepo(repo, node, overwrite, updatecheck=None): | ||
Simon Heimberg
|
r17895 | """Update the working directory to node. | ||
When overwrite is set, changes are clobbered, merged else | ||||
returns stats (see pydoc mercurial.merge.applyupdates)""" | ||||
Martin von Zweigbergk
|
r46180 | repo.ui.deprecwarn( | ||
b'prefer merge.update() or merge.clean_update() over hg.updaterepo()', | ||||
b'5.7', | ||||
) | ||||
Martin von Zweigbergk
|
r46134 | return mergemod._update( | ||
Augie Fackler
|
r43346 | repo, | ||
node, | ||||
branchmerge=False, | ||||
force=overwrite, | ||||
Augie Fackler
|
r43347 | labels=[b'working copy', b'destination'], | ||
Augie Fackler
|
r43346 | updatecheck=updatecheck, | ||
) | ||||
Simon Heimberg
|
r17895 | |||
Martin von Zweigbergk
|
r31166 | def update(repo, node, quietempty=False, updatecheck=None): | ||
"""update the working directory to node""" | ||||
Martin von Zweigbergk
|
r46151 | stats = mergemod.update(repo[node], updatecheck=updatecheck) | ||
timeless
|
r27404 | _showstats(repo, stats, quietempty) | ||
Gregory Szorc
|
r37143 | if stats.unresolvedcount: | ||
Augie Fackler
|
r43347 | repo.ui.status(_(b"use 'hg resolve' to retry unresolved file merges\n")) | ||
Gregory Szorc
|
r37143 | return stats.unresolvedcount > 0 | ||
Matt Mackall
|
r2775 | |||
Augie Fackler
|
r43346 | |||
Benoit Boissinot
|
r7546 | # naming conflict in clone() | ||
_update = update | ||||
Augie Fackler
|
r43346 | |||
timeless
|
r27403 | def clean(repo, node, show_stats=True, quietempty=False): | ||
Matt Mackall
|
r2808 | """forcibly switch the working directory to node, clobbering changes""" | ||
Martin von Zweigbergk
|
r46133 | stats = mergemod.clean_update(repo[node]) | ||
Martin von Zweigbergk
|
r44620 | assert stats.unresolvedcount == 0 | ||
Matt Mackall
|
r10282 | if show_stats: | ||
timeless
|
r27403 | _showstats(repo, stats, quietempty) | ||
r47495 | return False | |||
Matt Mackall
|
r2775 | |||
Augie Fackler
|
r43346 | |||
FUJIWARA Katsunori
|
r28501 | # naming conflict in updatetotally() | ||
_clean = clean | ||||
Augie Fackler
|
r43346 | _VALID_UPDATECHECKS = { | ||
mergemod.UPDATECHECK_ABORT, | ||||
mergemod.UPDATECHECK_NONE, | ||||
mergemod.UPDATECHECK_LINEAR, | ||||
mergemod.UPDATECHECK_NO_CONFLICT, | ||||
Augie Fackler
|
r43241 | } | ||
Augie Fackler
|
r43346 | |||
Martin von Zweigbergk
|
r31166 | def updatetotally(ui, repo, checkout, brev, clean=False, updatecheck=None): | ||
FUJIWARA Katsunori
|
r28501 | """Update the working directory with extra care for non-file components | ||
This takes care of non-file components below: | ||||
:bookmark: might be advanced or (in)activated | ||||
This takes arguments below: | ||||
:checkout: to which revision the working directory is updated | ||||
:brev: a name, which might be a bookmark to be activated after updating | ||||
:clean: whether changes in the working directory can be discarded | ||||
Martin von Zweigbergk
|
r31166 | :updatecheck: how to deal with a dirty working directory | ||
Augie Fackler
|
r43240 | Valid values for updatecheck are the UPDATECHECK_* constants | ||
defined in the merge module. Passing `None` will result in using the | ||||
configured default. | ||||
Martin von Zweigbergk
|
r31166 | |||
Augie Fackler
|
r43240 | * ABORT: abort if the working directory is dirty | ||
* NONE: don't check (merge working directory changes into destination) | ||||
* LINEAR: check that update is linear before merging working directory | ||||
Martin von Zweigbergk
|
r31166 | changes into destination | ||
Augie Fackler
|
r43240 | * NO_CONFLICT: check that the update does not result in file merges | ||
FUJIWARA Katsunori
|
r28501 | |||
This returns whether conflict is detected at updating or not. | ||||
""" | ||||
Martin von Zweigbergk
|
r31166 | if updatecheck is None: | ||
Augie Fackler
|
r43347 | updatecheck = ui.config(b'commands', b'update.check') | ||
Augie Fackler
|
r43241 | if updatecheck not in _VALID_UPDATECHECKS: | ||
Martin von Zweigbergk
|
r31167 | # If not configured, or invalid value configured | ||
Augie Fackler
|
r43240 | updatecheck = mergemod.UPDATECHECK_LINEAR | ||
Augie Fackler
|
r43241 | if updatecheck not in _VALID_UPDATECHECKS: | ||
Augie Fackler
|
r43346 | raise ValueError( | ||
r'Invalid updatecheck value %r (can accept %r)' | ||||
% (updatecheck, _VALID_UPDATECHECKS) | ||||
) | ||||
FUJIWARA Katsunori
|
r28503 | with repo.wlock(): | ||
FUJIWARA Katsunori
|
r28501 | movemarkfrom = None | ||
warndest = False | ||||
if checkout is None: | ||||
Martin von Zweigbergk
|
r30962 | updata = destutil.destupdate(repo, clean=clean) | ||
FUJIWARA Katsunori
|
r28501 | checkout, movemarkfrom, brev = updata | ||
warndest = True | ||||
if clean: | ||||
ret = _clean(repo, checkout) | ||||
else: | ||||
Augie Fackler
|
r43240 | if updatecheck == mergemod.UPDATECHECK_ABORT: | ||
Martin von Zweigbergk
|
r30963 | cmdutil.bailifchanged(repo, merge=False) | ||
Augie Fackler
|
r43240 | updatecheck = mergemod.UPDATECHECK_NONE | ||
Martin von Zweigbergk
|
r31166 | ret = _update(repo, checkout, updatecheck=updatecheck) | ||
FUJIWARA Katsunori
|
r28501 | |||
if not ret and movemarkfrom: | ||||
Augie Fackler
|
r43347 | if movemarkfrom == repo[b'.'].node(): | ||
Augie Fackler
|
r43346 | pass # no-op update | ||
Augie Fackler
|
r43347 | elif bookmarks.update(repo, [movemarkfrom], repo[b'.'].node()): | ||
b = ui.label(repo._activebookmark, b'bookmarks.active') | ||||
ui.status(_(b"updating bookmark %s\n") % b) | ||||
FUJIWARA Katsunori
|
r28501 | else: | ||
# this can happen with a non-linear update | ||||
Augie Fackler
|
r43347 | b = ui.label(repo._activebookmark, b'bookmarks') | ||
ui.status(_(b"(leaving bookmark %s)\n") % b) | ||||
FUJIWARA Katsunori
|
r28501 | bookmarks.deactivate(repo) | ||
elif brev in repo._bookmarks: | ||||
if brev != repo._activebookmark: | ||||
Augie Fackler
|
r43347 | b = ui.label(brev, b'bookmarks.active') | ||
ui.status(_(b"(activating bookmark %s)\n") % b) | ||||
FUJIWARA Katsunori
|
r28501 | bookmarks.activate(repo, brev) | ||
elif brev: | ||||
if repo._activebookmark: | ||||
Augie Fackler
|
r43347 | b = ui.label(repo._activebookmark, b'bookmarks') | ||
ui.status(_(b"(leaving bookmark %s)\n") % b) | ||||
FUJIWARA Katsunori
|
r28501 | bookmarks.deactivate(repo) | ||
if warndest: | ||||
destutil.statusotherdests(ui, repo) | ||||
return ret | ||||
Augie Fackler
|
r43346 | |||
def merge( | ||||
Augie Fackler
|
r46554 | ctx, | ||
force=False, | ||||
remind=True, | ||||
labels=None, | ||||
Augie Fackler
|
r43346 | ): | ||
Greg Ward
|
r13162 | """Branch merge with node, resolving changes. Return true if any | ||
unresolved conflicts.""" | ||||
Martin von Zweigbergk
|
r44916 | repo = ctx.repo() | ||
stats = mergemod.merge(ctx, force=force, labels=labels) | ||||
Matt Mackall
|
r3316 | _showstats(repo, stats) | ||
Gregory Szorc
|
r37143 | if stats.unresolvedcount: | ||
Augie Fackler
|
r43346 | repo.ui.status( | ||
_( | ||||
Augie Fackler
|
r43347 | b"use 'hg resolve' to retry unresolved file merges " | ||
b"or 'hg merge --abort' to abandon\n" | ||||
Augie Fackler
|
r43346 | ) | ||
) | ||||
Taapas Agrawal
|
r42803 | elif remind: | ||
Augie Fackler
|
r43347 | repo.ui.status(_(b"(branch merge, don't forget to commit)\n")) | ||
Gregory Szorc
|
r37143 | return stats.unresolvedcount > 0 | ||
Matt Mackall
|
r2808 | |||
Augie Fackler
|
r43346 | |||
Taapas Agrawal
|
r42810 | def abortmerge(ui, repo): | ||
Augie Fackler
|
r45383 | ms = mergestatemod.mergestate.read(repo) | ||
Taapas Agrawal
|
r42803 | if ms.active(): | ||
# there were conflicts | ||||
node = ms.localctx.hex() | ||||
else: | ||||
# there were no conficts, mergestate was not stored | ||||
Augie Fackler
|
r43347 | node = repo[b'.'].hex() | ||
Taapas Agrawal
|
r42803 | |||
Martin von Zweigbergk
|
r43387 | repo.ui.status(_(b"aborting the merge, updating back to %s\n") % node[:12]) | ||
Martin von Zweigbergk
|
r44743 | stats = mergemod.clean_update(repo[node]) | ||
Martin von Zweigbergk
|
r44636 | assert stats.unresolvedcount == 0 | ||
Taapas Agrawal
|
r42803 | _showstats(repo, stats) | ||
Augie Fackler
|
r43346 | |||
def _incoming( | ||||
r47712 | displaychlist, | |||
subreporecurse, | ||||
ui, | ||||
repo, | ||||
source, | ||||
opts, | ||||
buffered=False, | ||||
subpath=None, | ||||
Augie Fackler
|
r43346 | ): | ||
Nicolas Dumazet
|
r12730 | """ | ||
Helper for incoming / gincoming. | ||||
displaychlist gets called with | ||||
(remoterepo, incomingchangesetlist, displayer) parameters, | ||||
and is supposed to contain only code that can't be unified. | ||||
""" | ||||
r49054 | srcs = urlutil.get_pull_paths(repo, ui, [source]) | |||
r47694 | srcs = list(srcs) | |||
if len(srcs) != 1: | ||||
Matt Harbison
|
r47768 | msg = _(b'for now, incoming supports only a single source, %d provided') | ||
r47694 | msg %= len(srcs) | |||
raise error.Abort(msg) | ||||
r49054 | path = srcs[0] | |||
r50613 | if subpath is None: | |||
peer_path = path | ||||
url = path.loc | ||||
else: | ||||
# XXX path: we are losing the `path` object here. Keeping it would be | ||||
# valuable. For example as a "variant" as we do for pushes. | ||||
r47712 | subpath = urlutil.url(subpath) | |||
if subpath.isabs(): | ||||
r50613 | peer_path = url = bytes(subpath) | |||
r47712 | else: | |||
r50613 | p = urlutil.url(path.loc) | |||
Matt Harbison
|
r49400 | if p.islocal(): | ||
normpath = os.path.normpath | ||||
else: | ||||
normpath = posixpath.normpath | ||||
p.path = normpath(b'%s/%s' % (p.path, subpath)) | ||||
r50613 | peer_path = url = bytes(p) | |||
other = peer(repo, opts, peer_path) | ||||
Valentin Gatien-Baron
|
r47419 | cleanupfn = other.close | ||
try: | ||||
r50613 | ui.status(_(b'comparing with %s\n') % urlutil.hidepassword(url)) | |||
branches = (path.branch, opts.get(b'branch', [])) | ||||
Valentin Gatien-Baron
|
r47419 | revs, checkout = addbranchrevs(repo, other, branches, opts.get(b'rev')) | ||
Nicolas Dumazet
|
r12730 | |||
Valentin Gatien-Baron
|
r47419 | if revs: | ||
revs = [other.lookup(rev) for rev in revs] | ||||
other, chlist, cleanupfn = bundlerepo.getremotechanges( | ||||
Sushil khanchi
|
r48987 | ui, repo, other, revs, opts.get(b"bundle"), opts.get(b"force") | ||
Valentin Gatien-Baron
|
r47419 | ) | ||
Peter Arrenbrecht
|
r14161 | if not chlist: | ||
Augie Fackler
|
r43347 | ui.status(_(b"no changes found\n")) | ||
Peter Arrenbrecht
|
r14161 | return subreporecurse() | ||
Augie Fackler
|
r43347 | ui.pager(b'incoming') | ||
Augie Fackler
|
r43346 | displayer = logcmdutil.changesetdisplayer( | ||
ui, other, opts, buffered=buffered | ||||
) | ||||
Nicolas Dumazet
|
r12730 | displaychlist(other, chlist, displayer) | ||
displayer.close() | ||||
finally: | ||||
Peter Arrenbrecht
|
r14161 | cleanupfn() | ||
Nicolas Dumazet
|
r12730 | subreporecurse() | ||
Augie Fackler
|
r43346 | return 0 # exit code is zero since we found incoming changes | ||
Nicolas Dumazet
|
r12730 | |||
r47712 | def incoming(ui, repo, source, opts, subpath=None): | |||
Nicolas Dumazet
|
r12730 | def subreporecurse(): | ||
Erik Zielke
|
r12400 | ret = 1 | ||
Augie Fackler
|
r43347 | if opts.get(b'subrepos'): | ||
Erik Zielke
|
r12400 | ctx = repo[None] | ||
for subpath in sorted(ctx.substate): | ||||
sub = ctx.sub(subpath) | ||||
ret = min(ret, sub.incoming(ui, source, opts)) | ||||
return ret | ||||
Nicolas Dumazet
|
r12730 | def display(other, chlist, displayer): | ||
Yuya Nishihara
|
r35906 | limit = logcmdutil.getlimit(opts) | ||
Augie Fackler
|
r43347 | if opts.get(b'newest_first'): | ||
Nicolas Dumazet
|
r12729 | chlist.reverse() | ||
Martin Geisler
|
r12273 | count = 0 | ||
Nicolas Dumazet
|
r12729 | for n in chlist: | ||
Martin Geisler
|
r12273 | if limit is not None and count >= limit: | ||
break | ||||
Joerg Sonnenberger
|
r47771 | parents = [ | ||
p for p in other.changelog.parents(n) if p != repo.nullid | ||||
] | ||||
Augie Fackler
|
r43347 | if opts.get(b'no_merges') and len(parents) == 2: | ||
Martin Geisler
|
r12273 | continue | ||
count += 1 | ||||
displayer.show(other[n]) | ||||
Augie Fackler
|
r43346 | |||
r47712 | return _incoming( | |||
display, subreporecurse, ui, repo, source, opts, subpath=subpath | ||||
) | ||||
Martin Geisler
|
r12273 | |||
Augie Fackler
|
r43346 | |||
r52975 | def _outgoing_filter(repo, revs, opts): | |||
"""apply revision filtering/ordering option for outgoing""" | ||||
limit = logcmdutil.getlimit(opts) | ||||
no_merges = opts.get(b'no_merges') | ||||
if opts.get(b'newest_first'): | ||||
revs.reverse() | ||||
if limit is None and not no_merges: | ||||
for r in revs: | ||||
yield r | ||||
return | ||||
count = 0 | ||||
cl = repo.changelog | ||||
for n in revs: | ||||
if limit is not None and count >= limit: | ||||
break | ||||
parents = [p for p in cl.parents(n) if p != repo.nullid] | ||||
if no_merges and len(parents) == 2: | ||||
continue | ||||
count += 1 | ||||
yield n | ||||
def _outgoing_recurse(ui, repo, dests, opts): | ||||
ret = 1 | ||||
if opts.get(b'subrepos'): | ||||
ctx = repo[None] | ||||
for subpath in sorted(ctx.substate): | ||||
sub = ctx.sub(subpath) | ||||
ret = min(ret, sub.outgoing(ui, dests, opts)) | ||||
return ret | ||||
Felipe Resende
|
r52976 | def display_outgoing_revs(ui, repo, o, opts): | ||
r52977 | # make sure this is ordered by revision number | |||
cl = repo.changelog | ||||
o.sort(key=cl.rev) | ||||
Felipe Resende
|
r52976 | if opts.get(b'graph'): | ||
revdag = logcmdutil.graphrevs(repo, o, opts) | ||||
ui.pager(b'outgoing') | ||||
displayer = logcmdutil.changesetdisplayer(ui, repo, opts, buffered=True) | ||||
logcmdutil.displaygraph( | ||||
ui, repo, revdag, displayer, graphmod.asciiedges | ||||
) | ||||
else: | ||||
ui.pager(b'outgoing') | ||||
displayer = logcmdutil.changesetdisplayer(ui, repo, opts) | ||||
for n in _outgoing_filter(repo, o, opts): | ||||
displayer.show(repo[n]) | ||||
displayer.close() | ||||
Felipe Resende
|
r52979 | _no_subtoppath = object() | ||
r52975 | def outgoing(ui, repo, dests, opts, subpath=None): | |||
if opts.get(b'graph'): | ||||
logcmdutil.checkunsupportedgraphflags([], opts) | ||||
ret = 1 | ||||
r47693 | for path in urlutil.get_push_paths(repo, ui, dests): | |||
r50596 | dest = path.loc | |||
Felipe Resende
|
r52979 | prev_subtopath = getattr(repo, "_subtoppath", _no_subtoppath) | ||
try: | ||||
repo._subtoppath = dest | ||||
Felipe Resende
|
r52978 | if subpath is not None: | ||
subpath = urlutil.url(subpath) | ||||
if subpath.isabs(): | ||||
dest = bytes(subpath) | ||||
Matt Harbison
|
r49400 | else: | ||
Felipe Resende
|
r52978 | p = urlutil.url(dest) | ||
if p.islocal(): | ||||
normpath = os.path.normpath | ||||
else: | ||||
normpath = posixpath.normpath | ||||
p.path = normpath(b'%s/%s' % (p.path, subpath)) | ||||
dest = bytes(p) | ||||
branches = path.branch, opts.get(b'branch') or [] | ||||
r47692 | ||||
Felipe Resende
|
r52978 | ui.status(_(b'comparing with %s\n') % urlutil.hidepassword(dest)) | ||
revs, checkout = addbranchrevs( | ||||
repo, repo, branches, opts.get(b'rev') | ||||
) | ||||
if revs: | ||||
revs = [ | ||||
repo[rev].node() for rev in logcmdutil.revrange(repo, revs) | ||||
] | ||||
Raphaël Gomès
|
r52504 | |||
Felipe Resende
|
r52978 | other = peer(repo, opts, dest) | ||
try: | ||||
outgoing = discovery.findcommonoutgoing( | ||||
repo, other, revs, force=opts.get(b'force') | ||||
) | ||||
o = outgoing.missing | ||||
if not o: | ||||
scmutil.nochangesfound(repo.ui, repo, outgoing.excluded) | ||||
Felipe Resende
|
r52979 | else: | ||
ret = 0 | ||||
display_outgoing_revs(ui, repo, o, opts) | ||||
cmdutil.outgoinghooks(ui, repo, other, opts, o) | ||||
ret = min(ret, _outgoing_recurse(ui, repo, dests, opts)) | ||||
Felipe Resende
|
r52978 | except: # re-raises | ||
Felipe Resende
|
r52979 | raise | ||
finally: | ||||
Felipe Resende
|
r52978 | other.close() | ||
Felipe Resende
|
r52979 | finally: | ||
if prev_subtopath is _no_subtoppath: | ||||
del repo._subtoppath | ||||
else: | ||||
repo._subtoppath = prev_subtopath | ||||
return ret | ||||
Nicolas Dumazet
|
r12735 | |||
Augie Fackler
|
r43346 | |||
r42331 | def verify(repo, level=None): | |||
Matt Mackall
|
r2778 | """verify the consistency of a repository""" | ||
r42331 | ret = verifymod.verify(repo, level=level) | |||
Matt Harbison
|
r25591 | |||
# Broken subrepo references in hidden csets don't seem worth worrying about, | ||||
# since they can't be pushed/pulled, and --hidden can be used if they are a | ||||
# concern. | ||||
# pathto() is needed for -R case | ||||
Augie Fackler
|
r43346 | revs = repo.revs( | ||
Augie Fackler
|
r43347 | b"filelog(%s)", util.pathto(repo.root, repo.getcwd(), b'.hgsubstate') | ||
Augie Fackler
|
r43346 | ) | ||
Matt Harbison
|
r25591 | |||
if revs: | ||||
Augie Fackler
|
r43347 | repo.ui.status(_(b'checking subrepo links\n')) | ||
Matt Harbison
|
r25591 | for rev in revs: | ||
ctx = repo[rev] | ||||
try: | ||||
for subpath in ctx.substate: | ||||
Matt Harbison
|
r29021 | try: | ||
Augie Fackler
|
r43346 | ret = ( | ||
ctx.sub(subpath, allowcreate=False).verify() or ret | ||||
) | ||||
Matt Harbison
|
r29021 | except error.RepoError as e: | ||
Augie Fackler
|
r43347 | repo.ui.warn(b'%d: %s\n' % (rev, e)) | ||
Matt Harbison
|
r25591 | except Exception: | ||
Augie Fackler
|
r43346 | repo.ui.warn( | ||
Augie Fackler
|
r43347 | _(b'.hgsubstate is corrupt in revision %s\n') | ||
Joerg Sonnenberger
|
r46729 | % short(ctx.node()) | ||
Augie Fackler
|
r43346 | ) | ||
Matt Harbison
|
r25591 | |||
return ret | ||||
Matt Mackall
|
r11273 | |||
Augie Fackler
|
r43346 | |||
Matt Mackall
|
r11273 | def remoteui(src, opts): | ||
Matt Harbison
|
r44226 | """build a remote ui from ui or repo and opts""" | ||
r51821 | if hasattr(src, 'baseui'): # looks like a repository | |||
Augie Fackler
|
r43346 | dst = src.baseui.copy() # drop repo-specific config | ||
src = src.ui # copy target options from repo | ||||
else: # assume it's a global ui object | ||||
dst = src.copy() # keep all global options | ||||
Matt Mackall
|
r11273 | |||
# copy ssh-specific options | ||||
Augie Fackler
|
r43347 | for o in b'ssh', b'remotecmd': | ||
v = opts.get(o) or src.config(b'ui', o) | ||||
Matt Mackall
|
r11273 | if v: | ||
Augie Fackler
|
r43347 | dst.setconfig(b"ui", o, v, b'copied') | ||
Matt Mackall
|
r11273 | |||
# copy bundle-specific options | ||||
Augie Fackler
|
r43347 | r = src.config(b'bundle', b'mainreporoot') | ||
Matt Mackall
|
r11273 | if r: | ||
Augie Fackler
|
r43347 | dst.setconfig(b'bundle', b'mainreporoot', r, b'copied') | ||
Matt Mackall
|
r11273 | |||
Mads Kiilerich
|
r13192 | # copy selected local settings to the remote ui | ||
Augie Fackler
|
r43347 | for sect in (b'auth', b'hostfingerprints', b'hostsecurity', b'http_proxy'): | ||
Matt Mackall
|
r11273 | for key, val in src.configitems(sect): | ||
Augie Fackler
|
r43347 | dst.setconfig(sect, key, val, b'copied') | ||
v = src.config(b'web', b'cacerts') | ||||
Yuya Nishihara
|
r29594 | if v: | ||
Augie Fackler
|
r43347 | dst.setconfig(b'web', b'cacerts', util.expandpath(v), b'copied') | ||
Matt Mackall
|
r11273 | |||
return dst | ||||
Gregory Szorc
|
r26219 | |||
Augie Fackler
|
r43346 | |||
Gregory Szorc
|
r26219 | # Files of interest | ||
# Used to check if the repository has changed looking at mtime and size of | ||||
Mads Kiilerich
|
r26781 | # these files. | ||
Matt Harbison
|
r52567 | foi: "List[Tuple[str, bytes]]" = [ | ||
r51804 | ('spath', b'00changelog.i'), | |||
('spath', b'phaseroots'), # ! phase can change content at the same size | ||||
('spath', b'obsstore'), | ||||
('path', b'bookmarks'), # ! bookmark can change content at the same size | ||||
Augie Fackler
|
r43346 | ] | ||
Gregory Szorc
|
r26219 | |||
Gregory Szorc
|
r49801 | class cachedlocalrepo: | ||
Gregory Szorc
|
r26219 | """Holds a localrepository that can be cached and reused.""" | ||
def __init__(self, repo): | ||||
"""Create a new cached repo from an existing repo. | ||||
We assume the passed in repo was recently created. If the | ||||
repo has changed between when it was created and when it was | ||||
turned into a cache, it may not refresh properly. | ||||
""" | ||||
assert isinstance(repo, localrepo.localrepository) | ||||
self._repo = repo | ||||
self._state, self.mtime = self._repostate() | ||||
FUJIWARA Katsunori
|
r28119 | self._filtername = repo.filtername | ||
Gregory Szorc
|
r26219 | |||
def fetch(self): | ||||
"""Refresh (if necessary) and return a repository. | ||||
If the cached instance is out of date, it will be recreated | ||||
automatically and returned. | ||||
Returns a tuple of the repo and a boolean indicating whether a new | ||||
repo instance was created. | ||||
""" | ||||
# We compare the mtimes and sizes of some well-known files to | ||||
# determine if the repo changed. This is not precise, as mtimes | ||||
# are susceptible to clock skew and imprecise filesystems and | ||||
# file content can change while maintaining the same size. | ||||
state, mtime = self._repostate() | ||||
if state == self._state: | ||||
return self._repo, False | ||||
FUJIWARA Katsunori
|
r28119 | repo = repository(self._repo.baseui, self._repo.url()) | ||
if self._filtername: | ||||
self._repo = repo.filtered(self._filtername) | ||||
else: | ||||
self._repo = repo.unfiltered() | ||||
Gregory Szorc
|
r26219 | self._state = state | ||
self.mtime = mtime | ||||
return self._repo, True | ||||
def _repostate(self): | ||||
state = [] | ||||
maxmtime = -1 | ||||
for attr, fname in foi: | ||||
prefix = getattr(self._repo, attr) | ||||
p = os.path.join(prefix, fname) | ||||
try: | ||||
st = os.stat(p) | ||||
except OSError: | ||||
st = os.stat(prefix) | ||||
Augie Fackler
|
r36799 | state.append((st[stat.ST_MTIME], st.st_size)) | ||
maxmtime = max(maxmtime, st[stat.ST_MTIME]) | ||||
Gregory Szorc
|
r26219 | |||
return tuple(state), maxmtime | ||||
def copy(self): | ||||
Gregory Szorc
|
r26240 | """Obtain a copy of this class instance. | ||
A new localrepository instance is obtained. The new instance should be | ||||
completely independent of the original. | ||||
""" | ||||
repo = repository(self._repo.baseui, self._repo.origroot) | ||||
FUJIWARA Katsunori
|
r28119 | if self._filtername: | ||
repo = repo.filtered(self._filtername) | ||||
else: | ||||
repo = repo.unfiltered() | ||||
Gregory Szorc
|
r26240 | c = cachedlocalrepo(repo) | ||
Gregory Szorc
|
r26219 | c._state = self._state | ||
c.mtime = self.mtime | ||||
return c | ||||