hg.py
290 lines
| 9.0 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 | # | ||
# This software may be used and distributed according to the terms | ||||
# of the GNU General Public License, incorporated herein by reference. | ||||
mpm@selenic.com
|
r1089 | from node import * | ||
from repo import * | ||||
Matt Mackall
|
r3891 | from i18n import _ | ||
Matt Mackall
|
r3877 | import localrepo, bundlerepo, httprepo, sshrepo, statichttprepo | ||
Matt Mackall
|
r4478 | import errno, lock, os, shutil, util, cmdutil | ||
Matt Mackall
|
r3877 | import merge as _merge | ||
import verify as _verify | ||||
mpm@selenic.com
|
r0 | |||
Vadim Gelfer
|
r2740 | def _local(path): | ||
Brendan Cully
|
r3195 | return (os.path.isfile(util.drop_scheme('file', path)) and | ||
Vadim Gelfer
|
r2768 | bundlerepo or localrepo) | ||
Vadim Gelfer
|
r2469 | |||
Vadim Gelfer
|
r2472 | schemes = { | ||
Vadim Gelfer
|
r2740 | 'bundle': bundlerepo, | ||
'file': _local, | ||||
'hg': httprepo, | ||||
'http': httprepo, | ||||
'https': httprepo, | ||||
'old-http': statichttprepo, | ||||
'ssh': sshrepo, | ||||
'static-http': statichttprepo, | ||||
Vadim Gelfer
|
r2469 | } | ||
Vadim Gelfer
|
r2740 | def _lookup(path): | ||
scheme = 'file' | ||||
if path: | ||||
c = path.find(':') | ||||
if c > 0: | ||||
scheme = path[:c] | ||||
thing = schemes.get(scheme) or schemes['file'] | ||||
try: | ||||
return thing(path) | ||||
except TypeError: | ||||
return thing | ||||
Matt Mackall
|
r2775 | |||
Vadim Gelfer
|
r2719 | def islocal(repo): | ||
'''return true if repo or path is local''' | ||||
if isinstance(repo, str): | ||||
Vadim Gelfer
|
r2740 | try: | ||
return _lookup(repo).islocal(repo) | ||||
except AttributeError: | ||||
return False | ||||
Vadim Gelfer
|
r2719 | return repo.local() | ||
Vadim Gelfer
|
r2847 | repo_setup_hooks = [] | ||
Brendan Cully
|
r3195 | def repository(ui, path='', create=False): | ||
Matt Mackall
|
r2774 | """return a repository object for the specified path""" | ||
Vadim Gelfer
|
r2847 | repo = _lookup(path).instance(ui, path, create) | ||
Alexis S. L. Carvalho
|
r4074 | ui = getattr(repo, "ui", ui) | ||
Vadim Gelfer
|
r2847 | for hook in repo_setup_hooks: | ||
hook(ui, repo) | ||||
return repo | ||||
Vadim Gelfer
|
r2597 | |||
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 | |||
Vadim Gelfer
|
r2613 | def clone(ui, source, dest=None, pull=False, rev=None, update=True, | ||
stream=False): | ||||
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 | ||||
destination is local repository | ||||
""" | ||||
Matt Mackall
|
r4478 | |||
origsource = source | ||||
source, rev = cmdutil.parseurl(ui.expandpath(source), rev) | ||||
Vadim Gelfer
|
r2719 | if isinstance(source, str): | ||
src_repo = repository(ui, source) | ||||
else: | ||||
src_repo = source | ||||
source = src_repo.url() | ||||
Vadim Gelfer
|
r2597 | if dest is None: | ||
Vadim Gelfer
|
r2719 | dest = defaultdest(source) | ||
Thomas Arendsen Hein
|
r3841 | ui.status(_("destination directory: %s\n") % dest) | ||
Vadim Gelfer
|
r2719 | |||
def localpath(path): | ||||
if path.startswith('file://'): | ||||
return path[7:] | ||||
if path.startswith('file:'): | ||||
return path[5:] | ||||
return path | ||||
dest = localpath(dest) | ||||
source = localpath(source) | ||||
Vadim Gelfer
|
r2597 | |||
if os.path.exists(dest): | ||||
Thomas Arendsen Hein
|
r3072 | raise util.Abort(_("destination '%s' already exists") % dest) | ||
Vadim Gelfer
|
r2597 | |||
class DirCleanup(object): | ||||
def __init__(self, dir_): | ||||
self.rmtree = shutil.rmtree | ||||
self.dir_ = dir_ | ||||
def close(self): | ||||
self.dir_ = None | ||||
def __del__(self): | ||||
if self.dir_: | ||||
self.rmtree(self.dir_, True) | ||||
dir_cleanup = None | ||||
Benoit Boissinot
|
r3849 | if islocal(dest): | ||
dir_cleanup = DirCleanup(dest) | ||||
Vadim Gelfer
|
r2597 | |||
Matt Mackall
|
r4478 | abspath = origsource | ||
Vadim Gelfer
|
r2597 | copy = False | ||
Benoit Boissinot
|
r3849 | if src_repo.local() and islocal(dest): | ||
Matt Mackall
|
r4478 | abspath = os.path.abspath(origsource) | ||
Vadim Gelfer
|
r2597 | copy = not pull and not rev | ||
src_lock, dest_lock = None, None | ||||
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 | ||||
src_lock = src_repo.lock() | ||||
except lock.LockException: | ||||
copy = False | ||||
if copy: | ||||
Benoit Boissinot
|
r3851 | def force_copy(src, dst): | ||
try: | ||||
util.copyfiles(src, dst) | ||||
except OSError, inst: | ||||
if inst.errno != errno.ENOENT: | ||||
raise | ||||
Benoit Boissinot
|
r3791 | src_store = os.path.realpath(src_repo.spath) | ||
Benoit Boissinot
|
r3849 | if not os.path.exists(dest): | ||
os.mkdir(dest) | ||||
Benoit Boissinot
|
r3853 | dest_path = os.path.realpath(os.path.join(dest, ".hg")) | ||
Benoit Boissinot
|
r3849 | os.mkdir(dest_path) | ||
Benoit Boissinot
|
r3853 | if src_repo.spath != src_repo.path: | ||
dest_store = os.path.join(dest_path, "store") | ||||
os.mkdir(dest_store) | ||||
else: | ||||
dest_store = dest_path | ||||
Benoit Boissinot
|
r3851 | # copy the requires file | ||
force_copy(src_repo.join("requires"), | ||||
os.path.join(dest_path, "requires")) | ||||
# we lock here to avoid premature writing to the target | ||||
Benoit Boissinot
|
r3791 | dest_lock = lock.lock(os.path.join(dest_store, "lock")) | ||
Vadim Gelfer
|
r2597 | |||
Benoit Boissinot
|
r3713 | files = ("data", | ||
"00manifest.d", "00manifest.i", | ||||
"00changelog.d", "00changelog.i") | ||||
for f in files: | ||||
Benoit Boissinot
|
r3791 | src = os.path.join(src_store, f) | ||
dst = os.path.join(dest_store, f) | ||||
Benoit Boissinot
|
r3851 | force_copy(src, dst) | ||
Vadim Gelfer
|
r2597 | |||
Benoit Boissinot
|
r2631 | # we need to re-init the repo after manually copying the data | ||
# into it | ||||
Vadim Gelfer
|
r2597 | dest_repo = repository(ui, dest) | ||
else: | ||||
Benoit Boissinot
|
r3849 | dest_repo = repository(ui, dest, create=True) | ||
Vadim Gelfer
|
r2597 | revs = None | ||
if rev: | ||||
Eric Hopper
|
r3448 | if 'lookup' not in src_repo.capabilities: | ||
raise util.Abort(_("src repository does not support revision " | ||||
"lookup and so doesn't support clone by " | ||||
"revision")) | ||||
Vadim Gelfer
|
r2597 | revs = [src_repo.lookup(r) for r in rev] | ||
if dest_repo.local(): | ||||
Vadim Gelfer
|
r2613 | dest_repo.clone(src_repo, heads=revs, stream=stream) | ||
Vadim Gelfer
|
r2597 | elif src_repo.local(): | ||
src_repo.push(dest_repo, revs=revs) | ||||
else: | ||||
raise util.Abort(_("clone from remote to remote not supported")) | ||||
if src_lock: | ||||
src_lock.release() | ||||
Benoit Boissinot
|
r5185 | if dir_cleanup: | ||
dir_cleanup.close() | ||||
Vadim Gelfer
|
r2597 | if dest_repo.local(): | ||
fp = dest_repo.opener("hgrc", "w", text=True) | ||||
fp.write("[paths]\n") | ||||
fp.write("default = %s\n" % abspath) | ||||
fp.close() | ||||
if dest_lock: | ||||
dest_lock.release() | ||||
if update: | ||||
Matt Mackall
|
r4477 | try: | ||
checkout = dest_repo.lookup("default") | ||||
except: | ||||
checkout = dest_repo.changelog.tip() | ||||
_update(dest_repo, checkout) | ||||
Vadim Gelfer
|
r2597 | |||
return src_repo, dest_repo | ||||
Matt Mackall
|
r2775 | |||
Matt Mackall
|
r3316 | def _showstats(repo, stats): | ||
stats = ((stats[0], _("updated")), | ||||
(stats[1], _("merged")), | ||||
(stats[2], _("removed")), | ||||
(stats[3], _("unresolved"))) | ||||
note = ", ".join([_("%d files %s") % s for s in stats]) | ||||
repo.ui.status("%s\n" % note) | ||||
def _update(repo, node): return update(repo, node) | ||||
Matt Mackall
|
r2808 | def update(repo, node): | ||
"""update the working directory to node, merging linear changes""" | ||||
Thomas Arendsen Hein
|
r3869 | pl = repo.parents() | ||
Matt Mackall
|
r3316 | stats = _merge.update(repo, node, False, False, None, None) | ||
_showstats(repo, stats) | ||||
if stats[3]: | ||||
repo.ui.status(_("There are unresolved merges with" | ||||
" locally modified files.\n")) | ||||
Thomas Arendsen Hein
|
r3869 | if stats[1]: | ||
repo.ui.status(_("You can finish the partial merge using:\n")) | ||||
else: | ||||
repo.ui.status(_("You can redo the full merge using:\n")) | ||||
# len(pl)==1, otherwise _merge.update() would have raised util.Abort: | ||||
repo.ui.status(_(" hg update %s\n hg update %s\n") | ||||
% (pl[0].rev(), repo.changectx(node).rev())) | ||||
Matt Mackall
|
r3316 | return stats[3] | ||
Matt Mackall
|
r2775 | |||
Matt Mackall
|
r2808 | def clean(repo, node, wlock=None, show_stats=True): | ||
"""forcibly switch the working directory to node, clobbering changes""" | ||||
Matt Mackall
|
r3316 | stats = _merge.update(repo, node, False, True, None, wlock) | ||
if show_stats: _showstats(repo, stats) | ||||
return stats[3] | ||||
Matt Mackall
|
r2775 | |||
Matt Mackall
|
r2808 | def merge(repo, node, force=None, remind=True, wlock=None): | ||
"""branch merge with node, resolving changes""" | ||||
Matt Mackall
|
r3316 | stats = _merge.update(repo, node, True, force, False, wlock) | ||
_showstats(repo, stats) | ||||
if stats[3]: | ||||
pl = repo.parents() | ||||
repo.ui.status(_("There are unresolved merges," | ||||
" you can redo the full merge using:\n" | ||||
" hg update -C %s\n" | ||||
Marcos Chaves
|
r3679 | " hg merge %s\n") | ||
Thomas Arendsen Hein
|
r3680 | % (pl[0].rev(), pl[1].rev())) | ||
Matt Mackall
|
r3316 | elif remind: | ||
repo.ui.status(_("(branch merge, don't forget to commit)\n")) | ||||
return stats[3] | ||||
Matt Mackall
|
r2808 | |||
Matt Mackall
|
r2812 | def revert(repo, node, choose, wlock): | ||
Matt Mackall
|
r2808 | """revert changes to revision in node without updating dirstate""" | ||
Matt Mackall
|
r3316 | return _merge.update(repo, node, False, True, choose, wlock)[3] | ||
Matt Mackall
|
r2778 | |||
def verify(repo): | ||||
"""verify the consistency of a repository""" | ||||
return _verify.verify(repo) | ||||