hg.py
825 lines
| 28.2 KiB
| text/x-python
|
PythonLexer
/ mercurial / hg.py
mpm@selenic.com
|
r0 | # hg.py - repository classes for mercurial | ||
# | ||||
Thomas Arendsen Hein
|
r4635 | # Copyright 2005-2007 Matt Mackall <mpm@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 | |||
Gregory Szorc
|
r25939 | from __future__ import absolute_import | ||
import errno | ||||
import os | ||||
import shutil | ||||
from .i18n import _ | ||||
from .node import nullid | ||||
Jordi Gutiérrez Hermoso
|
r22837 | |||
Gregory Szorc
|
r25939 | from . import ( | ||
bookmarks, | ||||
bundlerepo, | ||||
cmdutil, | ||||
discovery, | ||||
error, | ||||
exchange, | ||||
extensions, | ||||
httppeer, | ||||
localrepo, | ||||
lock, | ||||
merge as mergemod, | ||||
node, | ||||
phases, | ||||
repoview, | ||||
scmutil, | ||||
sshpeer, | ||||
statichttprepo, | ||||
ui as uimod, | ||||
unionrepo, | ||||
url, | ||||
util, | ||||
verify as verifymod, | ||||
) | ||||
release = lock.release | ||||
mpm@selenic.com
|
r0 | |||
Vadim Gelfer
|
r2740 | def _local(path): | ||
Mads Kiilerich
|
r14825 | path = util.expandpath(util.urllocalpath(path)) | ||
Alexander Solovyov
|
r11154 | return (os.path.isfile(path) and bundlerepo or localrepo) | ||
Vadim Gelfer
|
r2469 | |||
Sune Foldager
|
r17191 | def addbranchrevs(lrepo, other, branches, revs): | ||
peer = other.peer() # a courtesy to callers using a localrepo for other | ||||
Sune Foldager
|
r11322 | hashbranch, branches = branches | ||
if not hashbranch and not branches: | ||||
Pierre-Yves David
|
r22818 | x = revs or None | ||
if util.safehasattr(revs, 'first'): | ||||
y = revs.first() | ||||
elif revs: | ||||
y = revs[0] | ||||
else: | ||||
y = None | ||||
return x, y | ||||
Jordi Gutiérrez Hermoso
|
r24306 | if revs: | ||
revs = list(revs) | ||||
else: | ||||
revs = [] | ||||
Sune Foldager
|
r17191 | if not peer.capable('branchmap'): | ||
Sune Foldager
|
r11322 | if branches: | ||
raise util.Abort(_("remote branch lookup not supported")) | ||||
revs.append(hashbranch) | ||||
Sune Foldager
|
r10380 | return revs, revs[0] | ||
Sune Foldager
|
r17191 | branchmap = peer.branchmap() | ||
Sune Foldager
|
r11322 | |||
Matt Mackall
|
r13047 | def primary(branch): | ||
if branch == '.': | ||||
Sune Foldager
|
r17191 | if not lrepo: | ||
Sune Foldager
|
r10365 | raise util.Abort(_("dirstate branch not accessible")) | ||
Matt Mackall
|
r13047 | branch = lrepo.dirstate.branch() | ||
if branch in branchmap: | ||||
revs.extend(node.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): | ||
Sune Foldager
|
r11322 | raise error.RepoLookupError(_("unknown branch '%s'") % branch) | ||
if hashbranch: | ||||
Matt Mackall
|
r13047 | if not primary(hashbranch): | ||
Sune Foldager
|
r11322 | revs.append(hashbranch) | ||
Sune Foldager
|
r10365 | return revs, revs[0] | ||
Brodie Rao
|
r13824 | def parseurl(path, branches=None): | ||
Sune Foldager
|
r11322 | '''parse url#branch, returning (url, (branch, branches))''' | ||
Matt Mackall
|
r5177 | |||
Brodie Rao
|
r14076 | u = util.url(path) | ||
Thomas Arendsen Hein
|
r13897 | branch = None | ||
if u.fragment: | ||||
branch = u.fragment | ||||
u.fragment = None | ||||
Brodie Rao
|
r13824 | return str(u), (branch, branches or []) | ||
Matt Mackall
|
r5177 | |||
Matt Mackall
|
r14606 | schemes = { | ||
Matt Mackall
|
r14568 | 'bundle': bundlerepo, | ||
Mads Kiilerich
|
r18944 | 'union': unionrepo, | ||
Matt Mackall
|
r14568 | 'file': _local, | ||
Peter Arrenbrecht
|
r17192 | 'http': httppeer, | ||
'https': httppeer, | ||||
'ssh': sshpeer, | ||||
Matt Mackall
|
r14568 | 'static-http': statichttprepo, | ||
} | ||||
def _peerlookup(path): | ||||
u = util.url(path) | ||||
scheme = u.scheme or 'file' | ||||
Matt Mackall
|
r14606 | thing = schemes.get(scheme) or schemes['file'] | ||
Matt Mackall
|
r14568 | try: | ||
return thing(path) | ||||
except TypeError: | ||||
Yuya Nishihara
|
r25365 | # we can't test callable(thing) because 'thing' can be an unloaded | ||
# module that implements __call__ | ||||
if not util.safehasattr(thing, 'instance'): | ||||
raise | ||||
Matt Mackall
|
r14568 | return thing | ||
Matt Mackall
|
r14605 | def islocal(repo): | ||
Siddharth Agarwal
|
r20355 | '''return true if repo (or path pointing to repo) is local''' | ||
Matt Mackall
|
r14605 | if isinstance(repo, str): | ||
try: | ||||
return _peerlookup(repo).islocal(repo) | ||||
except AttributeError: | ||||
return False | ||||
return repo.local() | ||||
Siddharth Agarwal
|
r17887 | def openpath(ui, path): | ||
'''open path with open if local, url.open if remote''' | ||||
Siddharth Agarwal
|
r20354 | pathurl = util.url(path, parsequery=False, parsefragment=False) | ||
if pathurl.islocal(): | ||||
return util.posixfile(pathurl.localpath(), 'rb') | ||||
Siddharth Agarwal
|
r17887 | else: | ||
return url.open(ui, path) | ||||
FUJIWARA Katsunori
|
r20858 | # a list of (ui, repo) functions called for wire peer initialization | ||
wirepeersetupfuncs = [] | ||||
Sune Foldager
|
r17191 | def _peerorrepo(ui, path, create=False): | ||
Matt Mackall
|
r14605 | """return a repository object for the specified path""" | ||
Sune Foldager
|
r17191 | obj = _peerlookup(path).instance(ui, path, create) | ||
ui = getattr(obj, "ui", ui) | ||||
FUJIWARA Katsunori
|
r19777 | for name, module in extensions.extensions(ui): | ||
Matt Mackall
|
r14605 | hook = getattr(module, 'reposetup', None) | ||
if hook: | ||||
Sune Foldager
|
r17191 | hook(ui, obj) | ||
FUJIWARA Katsunori
|
r20858 | if not obj.local(): | ||
for f in wirepeersetupfuncs: | ||||
f(ui, obj) | ||||
Sune Foldager
|
r17191 | return obj | ||
def repository(ui, path='', create=False): | ||||
"""return a repository object for the specified path""" | ||||
peer = _peerorrepo(ui, path, create) | ||||
repo = peer.local() | ||||
if not repo: | ||||
raise util.Abort(_("repository '%s' is not local") % | ||||
(path or peer.url())) | ||||
Kevin Bullock
|
r18382 | return repo.filtered('visible') | ||
Matt Mackall
|
r14605 | |||
Idan Kamara
|
r14839 | def peer(uiorrepo, opts, path, create=False): | ||
Matt Mackall
|
r14554 | '''return a repository peer for the specified path''' | ||
Idan Kamara
|
r14839 | rui = remoteui(uiorrepo, opts) | ||
Sune Foldager
|
r17191 | return _peerorrepo(rui, path, create).peer() | ||
Matt Mackall
|
r14554 | |||
Vadim Gelfer
|
r2719 | def defaultdest(source): | ||
Yuya Nishihara
|
r20799 | '''return default destination of clone if none is given | ||
>>> defaultdest('foo') | ||||
'foo' | ||||
>>> defaultdest('/foo/bar') | ||||
'bar' | ||||
>>> defaultdest('/') | ||||
'' | ||||
>>> defaultdest('') | ||||
Yuya Nishihara
|
r20800 | '' | ||
Yuya Nishihara
|
r20799 | >>> defaultdest('http://example.org/') | ||
Yuya Nishihara
|
r20800 | '' | ||
Yuya Nishihara
|
r20799 | >>> defaultdest('http://example.org/foo/') | ||
'foo' | ||||
''' | ||||
Yuya Nishihara
|
r20800 | path = util.url(source).path | ||
if not path: | ||||
return '' | ||||
return os.path.basename(os.path.normpath(path)) | ||||
Matt Mackall
|
r2774 | |||
Ryan McElroy
|
r23614 | def share(ui, source, dest=None, update=True, bookmarks=True): | ||
Matt Mackall
|
r8800 | '''create a shared repository''' | ||
if not islocal(source): | ||||
raise util.Abort(_('can only share local repositories')) | ||||
Matt Mackall
|
r8807 | if not dest: | ||
Brendan Cully
|
r10099 | dest = defaultdest(source) | ||
Matt Mackall
|
r9344 | else: | ||
dest = ui.expandpath(dest) | ||||
Matt Mackall
|
r8807 | |||
Matt Mackall
|
r8800 | if isinstance(source, str): | ||
origsource = ui.expandpath(source) | ||||
Sune Foldager
|
r10365 | source, branches = parseurl(origsource) | ||
Matt Mackall
|
r8800 | srcrepo = repository(ui, source) | ||
Sune Foldager
|
r10365 | rev, checkout = addbranchrevs(srcrepo, srcrepo, branches, None) | ||
Matt Mackall
|
r8800 | else: | ||
Sune Foldager
|
r17191 | srcrepo = source.local() | ||
Matt Mackall
|
r8800 | origsource = source = srcrepo.url() | ||
checkout = None | ||||
sharedpath = srcrepo.sharedpath # if our source is already sharing | ||||
Chinmay Joshi
|
r21800 | destwvfs = scmutil.vfs(dest, realpath=True) | ||
Chinmay Joshi
|
r21801 | destvfs = scmutil.vfs(os.path.join(destwvfs.base, '.hg'), realpath=True) | ||
Matt Mackall
|
r8800 | |||
Chinmay Joshi
|
r21801 | if destvfs.lexists(): | ||
Matt Mackall
|
r8800 | raise util.Abort(_('destination already exists')) | ||
Chinmay Joshi
|
r21800 | if not destwvfs.isdir(): | ||
destwvfs.mkdir() | ||||
Chinmay Joshi
|
r21801 | destvfs.makedir() | ||
Matt Mackall
|
r8800 | |||
requirements = '' | ||||
try: | ||||
Angel Ezquerra
|
r23877 | requirements = srcrepo.vfs.read('requires') | ||
Gregory Szorc
|
r25660 | except IOError as inst: | ||
Matt Mackall
|
r8800 | if inst.errno != errno.ENOENT: | ||
raise | ||||
requirements += 'shared\n' | ||||
Chinmay Joshi
|
r21802 | destvfs.write('requires', requirements) | ||
destvfs.write('sharedpath', sharedpath) | ||||
Matt Mackall
|
r8800 | |||
Chinmay Joshi
|
r21800 | r = repository(ui, destwvfs.base) | ||
Dan Villiom Podlaski Christiansen
|
r14156 | |||
Matt Mackall
|
r8800 | default = srcrepo.ui.config('paths', 'default') | ||
Matt Harbison
|
r18511 | if default: | ||
Angel Ezquerra
|
r23877 | fp = r.vfs("hgrc", "w", text=True) | ||
Matt Harbison
|
r18511 | fp.write("[paths]\n") | ||
fp.write("default = %s\n" % default) | ||||
fp.close() | ||||
Matt Mackall
|
r8800 | |||
if update: | ||||
r.ui.status(_("updating working directory\n")) | ||||
if update is not True: | ||||
checkout = update | ||||
for test in (checkout, 'default', 'tip'): | ||||
Matt Mackall
|
r9423 | if test is None: | ||
continue | ||||
Matt Mackall
|
r8800 | try: | ||
uprev = r.lookup(test) | ||||
break | ||||
Matt Mackall
|
r9423 | except error.RepoLookupError: | ||
Matt Mackall
|
r8800 | continue | ||
_update(r, uprev) | ||||
Ryan McElroy
|
r23614 | if bookmarks: | ||
Angel Ezquerra
|
r23883 | fp = r.vfs('shared', 'w') | ||
fp.write('bookmarks\n') | ||||
fp.close() | ||||
Ryan McElroy
|
r23614 | |||
Simon Heimberg
|
r15078 | def copystore(ui, srcrepo, destpath): | ||
'''copy files from store of srcrepo in destpath | ||||
returns destlock | ||||
''' | ||||
destlock = None | ||||
try: | ||||
hardlink = None | ||||
num = 0 | ||||
Augie Fackler
|
r24440 | closetopic = [None] | ||
def prog(topic, pos): | ||||
if pos is None: | ||||
closetopic[0] = topic | ||||
else: | ||||
ui.progress(topic, pos + num) | ||||
Matt Mackall
|
r25624 | srcpublishing = srcrepo.publishing() | ||
FUJIWARA Katsunori
|
r20089 | srcvfs = scmutil.vfs(srcrepo.sharedpath) | ||
dstvfs = scmutil.vfs(destpath) | ||||
Simon Heimberg
|
r15078 | for f in srcrepo.store.copylist(): | ||
Pierre-Yves David
|
r15741 | if srcpublishing and f.endswith('phaseroots'): | ||
continue | ||||
FUJIWARA Katsunori
|
r20089 | dstbase = os.path.dirname(f) | ||
if dstbase and not dstvfs.exists(dstbase): | ||||
dstvfs.mkdir(dstbase) | ||||
if srcvfs.exists(f): | ||||
if f.endswith('data'): | ||||
FUJIWARA Katsunori
|
r20825 | # 'dstbase' may be empty (e.g. revlog format 0) | ||
lockfile = os.path.join(dstbase, "lock") | ||||
Simon Heimberg
|
r15078 | # lock to avoid premature writing to the target | ||
FUJIWARA Katsunori
|
r20825 | destlock = lock.lock(dstvfs, lockfile) | ||
FUJIWARA Katsunori
|
r20089 | hardlink, n = util.copyfiles(srcvfs.join(f), dstvfs.join(f), | ||
Augie Fackler
|
r24440 | hardlink, progress=prog) | ||
Simon Heimberg
|
r15078 | num += n | ||
if hardlink: | ||||
ui.debug("linked %d files\n" % num) | ||||
Augie Fackler
|
r24440 | if closetopic[0]: | ||
ui.progress(closetopic[0], None) | ||||
Simon Heimberg
|
r15078 | else: | ||
ui.debug("copied %d files\n" % num) | ||||
Augie Fackler
|
r24440 | if closetopic[0]: | ||
ui.progress(closetopic[0], None) | ||||
Simon Heimberg
|
r15078 | return destlock | ||
Brodie Rao
|
r16705 | except: # re-raises | ||
Simon Heimberg
|
r15078 | release(destlock) | ||
raise | ||||
Gregory Szorc
|
r25761 | def clonewithshare(ui, peeropts, sharepath, source, srcpeer, dest, pull=False, | ||
rev=None, update=True, stream=False): | ||||
"""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: | ||||
if not srcpeer.capable('lookup'): | ||||
raise util.Abort(_("src repository does not support " | ||||
"revision lookup and so doesn't " | ||||
"support clone by revision")) | ||||
revs = [srcpeer.lookup(r) for r in rev] | ||||
basename = os.path.basename(sharepath) | ||||
if os.path.exists(sharepath): | ||||
ui.status(_('(sharing from existing pooled repository %s)\n') % | ||||
basename) | ||||
else: | ||||
ui.status(_('(sharing from new pooled repository %s)\n') % basename) | ||||
# Always use pull mode because hardlinks in share mode don't work well. | ||||
# Never update because working copies aren't necessary in share mode. | ||||
clone(ui, peeropts, source, dest=sharepath, pull=True, | ||||
rev=rev, update=False, stream=stream) | ||||
sharerepo = repository(ui, path=sharepath) | ||||
share(ui, sharerepo, dest=dest, update=update, bookmarks=False) | ||||
# 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. | ||||
destrepo = repository(ui, path=dest) | ||||
exchange.pull(destrepo, srcpeer, heads=revs) | ||||
return srcpeer, peer(ui, peeropts, dest) | ||||
Martin Geisler
|
r14607 | def clone(ui, peeropts, source, dest=None, pull=False, rev=None, | ||
Gregory Szorc
|
r25761 | update=True, stream=False, branch=None, shareopts=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 | |||
Vadim Gelfer
|
r2597 | rev: revision to clone up to (implies pull=True) | ||
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." | ||||
Vadim Gelfer
|
r2597 | """ | ||
Matt Mackall
|
r4478 | |||
Vadim Gelfer
|
r2719 | if isinstance(source, str): | ||
Alexis S. L. Carvalho
|
r6089 | origsource = ui.expandpath(source) | ||
Sune Foldager
|
r10379 | source, branch = parseurl(origsource, branch) | ||
Sune Foldager
|
r17191 | srcpeer = peer(ui, peeropts, source) | ||
Vadim Gelfer
|
r2719 | else: | ||
Sune Foldager
|
r17191 | srcpeer = source.peer() # in case we were called with a localrepo | ||
Nicolas Dumazet
|
r11818 | branch = (None, branch or []) | ||
Sune Foldager
|
r17191 | origsource = source = srcpeer.url() | ||
rev, checkout = addbranchrevs(srcpeer, srcpeer, branch, rev) | ||||
Vadim Gelfer
|
r2719 | |||
Vadim Gelfer
|
r2597 | if dest is None: | ||
Vadim Gelfer
|
r2719 | dest = defaultdest(source) | ||
Yuya Nishihara
|
r20800 | if dest: | ||
ui.status(_("destination directory: %s\n") % dest) | ||||
Matt Mackall
|
r9344 | else: | ||
dest = ui.expandpath(dest) | ||||
Vadim Gelfer
|
r2719 | |||
Mads Kiilerich
|
r14825 | dest = util.urllocalpath(dest) | ||
source = util.urllocalpath(source) | ||||
Vadim Gelfer
|
r2597 | |||
FUJIWARA Katsunori
|
r17159 | if not dest: | ||
raise util.Abort(_("empty destination path is not valid")) | ||||
Chinmay Joshi
|
r21803 | |||
destvfs = scmutil.vfs(dest, expandpath=True) | ||||
if destvfs.lexists(): | ||||
if not destvfs.isdir(): | ||||
Steve Borho
|
r7927 | raise util.Abort(_("destination '%s' already exists") % dest) | ||
Chinmay Joshi
|
r21804 | elif destvfs.listdir(): | ||
Steve Borho
|
r7927 | raise util.Abort(_("destination '%s' is not empty") % dest) | ||
Vadim Gelfer
|
r2597 | |||
Gregory Szorc
|
r25761 | shareopts = shareopts or {} | ||
sharepool = shareopts.get('pool') | ||||
sharenamemode = shareopts.get('mode') | ||||
FUJIWARA Katsunori
|
r26026 | if sharepool and islocal(dest): | ||
Gregory Szorc
|
r25761 | sharepath = None | ||
if sharenamemode == '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: | ||||
rootnode = srcpeer.lookup('0') | ||||
if rootnode != node.nullid: | ||||
sharepath = os.path.join(sharepool, node.hex(rootnode)) | ||||
else: | ||||
ui.status(_('(not using pooled storage: ' | ||||
'remote appears to be empty)\n')) | ||||
except error.RepoLookupError: | ||||
ui.status(_('(not using pooled storage: ' | ||||
'unable to resolve identity of remote)\n')) | ||||
elif sharenamemode == 'remote': | ||||
sharepath = os.path.join(sharepool, util.sha1(source).hexdigest()) | ||||
else: | ||||
raise util.Abort('unknown share naming mode: %s' % sharenamemode) | ||||
if sharepath: | ||||
return clonewithshare(ui, peeropts, sharepath, source, srcpeer, | ||||
dest, pull=pull, rev=rev, update=update, | ||||
stream=stream) | ||||
Augie Fackler
|
r18441 | srclock = destlock = cleandir = None | ||
Sune Foldager
|
r17191 | srcrepo = srcpeer.local() | ||
Matt Mackall
|
r4915 | try: | ||
Brendan Cully
|
r14377 | abspath = origsource | ||
if islocal(origsource): | ||||
Mads Kiilerich
|
r14825 | abspath = os.path.abspath(util.urllocalpath(origsource)) | ||
Brendan Cully
|
r14377 | |||
Matt Mackall
|
r4915 | if islocal(dest): | ||
Augie Fackler
|
r18441 | cleandir = dest | ||
Vadim Gelfer
|
r2597 | |||
Matt Mackall
|
r4915 | copy = False | ||
Sune Foldager
|
r17194 | if (srcrepo and srcrepo.cancopy() and islocal(dest) | ||
Pierre-Yves David
|
r17671 | and not phases.hassecret(srcrepo)): | ||
Matt Mackall
|
r4915 | copy = not pull and not rev | ||
Vadim Gelfer
|
r2597 | |||
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: | ||
Martin Geisler
|
r14463 | srcrepo.hook('preoutgoing', throw=True, source='clone') | ||
Matt Mackall
|
r15381 | hgdir = os.path.realpath(os.path.join(dest, ".hg")) | ||
Matt Mackall
|
r4915 | if not os.path.exists(dest): | ||
os.mkdir(dest) | ||||
Steve Borho
|
r7935 | else: | ||
# only clean up directories we create ourselves | ||||
Augie Fackler
|
r18441 | cleandir = hgdir | ||
Matt Mackall
|
r5569 | try: | ||
Martin Geisler
|
r14463 | destpath = hgdir | ||
util.makedir(destpath, notindexed=True) | ||||
Gregory Szorc
|
r25660 | except OSError as inst: | ||
Matt Mackall
|
r5569 | if inst.errno == errno.EEXIST: | ||
Augie Fackler
|
r18441 | cleandir = None | ||
Matt Mackall
|
r5569 | raise util.Abort(_("destination '%s' already exists") | ||
% dest) | ||||
raise | ||||
Vadim Gelfer
|
r2597 | |||
Simon Heimberg
|
r15078 | destlock = copystore(ui, srcrepo, destpath) | ||
Pierre-Yves David
|
r22646 | # copy bookmarks over | ||
srcbookmarks = srcrepo.join('bookmarks') | ||||
dstbookmarks = os.path.join(destpath, 'bookmarks') | ||||
if os.path.exists(srcbookmarks): | ||||
util.copyfile(srcbookmarks, dstbookmarks) | ||||
Vadim Gelfer
|
r2597 | |||
Tomasz Kleczek
|
r17740 | # Recomputing branch cache might be slow on big repos, | ||
# so just copy it | ||||
Siddharth Agarwal
|
r22264 | def copybranchcache(fname): | ||
srcbranchcache = srcrepo.join('cache/%s' % fname) | ||||
dstbranchcache = os.path.join(dstcachedir, fname) | ||||
if os.path.exists(srcbranchcache): | ||||
if not os.path.exists(dstcachedir): | ||||
os.mkdir(dstcachedir) | ||||
util.copyfile(srcbranchcache, dstbranchcache) | ||||
Tomasz Kleczek
|
r17740 | dstcachedir = os.path.join(destpath, 'cache') | ||
Siddharth Agarwal
|
r22264 | # In local clones we're copying all nodes, not just served | ||
Mads Kiilerich
|
r23139 | # ones. Therefore copy all branch caches over. | ||
Siddharth Agarwal
|
r22264 | copybranchcache('branch2') | ||
for cachename in repoview.filtertable: | ||||
copybranchcache('branch2-%s' % cachename) | ||||
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) | ||
Martin Geisler
|
r14463 | srcrepo.hook('outgoing', source='clone', | ||
Martin Geisler
|
r12144 | node=node.hex(node.nullid)) | ||
Matt Mackall
|
r4915 | else: | ||
Matt Mackall
|
r5569 | try: | ||
Simon Heimberg
|
r17875 | destpeer = peer(srcrepo or ui, peeropts, dest, create=True) | ||
# only pass ui when no srcrepo | ||||
Gregory Szorc
|
r25660 | except OSError as inst: | ||
Matt Mackall
|
r5569 | if inst.errno == errno.EEXIST: | ||
Augie Fackler
|
r18441 | cleandir = None | ||
Matt Mackall
|
r5569 | raise util.Abort(_("destination '%s' already exists") | ||
% dest) | ||||
raise | ||||
Vadim Gelfer
|
r2597 | |||
Matt Mackall
|
r4915 | revs = None | ||
if rev: | ||||
Sune Foldager
|
r17191 | if not srcpeer.capable('lookup'): | ||
Martin Geisler
|
r9171 | raise util.Abort(_("src repository does not support " | ||
"revision lookup and so doesn't " | ||||
"support clone by revision")) | ||||
Sune Foldager
|
r17191 | revs = [srcpeer.lookup(r) for r in rev] | ||
Brett Carter
|
r8417 | checkout = revs[0] | ||
Sune Foldager
|
r17191 | if destpeer.local(): | ||
Siddharth Agarwal
|
r23545 | if not stream: | ||
if pull: | ||||
stream = False | ||||
else: | ||||
stream = None | ||||
Sune Foldager
|
r17191 | destpeer.local().clone(srcpeer, heads=revs, stream=stream) | ||
elif srcrepo: | ||||
Pierre-Yves David
|
r22647 | exchange.push(srcrepo, destpeer, revs=revs, | ||
bookmarks=srcrepo._bookmarks.keys()) | ||||
Matt Mackall
|
r4915 | else: | ||
raise util.Abort(_("clone from remote to remote not supported")) | ||||
Vadim Gelfer
|
r2597 | |||
Augie Fackler
|
r18441 | cleandir = None | ||
Vadim Gelfer
|
r2597 | |||
Sune Foldager
|
r17191 | destrepo = destpeer.local() | ||
if destrepo: | ||||
Jordi Gutiérrez Hermoso
|
r22837 | template = uimod.samplehgrcs['cloned'] | ||
Angel Ezquerra
|
r23877 | fp = destrepo.vfs("hgrc", "w", text=True) | ||
Augie Fackler
|
r15552 | u = util.url(abspath) | ||
u.passwd = None | ||||
defaulturl = str(u) | ||||
Augie Fackler
|
r22380 | fp.write(template % defaulturl) | ||
Matt Mackall
|
r4915 | fp.close() | ||
Benoit Boissinot
|
r5185 | |||
Mads Kiilerich
|
r20790 | destrepo.ui.setconfig('paths', 'default', defaulturl, 'clone') | ||
Matt Mackall
|
r8814 | |||
Matt Mackall
|
r4915 | if update: | ||
Bryan O'Sullivan
|
r6526 | if update is not True: | ||
Augie Fackler
|
r17342 | checkout = srcpeer.lookup(update) | ||
Thomas Arendsen Hein
|
r17867 | uprev = None | ||
Adrian Buehlmann
|
r17882 | status = None | ||
Thomas Arendsen Hein
|
r17867 | if checkout is not None: | ||
Alexis S. L. Carvalho
|
r5248 | try: | ||
Thomas Arendsen Hein
|
r17867 | uprev = destrepo.lookup(checkout) | ||
Matt Mackall
|
r9423 | except error.RepoLookupError: | ||
liscju
|
r26103 | try: | ||
uprev = destrepo.lookup(update) | ||||
except error.RepoLookupError: | ||||
pass | ||||
Thomas Arendsen Hein
|
r17867 | if uprev is None: | ||
try: | ||||
uprev = destrepo._bookmarks['@'] | ||||
Thomas Arendsen Hein
|
r17870 | update = '@' | ||
Adrian Buehlmann
|
r17882 | bn = destrepo[uprev].branch() | ||
if bn == 'default': | ||||
status = _("updating to bookmark @\n") | ||||
else: | ||||
FUJIWARA Katsunori
|
r20868 | status = (_("updating to bookmark @ on branch %s\n") | ||
Adrian Buehlmann
|
r17882 | % bn) | ||
Thomas Arendsen Hein
|
r17867 | except KeyError: | ||
try: | ||||
uprev = destrepo.branchtip('default') | ||||
except error.RepoLookupError: | ||||
uprev = destrepo.lookup('tip') | ||||
Adrian Buehlmann
|
r17882 | if not status: | ||
bn = destrepo[uprev].branch() | ||||
status = _("updating to branch %s\n") % bn | ||||
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) | ||
Matt Mackall
|
r4915 | finally: | ||
Matt Mackall
|
r15908 | release(srclock, destlock) | ||
Augie Fackler
|
r18441 | if cleandir is not None: | ||
shutil.rmtree(cleandir, True) | ||||
Sune Foldager
|
r17191 | if srcpeer is not None: | ||
srcpeer.close() | ||||
simon@laptop-tosh
|
r19313 | return srcpeer, destpeer | ||
Matt Mackall
|
r2775 | |||
Matt Mackall
|
r3316 | def _showstats(repo, stats): | ||
Martin Geisler
|
r9454 | repo.ui.status(_("%d files updated, %d files merged, " | ||
"%d files removed, %d files unresolved\n") % stats) | ||||
Matt Mackall
|
r3316 | |||
Simon Heimberg
|
r17895 | def updaterepo(repo, node, overwrite): | ||
"""Update the working directory to node. | ||||
When overwrite is set, changes are clobbered, merged else | ||||
returns stats (see pydoc mercurial.merge.applyupdates)""" | ||||
Durham Goode
|
r21537 | return mergemod.update(repo, node, False, overwrite, None, | ||
labels=['working copy', 'destination']) | ||||
Simon Heimberg
|
r17895 | |||
Matt Mackall
|
r2808 | def update(repo, node): | ||
"""update the working directory to node, merging linear changes""" | ||||
Simon Heimberg
|
r17895 | stats = updaterepo(repo, node, False) | ||
Matt Mackall
|
r3316 | _showstats(repo, stats) | ||
if stats[3]: | ||||
Matt Mackall
|
r6518 | repo.ui.status(_("use 'hg resolve' to retry unresolved file merges\n")) | ||
Matt Mackall
|
r5635 | return stats[3] > 0 | ||
Matt Mackall
|
r2775 | |||
Benoit Boissinot
|
r7546 | # naming conflict in clone() | ||
_update = update | ||||
Matt Mackall
|
r4917 | def clean(repo, node, show_stats=True): | ||
Matt Mackall
|
r2808 | """forcibly switch the working directory to node, clobbering changes""" | ||
Simon Heimberg
|
r17895 | stats = updaterepo(repo, node, True) | ||
Siddharth Agarwal
|
r19332 | util.unlinkpath(repo.join('graftstate'), ignoremissing=True) | ||
Matt Mackall
|
r10282 | if show_stats: | ||
_showstats(repo, stats) | ||||
Matt Mackall
|
r5635 | return stats[3] > 0 | ||
Matt Mackall
|
r2775 | |||
Matt Mackall
|
r4917 | def merge(repo, node, force=None, remind=True): | ||
Greg Ward
|
r13162 | """Branch merge with node, resolving changes. Return true if any | ||
unresolved conflicts.""" | ||||
Benoit Boissinot
|
r10651 | stats = mergemod.update(repo, node, True, force, False) | ||
Matt Mackall
|
r3316 | _showstats(repo, stats) | ||
if stats[3]: | ||||
Augie Fackler
|
r7821 | repo.ui.status(_("use 'hg resolve' to retry unresolved file merges " | ||
Brodie Rao
|
r12314 | "or 'hg update -C .' to abandon\n")) | ||
Matt Mackall
|
r3316 | elif remind: | ||
repo.ui.status(_("(branch merge, don't forget to commit)\n")) | ||||
Matt Mackall
|
r5635 | return stats[3] > 0 | ||
Matt Mackall
|
r2808 | |||
Nicolas Dumazet
|
r12730 | def _incoming(displaychlist, subreporecurse, ui, repo, source, | ||
opts, buffered=False): | ||||
""" | ||||
Helper for incoming / gincoming. | ||||
displaychlist gets called with | ||||
(remoterepo, incomingchangesetlist, displayer) parameters, | ||||
and is supposed to contain only code that can't be unified. | ||||
""" | ||||
source, branches = parseurl(ui.expandpath(source), opts.get('branch')) | ||||
Matt Mackall
|
r14556 | other = peer(repo, opts, source) | ||
Brodie Rao
|
r14076 | ui.status(_('comparing with %s\n') % util.hidepassword(source)) | ||
Nicolas Dumazet
|
r12730 | revs, checkout = addbranchrevs(repo, other, branches, opts.get('rev')) | ||
if revs: | ||||
revs = [other.lookup(rev) for rev in revs] | ||||
Peter Arrenbrecht
|
r14161 | other, chlist, cleanupfn = bundlerepo.getremotechanges(ui, repo, other, | ||
revs, opts["bundle"], opts["force"]) | ||||
try: | ||||
if not chlist: | ||||
ui.status(_("no changes found\n")) | ||||
return subreporecurse() | ||||
Nicolas Dumazet
|
r12730 | |||
displayer = cmdutil.show_changeset(ui, other, opts, buffered) | ||||
displaychlist(other, chlist, displayer) | ||||
displayer.close() | ||||
finally: | ||||
Peter Arrenbrecht
|
r14161 | cleanupfn() | ||
Nicolas Dumazet
|
r12730 | subreporecurse() | ||
return 0 # exit code is zero since we found incoming changes | ||||
Martin Geisler
|
r12273 | def incoming(ui, repo, source, opts): | ||
Nicolas Dumazet
|
r12730 | def subreporecurse(): | ||
Erik Zielke
|
r12400 | ret = 1 | ||
if opts.get('subrepos'): | ||||
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): | ||
limit = cmdutil.loglimit(opts) | ||||
Martin Geisler
|
r12273 | if opts.get('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 | ||||
parents = [p for p in other.changelog.parents(n) if p != nullid] | ||||
if opts.get('no_merges') and len(parents) == 2: | ||||
continue | ||||
count += 1 | ||||
displayer.show(other[n]) | ||||
Nicolas Dumazet
|
r12730 | return _incoming(display, subreporecurse, ui, repo, source, opts) | ||
Martin Geisler
|
r12273 | |||
Nicolas Dumazet
|
r12735 | def _outgoing(ui, repo, dest, opts): | ||
dest = ui.expandpath(dest or 'default-push', dest or 'default') | ||||
dest, branches = parseurl(dest, opts.get('branch')) | ||||
Brodie Rao
|
r14076 | ui.status(_('comparing with %s\n') % util.hidepassword(dest)) | ||
Nicolas Dumazet
|
r12735 | revs, checkout = addbranchrevs(repo, repo, branches, opts.get('rev')) | ||
if revs: | ||||
Matt Harbison
|
r17198 | revs = [repo.lookup(rev) for rev in scmutil.revrange(repo, revs)] | ||
Nicolas Dumazet
|
r12735 | |||
Matt Mackall
|
r14556 | other = peer(repo, opts, dest) | ||
Pierre-Yves David
|
r18493 | outgoing = discovery.findcommonoutgoing(repo.unfiltered(), other, revs, | ||
Pierre-Yves David
|
r15837 | force=opts.get('force')) | ||
o = outgoing.missing | ||||
Nicolas Dumazet
|
r12735 | if not o: | ||
Patrick Mezard
|
r17248 | scmutil.nochangesfound(repo.ui, repo, outgoing.excluded) | ||
FUJIWARA Katsunori
|
r21050 | return o, other | ||
Nicolas Dumazet
|
r12735 | |||
Martin Geisler
|
r12271 | def outgoing(ui, repo, dest, opts): | ||
Erik Zielke
|
r12400 | def recurse(): | ||
ret = 1 | ||||
if opts.get('subrepos'): | ||||
ctx = repo[None] | ||||
for subpath in sorted(ctx.substate): | ||||
sub = ctx.sub(subpath) | ||||
ret = min(ret, sub.outgoing(ui, dest, opts)) | ||||
return ret | ||||
Martin Geisler
|
r12271 | limit = cmdutil.loglimit(opts) | ||
FUJIWARA Katsunori
|
r21050 | o, other = _outgoing(ui, repo, dest, opts) | ||
FUJIWARA Katsunori
|
r21049 | if not o: | ||
FUJIWARA Katsunori
|
r21051 | cmdutil.outgoinghooks(ui, repo, other, opts, o) | ||
Erik Zielke
|
r12400 | return recurse() | ||
Martin Geisler
|
r12271 | if opts.get('newest_first'): | ||
o.reverse() | ||||
displayer = cmdutil.show_changeset(ui, repo, opts) | ||||
count = 0 | ||||
for n in o: | ||||
if limit is not None and count >= limit: | ||||
break | ||||
parents = [p for p in repo.changelog.parents(n) if p != nullid] | ||||
if opts.get('no_merges') and len(parents) == 2: | ||||
continue | ||||
count += 1 | ||||
displayer.show(repo[n]) | ||||
displayer.close() | ||||
FUJIWARA Katsunori
|
r21051 | cmdutil.outgoinghooks(ui, repo, other, opts, o) | ||
Erik Zielke
|
r12400 | recurse() | ||
return 0 # exit code is zero since we found outgoing changes | ||||
Martin Geisler
|
r12271 | |||
Matt Mackall
|
r4917 | def revert(repo, node, choose): | ||
Matt Mackall
|
r2808 | """revert changes to revision in node without updating dirstate""" | ||
Benoit Boissinot
|
r10651 | return mergemod.update(repo, node, False, True, choose)[3] > 0 | ||
Matt Mackall
|
r2778 | |||
def verify(repo): | ||||
"""verify the consistency of a repository""" | ||||
Matt Harbison
|
r25591 | ret = verifymod.verify(repo) | ||
# 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 | ||||
revs = repo.revs("filelog(%s)", | ||||
util.pathto(repo.root, repo.getcwd(), '.hgsubstate')) | ||||
if revs: | ||||
repo.ui.status(_('checking subrepo links\n')) | ||||
for rev in revs: | ||||
ctx = repo[rev] | ||||
try: | ||||
for subpath in ctx.substate: | ||||
ret = ctx.sub(subpath).verify() or ret | ||||
except Exception: | ||||
repo.ui.warn(_('.hgsubstate is corrupt in revision %s\n') % | ||||
node.short(ctx.node())) | ||||
return ret | ||||
Matt Mackall
|
r11273 | |||
def remoteui(src, opts): | ||||
'build a remote ui from ui or repo and opts' | ||||
Augie Fackler
|
r14952 | if util.safehasattr(src, 'baseui'): # looks like a repository | ||
Matt Mackall
|
r11273 | 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 | ||||
# copy ssh-specific options | ||||
for o in 'ssh', 'remotecmd': | ||||
v = opts.get(o) or src.config('ui', o) | ||||
if v: | ||||
Mads Kiilerich
|
r20790 | dst.setconfig("ui", o, v, 'copied') | ||
Matt Mackall
|
r11273 | |||
# copy bundle-specific options | ||||
r = src.config('bundle', 'mainreporoot') | ||||
if r: | ||||
Mads Kiilerich
|
r20790 | dst.setconfig('bundle', 'mainreporoot', r, 'copied') | ||
Matt Mackall
|
r11273 | |||
Mads Kiilerich
|
r13192 | # copy selected local settings to the remote ui | ||
Mads Kiilerich
|
r13314 | for sect in ('auth', 'hostfingerprints', 'http_proxy'): | ||
Matt Mackall
|
r11273 | for key, val in src.configitems(sect): | ||
Mads Kiilerich
|
r20790 | dst.setconfig(sect, key, val, 'copied') | ||
Mads Kiilerich
|
r13192 | v = src.config('web', 'cacerts') | ||
Yuya Nishihara
|
r24290 | if v == '!': | ||
dst.setconfig('web', 'cacerts', v, 'copied') | ||||
elif v: | ||||
Mads Kiilerich
|
r20790 | dst.setconfig('web', 'cacerts', util.expandpath(v), 'copied') | ||
Matt Mackall
|
r11273 | |||
return dst | ||||