hg.py
584 lines
| 19.3 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 | |||
Matt Mackall
|
r3891 | from i18n import _ | ||
Ronny Pfannschmidt
|
r8109 | from lock import release | ||
Alexander Solovyov
|
r14064 | from node import hex, nullid | ||
David Soria Parra
|
r13604 | import localrepo, bundlerepo, httprepo, sshrepo, statichttprepo, bookmarks | ||
Matt Mackall
|
r15993 | import lock, util, extensions, error, node, scmutil | ||
Brodie Rao
|
r14076 | import cmdutil, discovery | ||
Benoit Boissinot
|
r10651 | import merge as mergemod | ||
import verify as verifymod | ||||
Simon Heimberg
|
r8312 | import errno, os, shutil | ||
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
|
r10365 | def addbranchrevs(lrepo, repo, branches, revs): | ||
Sune Foldager
|
r11322 | hashbranch, branches = branches | ||
if not hashbranch and not branches: | ||||
Sune Foldager
|
r10365 | return revs or None, revs and revs[0] or None | ||
Sune Foldager
|
r10380 | revs = revs and list(revs) or [] | ||
if not repo.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
|
r10365 | branchmap = repo.branchmap() | ||
Sune Foldager
|
r11322 | |||
Matt Mackall
|
r13047 | def primary(branch): | ||
if branch == '.': | ||||
Sune Foldager
|
r10365 | if not lrepo or not lrepo.local(): | ||
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, | ||
'file': _local, | ||||
'http': httprepo, | ||||
'https': httprepo, | ||||
'ssh': sshrepo, | ||||
'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: | ||||
return thing | ||||
Matt Mackall
|
r14605 | def islocal(repo): | ||
'''return true if repo or path is local''' | ||||
if isinstance(repo, str): | ||||
try: | ||||
return _peerlookup(repo).islocal(repo) | ||||
except AttributeError: | ||||
return False | ||||
return repo.local() | ||||
def repository(ui, path='', create=False): | ||||
"""return a repository object for the specified path""" | ||||
repo = _peerlookup(path).instance(ui, path, create) | ||||
ui = getattr(repo, "ui", ui) | ||||
for name, module in extensions.extensions(): | ||||
hook = getattr(module, 'reposetup', None) | ||||
if hook: | ||||
hook(ui, repo) | ||||
return repo | ||||
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) | ||
David Golub
|
r14737 | return repository(rui, path, create) | ||
Matt Mackall
|
r14554 | |||
Vadim Gelfer
|
r2719 | def defaultdest(source): | ||
'''return default destination of clone if none is given''' | ||||
return os.path.basename(os.path.normpath(source)) | ||||
Matt Mackall
|
r2774 | |||
Matt Mackall
|
r8807 | def share(ui, source, dest=None, update=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: | ||
srcrepo = source | ||||
origsource = source = srcrepo.url() | ||||
checkout = None | ||||
sharedpath = srcrepo.sharedpath # if our source is already sharing | ||||
Matt Mackall
|
r15381 | root = os.path.realpath(dest) | ||
Matt Mackall
|
r8800 | roothg = os.path.join(root, '.hg') | ||
if os.path.exists(roothg): | ||||
raise util.Abort(_('destination already exists')) | ||||
if not os.path.isdir(root): | ||||
os.mkdir(root) | ||||
Adrian Buehlmann
|
r13795 | util.makedir(roothg, notindexed=True) | ||
Matt Mackall
|
r8800 | |||
requirements = '' | ||||
try: | ||||
Dan Villiom Podlaski Christiansen
|
r14168 | requirements = srcrepo.opener.read('requires') | ||
Matt Mackall
|
r8800 | except IOError, inst: | ||
if inst.errno != errno.ENOENT: | ||||
raise | ||||
requirements += 'shared\n' | ||||
Dan Villiom Podlaski Christiansen
|
r14168 | util.writefile(os.path.join(roothg, 'requires'), requirements) | ||
util.writefile(os.path.join(roothg, 'sharedpath'), sharedpath) | ||||
Matt Mackall
|
r8800 | |||
Dan Villiom Podlaski Christiansen
|
r14156 | r = repository(ui, root) | ||
Matt Mackall
|
r8800 | default = srcrepo.ui.config('paths', 'default') | ||
if default: | ||||
Dan Villiom Podlaski Christiansen
|
r14156 | fp = r.opener("hgrc", "w", text=True) | ||
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) | ||||
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 | ||||
Pierre-Yves David
|
r15741 | srcpublishing = srcrepo.ui.configbool('phases', 'publish', True) | ||
Simon Heimberg
|
r15078 | for f in srcrepo.store.copylist(): | ||
Pierre-Yves David
|
r15741 | if srcpublishing and f.endswith('phaseroots'): | ||
continue | ||||
Simon Heimberg
|
r15078 | src = os.path.join(srcrepo.sharedpath, f) | ||
dst = os.path.join(destpath, f) | ||||
dstbase = os.path.dirname(dst) | ||||
if dstbase and not os.path.exists(dstbase): | ||||
os.mkdir(dstbase) | ||||
if os.path.exists(src): | ||||
if dst.endswith('data'): | ||||
# lock to avoid premature writing to the target | ||||
destlock = lock.lock(os.path.join(dstbase, "lock")) | ||||
hardlink, n = util.copyfiles(src, dst, hardlink) | ||||
num += n | ||||
if hardlink: | ||||
ui.debug("linked %d files\n" % num) | ||||
else: | ||||
ui.debug("copied %d files\n" % num) | ||||
return destlock | ||||
except: | ||||
release(destlock) | ||||
raise | ||||
Martin Geisler
|
r14607 | def clone(ui, peeropts, source, dest=None, pull=False, rev=None, | ||
update=True, stream=False, branch=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 | ||||
function. Returns a pair of repository objects, the source and | ||||
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) | ||||
pull: always pull from source repository, even in local case | ||||
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 | ||||
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) | ||
Martin Geisler
|
r14607 | srcrepo = repository(remoteui(ui, peeropts), source) | ||
Vadim Gelfer
|
r2719 | else: | ||
Martin Geisler
|
r14463 | srcrepo = source | ||
Nicolas Dumazet
|
r11818 | branch = (None, branch or []) | ||
Martin Geisler
|
r14463 | origsource = source = srcrepo.url() | ||
rev, checkout = addbranchrevs(srcrepo, srcrepo, branch, rev) | ||||
Vadim Gelfer
|
r2719 | |||
Vadim Gelfer
|
r2597 | if dest is None: | ||
Vadim Gelfer
|
r2719 | dest = defaultdest(source) | ||
Thomas Arendsen Hein
|
r3841 | 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 | |||
if os.path.exists(dest): | ||||
Steve Borho
|
r7927 | if not os.path.isdir(dest): | ||
raise util.Abort(_("destination '%s' already exists") % dest) | ||||
elif os.listdir(dest): | ||||
raise util.Abort(_("destination '%s' is not empty") % dest) | ||||
Vadim Gelfer
|
r2597 | |||
class DirCleanup(object): | ||||
def __init__(self, dir_): | ||||
self.rmtree = shutil.rmtree | ||||
self.dir_ = dir_ | ||||
def close(self): | ||||
self.dir_ = None | ||||
Ronny Pfannschmidt
|
r8110 | def cleanup(self): | ||
Vadim Gelfer
|
r2597 | if self.dir_: | ||
self.rmtree(self.dir_, True) | ||||
Matt Mackall
|
r15908 | srclock = destlock = dircleanup = None | ||
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): | ||
Martin Geisler
|
r14463 | dircleanup = DirCleanup(dest) | ||
Vadim Gelfer
|
r2597 | |||
Matt Mackall
|
r4915 | copy = False | ||
Martin Geisler
|
r14463 | if srcrepo.cancopy() and islocal(dest): | ||
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 | ||||
Martin Geisler
|
r14463 | dircleanup.dir_ = hgdir | ||
Matt Mackall
|
r5569 | try: | ||
Martin Geisler
|
r14463 | destpath = hgdir | ||
util.makedir(destpath, notindexed=True) | ||||
Matt Mackall
|
r5569 | except OSError, inst: | ||
if inst.errno == errno.EEXIST: | ||||
Martin Geisler
|
r14463 | dircleanup.close() | ||
Matt Mackall
|
r5569 | raise util.Abort(_("destination '%s' already exists") | ||
% dest) | ||||
raise | ||||
Vadim Gelfer
|
r2597 | |||
Simon Heimberg
|
r15078 | destlock = copystore(ui, srcrepo, destpath) | ||
Vadim Gelfer
|
r2597 | |||
Matt Mackall
|
r4915 | # we need to re-init the repo after manually copying the data | ||
# into it | ||||
Martin Geisler
|
r14607 | destrepo = repository(remoteui(ui, 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: | ||
Martin Geisler
|
r14607 | destrepo = repository(remoteui(ui, peeropts), dest, | ||
create=True) | ||||
Matt Mackall
|
r5569 | except OSError, inst: | ||
if inst.errno == errno.EEXIST: | ||||
Martin Geisler
|
r14463 | dircleanup.close() | ||
Matt Mackall
|
r5569 | raise util.Abort(_("destination '%s' already exists") | ||
% dest) | ||||
raise | ||||
Vadim Gelfer
|
r2597 | |||
Matt Mackall
|
r4915 | revs = None | ||
if rev: | ||||
Peter Arrenbrecht
|
r14552 | if not srcrepo.capable('lookup'): | ||
Martin Geisler
|
r9171 | raise util.Abort(_("src repository does not support " | ||
"revision lookup and so doesn't " | ||||
"support clone by revision")) | ||||
Martin Geisler
|
r14463 | revs = [srcrepo.lookup(r) for r in rev] | ||
Brett Carter
|
r8417 | checkout = revs[0] | ||
Martin Geisler
|
r14463 | if destrepo.local(): | ||
destrepo.clone(srcrepo, heads=revs, stream=stream) | ||||
elif srcrepo.local(): | ||||
srcrepo.push(destrepo, revs=revs) | ||||
Matt Mackall
|
r4915 | else: | ||
raise util.Abort(_("clone from remote to remote not supported")) | ||||
Vadim Gelfer
|
r2597 | |||
Martin Geisler
|
r14463 | if dircleanup: | ||
dircleanup.close() | ||||
Vadim Gelfer
|
r2597 | |||
Arne Babenhauserheide
|
r15590 | # clone all bookmarks | ||
if destrepo.local() and srcrepo.capable("pushkey"): | ||||
rb = srcrepo.listkeys('bookmarks') | ||||
for k, n in rb.iteritems(): | ||||
try: | ||||
m = destrepo.lookup(n) | ||||
destrepo._bookmarks[k] = m | ||||
except error.RepoLookupError: | ||||
pass | ||||
if rb: | ||||
bookmarks.write(destrepo) | ||||
elif srcrepo.local() and destrepo.capable("pushkey"): | ||||
for k, n in srcrepo._bookmarks.iteritems(): | ||||
destrepo.pushkey('bookmarks', k, '', hex(n)) | ||||
Martin Geisler
|
r14463 | if destrepo.local(): | ||
fp = destrepo.opener("hgrc", "w", text=True) | ||||
Matt Mackall
|
r4915 | fp.write("[paths]\n") | ||
Augie Fackler
|
r15552 | u = util.url(abspath) | ||
u.passwd = None | ||||
defaulturl = str(u) | ||||
fp.write("default = %s\n" % defaulturl) | ||||
Matt Mackall
|
r4915 | fp.close() | ||
Benoit Boissinot
|
r5185 | |||
Augie Fackler
|
r15552 | destrepo.ui.setconfig('paths', 'default', defaulturl) | ||
Matt Mackall
|
r8814 | |||
Matt Mackall
|
r4915 | if update: | ||
Bryan O'Sullivan
|
r6526 | if update is not True: | ||
checkout = update | ||||
Martin Geisler
|
r14463 | if srcrepo.local(): | ||
checkout = srcrepo.lookup(update) | ||||
Dirkjan Ochtman
|
r7045 | for test in (checkout, 'default', 'tip'): | ||
Matt Mackall
|
r9423 | if test is None: | ||
continue | ||||
Alexis S. L. Carvalho
|
r5248 | try: | ||
Martin Geisler
|
r14463 | uprev = destrepo.lookup(test) | ||
Dirkjan Ochtman
|
r7045 | break | ||
Matt Mackall
|
r9423 | except error.RepoLookupError: | ||
Dirkjan Ochtman
|
r7045 | continue | ||
Martin Geisler
|
r14463 | bn = destrepo[uprev].branch() | ||
destrepo.ui.status(_("updating to branch %s\n") % bn) | ||||
_update(destrepo, uprev) | ||||
Vadim Gelfer
|
r2597 | |||
Martin Geisler
|
r14463 | return srcrepo, destrepo | ||
Matt Mackall
|
r4915 | finally: | ||
Matt Mackall
|
r15908 | release(srclock, destlock) | ||
Martin Geisler
|
r14463 | if dircleanup is not None: | ||
dircleanup.cleanup() | ||||
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 | |||
Matt Mackall
|
r2808 | def update(repo, node): | ||
"""update the working directory to node, merging linear changes""" | ||||
Benoit Boissinot
|
r10651 | stats = mergemod.update(repo, node, False, False, None) | ||
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""" | ||
Benoit Boissinot
|
r10651 | stats = mergemod.update(repo, node, False, True, None) | ||
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) | ||||
# XXX once graphlog extension makes it into core, | ||||
# should be replaced by a if graph/else | ||||
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: | ||||
revs = [repo.lookup(rev) for rev in revs] | ||||
Matt Mackall
|
r14556 | other = peer(repo, opts, dest) | ||
Pierre-Yves David
|
r15837 | outgoing = discovery.findcommonoutgoing(repo, other, revs, | ||
force=opts.get('force')) | ||||
o = outgoing.missing | ||||
Nicolas Dumazet
|
r12735 | if not o: | ||
Matt Mackall
|
r15993 | scmutil.nochangesfound(repo.ui, outgoing.excluded) | ||
Nicolas Dumazet
|
r12735 | return None | ||
Peter Arrenbrecht
|
r14073 | return o | ||
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) | ||
Nicolas Dumazet
|
r12735 | o = _outgoing(ui, repo, dest, opts) | ||
if o is None: | ||||
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() | ||||
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""" | ||||
Benoit Boissinot
|
r10651 | return verifymod.verify(repo) | ||
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: | ||||
dst.setconfig("ui", o, v) | ||||
# copy bundle-specific options | ||||
r = src.config('bundle', 'mainreporoot') | ||||
if r: | ||||
dst.setconfig('bundle', 'mainreporoot', r) | ||||
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): | ||
dst.setconfig(sect, key, val) | ||||
Mads Kiilerich
|
r13192 | v = src.config('web', 'cacerts') | ||
if v: | ||||
Eduard-Cristian Stefan
|
r13231 | dst.setconfig('web', 'cacerts', util.expandpath(v)) | ||
Matt Mackall
|
r11273 | |||
return dst | ||||