##// END OF EJS Templates
merge: minor simplification
merge: minor simplification

File last commit:

r2841:e3fb4223 default
r2842:06c05c67 default
Show More
localrepo.py
1758 lines | 65.1 KiB | text/x-python | PythonLexer
mpm@selenic.com
Break apart hg.py...
r1089 # localrepo.py - read/write repository class for mercurial
#
# Copyright 2005 Matt Mackall <mpm@selenic.com>
#
# This software may be used and distributed according to the terms
# of the GNU General Public License, incorporated herein by reference.
mpm@selenic.com
localrepo: more import/namespace tidying
r1100 from node import *
Benoit Boissinot
i18n first part: make '_' available for files who need it
r1400 from i18n import gettext as _
mpm@selenic.com
Break apart hg.py...
r1089 from demandload import *
Vadim Gelfer
add support for streaming clone....
r2612 import repo
Vadim Gelfer
fix race in localrepo.addchangegroup....
r1998 demandload(globals(), "appendfile changegroup")
Vadim Gelfer
add support for streaming clone....
r2612 demandload(globals(), "changelog dirstate filelog manifest context")
Vadim Gelfer
support hooks written in python....
r2155 demandload(globals(), "re lock transaction tempfile stat mdiff errno ui")
Vadim Gelfer
add support for streaming clone....
r2612 demandload(globals(), "os revlog time util")
mpm@selenic.com
Break apart hg.py...
r1089
Vadim Gelfer
add support for streaming clone....
r2612 class localrepository(repo.repository):
Vadim Gelfer
extend network protocol to stop clients from locking servers...
r2439 capabilities = ()
mason@suse.com
Automatic nesting into running transactions in the same repository....
r1806 def __del__(self):
self.transhandle = None
Thomas Arendsen Hein
Create local ui object per repository, so .hg/hgrc don't get mixed....
r1839 def __init__(self, parentui, path=None, create=0):
Vadim Gelfer
add support for streaming clone....
r2612 repo.repository.__init__(self)
mpm@selenic.com
Separate out old-http support...
r1101 if not path:
p = os.getcwd()
while not os.path.isdir(os.path.join(p, ".hg")):
oldp = p
p = os.path.dirname(p)
Thomas Arendsen Hein
Cleanup of indentation, spacing, newlines, strings and line length
r1615 if p == oldp:
raise repo.RepoError(_("no repo found"))
mpm@selenic.com
Separate out old-http support...
r1101 path = p
self.path = os.path.join(path, ".hg")
mpm@selenic.com
Break apart hg.py...
r1089
mpm@selenic.com
Separate out old-http support...
r1101 if not create and not os.path.isdir(self.path):
Thomas Arendsen Hein
Better error message (without /.hg appended) when repository is not found....
r1588 raise repo.RepoError(_("repository %s not found") % path)
mpm@selenic.com
Break apart hg.py...
r1089
self.root = os.path.abspath(path)
Vadim Gelfer
fix backtrace printed when cannot get lock....
r2016 self.origroot = path
Thomas Arendsen Hein
Create local ui object per repository, so .hg/hgrc don't get mixed....
r1839 self.ui = ui.ui(parentui=parentui)
mpm@selenic.com
localrepo: minor opener usage restructuring
r1102 self.opener = util.opener(self.path)
self.wopener = util.opener(self.root)
mason@suse.com
Implement revlogng....
r2072
try:
self.ui.readconfig(self.join("hgrc"), self.root)
except IOError:
pass
v = self.ui.revlogopts
mason@suse.com
Use revlogng and inlined data files by default...
r2222 self.revlogversion = int(v.get('format', revlog.REVLOG_DEFAULT_FORMAT))
Thomas Arendsen Hein
Show repo's revlog format on verify only if it doesn't match the default format....
r2152 self.revlogv1 = self.revlogversion != revlog.REVLOGV0
mason@suse.com
Use revlogng and inlined data files by default...
r2222 fl = v.get('flags', None)
mason@suse.com
Implement data inlined with the index file...
r2073 flags = 0
mason@suse.com
Use revlogng and inlined data files by default...
r2222 if fl != None:
for x in fl.split():
flags |= revlog.flagstr(x)
elif self.revlogv1:
flags = revlog.REVLOG_DEFAULT_FLAGS
mason@suse.com
Implement data inlined with the index file...
r2073
v = self.revlogversion | flags
self.manifest = manifest.manifest(self.opener, v)
self.changelog = changelog.changelog(self.opener, v)
mason@suse.com
Implement revlogng....
r2072
mason@suse.com
Implement data inlined with the index file...
r2073 # the changelog might not have the inline index flag
# on. If the format of the changelog is the same as found in
# .hgrc, apply any flags found in the .hgrc as well.
# Otherwise, just version from the changelog
v = self.changelog.version
if v == self.revlogversion:
v |= flags
self.revlogversion = v
mpm@selenic.com
Break apart hg.py...
r1089 self.tagscache = None
self.nodetagscache = None
mpm@selenic.com
Add file encoding/decoding support
r1258 self.encodepats = None
self.decodepats = None
mason@suse.com
Automatic nesting into running transactions in the same repository....
r1806 self.transhandle = None
mpm@selenic.com
Break apart hg.py...
r1089
Thomas Arendsen Hein
Create [web] section with short username as contact on hg init and hg clone.
r1133 if create:
Benoit Boissinot
coding style: use spaces instead of tabs
r2578 if not os.path.exists(path):
os.mkdir(path)
Thomas Arendsen Hein
Create [web] section with short username as contact on hg init and hg clone.
r1133 os.mkdir(self.path)
os.mkdir(self.join("data"))
Thomas Arendsen Hein
Create local ui object per repository, so .hg/hgrc don't get mixed....
r1839 self.dirstate = dirstate.dirstate(self.opener, self.ui, self.root)
Vadim Gelfer
support hooks written in python....
r2155
Vadim Gelfer
hooks: add url to changegroup, incoming, prechangegroup, pretxnchangegroup hooks...
r2673 def url(self):
return 'file:' + self.root
Vadim Gelfer
make hook code nicer....
r1718 def hook(self, name, throw=False, **args):
Vadim Gelfer
support hooks written in python....
r2155 def callhook(hname, funcname):
'''call python hook. hook is callable object, looked up as
name in python module. if callable returns "true", hook
Vadim Gelfer
reverse sense of return value from python hooks....
r2221 fails, else passes. if hook raises exception, treated as
hook failure. exception propagates if throw is "true".
reason for "true" meaning "hook failed" is so that
unmodified commands (e.g. mercurial.commands.update) can
be run as hooks without wrappers to convert return values.'''
Vadim Gelfer
support hooks written in python....
r2155
self.ui.note(_("calling hook %s: %s\n") % (hname, funcname))
d = funcname.rfind('.')
if d == -1:
raise util.Abort(_('%s hook is invalid ("%s" not in a module)')
% (hname, funcname))
modname = funcname[:d]
try:
obj = __import__(modname)
except ImportError:
Benoit Boissinot
allow use of extensions in python hooks...
r2581 try:
# extensions are loaded with hgext_ prefix
obj = __import__("hgext_%s" % modname)
except ImportError:
raise util.Abort(_('%s hook is invalid '
'(import of "%s" failed)') %
(hname, modname))
Vadim Gelfer
support hooks written in python....
r2155 try:
for p in funcname.split('.')[1:]:
obj = getattr(obj, p)
except AttributeError, err:
raise util.Abort(_('%s hook is invalid '
'("%s" is not defined)') %
(hname, funcname))
if not callable(obj):
raise util.Abort(_('%s hook is invalid '
'("%s" is not callable)') %
(hname, funcname))
try:
Vadim Gelfer
fix minor bugs in localrepo.hook.
r2190 r = obj(ui=self.ui, repo=self, hooktype=name, **args)
Vadim Gelfer
support hooks written in python....
r2155 except (KeyboardInterrupt, util.SignalInterrupt):
raise
except Exception, exc:
if isinstance(exc, util.Abort):
self.ui.warn(_('error: %s hook failed: %s\n') %
(hname, exc.args[0] % exc.args[1:]))
else:
self.ui.warn(_('error: %s hook raised an exception: '
'%s\n') % (hname, exc))
if throw:
raise
Vadim Gelfer
add ui.print_exc(), make all traceback printing central.
r2335 self.ui.print_exc()
Vadim Gelfer
reverse sense of return value from python hooks....
r2221 return True
if r:
Vadim Gelfer
support hooks written in python....
r2155 if throw:
raise util.Abort(_('%s hook failed') % hname)
Vadim Gelfer
reverse sense of return value from python hooks....
r2221 self.ui.warn(_('warning: %s hook failed\n') % hname)
Vadim Gelfer
support hooks written in python....
r2155 return r
Benoit Boissinot
allow multiples hook...
r1480 def runhook(name, cmd):
self.ui.note(_("running hook %s: %s\n") % (name, cmd))
Vadim Gelfer
remove non-prefixed environment variables from hooks.
r2288 env = dict([('HG_' + k.upper(), v) for k, v in args.iteritems()])
Vadim Gelfer
merge util.esystem and util.system.
r1882 r = util.system(cmd, environ=env, cwd=self.root)
mpm@selenic.com
Break apart hg.py...
r1089 if r:
Vadim Gelfer
make hook code nicer....
r1718 desc, r = util.explain_exit(r)
if throw:
raise util.Abort(_('%s hook %s') % (name, desc))
Vadim Gelfer
reverse sense of return value from python hooks....
r2221 self.ui.warn(_('warning: %s hook %s\n') % (name, desc))
return r
Benoit Boissinot
allow multiples hook...
r1480
Vadim Gelfer
reverse sense of return value from python hooks....
r2221 r = False
Thomas Arendsen Hein
Call hooks in alphabetical (=defined) order, of course still grouped by type....
r1838 hooks = [(hname, cmd) for hname, cmd in self.ui.configitems("hooks")
if hname.split(".", 1)[0] == name and cmd]
hooks.sort()
for hname, cmd in hooks:
Vadim Gelfer
support hooks written in python....
r2155 if cmd.startswith('python:'):
Vadim Gelfer
reverse sense of return value from python hooks....
r2221 r = callhook(hname, cmd[7:].strip()) or r
Vadim Gelfer
support hooks written in python....
r2155 else:
Vadim Gelfer
reverse sense of return value from python hooks....
r2221 r = runhook(hname, cmd) or r
Benoit Boissinot
allow multiples hook...
r1480 return r
mpm@selenic.com
Break apart hg.py...
r1089
Vadim Gelfer
move most of tag code to localrepository class.
r2601 tag_disallowed = ':\r\n'
def tag(self, name, node, local=False, message=None, user=None, date=None):
'''tag a revision with a symbolic name.
if local is True, the tag is stored in a per-repository file.
otherwise, it is stored in the .hgtags file, and a new
changeset is committed with the change.
keyword arguments:
local: whether to store tag in non-version-controlled file
(default False)
message: commit message to use if committing
user: name of user to use if committing
date: date tuple to use if committing'''
for c in self.tag_disallowed:
if c in name:
raise util.Abort(_('%r cannot be used in a tag name') % c)
self.hook('pretag', throw=True, node=node, tag=name, local=local)
if local:
self.opener('localtags', 'a').write('%s %s\n' % (node, name))
self.hook('tag', node=node, tag=name, local=local)
return
for x in self.changes():
if '.hgtags' in x:
raise util.Abort(_('working copy of .hgtags is changed '
'(please commit .hgtags manually)'))
self.wfile('.hgtags', 'ab').write('%s %s\n' % (node, name))
if self.dirstate.state('.hgtags') == '?':
self.add(['.hgtags'])
if not message:
message = _('Added tag %s for changeset %s') % (name, node)
self.commit(['.hgtags'], message, user, date)
self.hook('tag', node=node, tag=name, local=local)
mpm@selenic.com
Break apart hg.py...
r1089 def tags(self):
'''return a mapping of tag to node'''
if not self.tagscache:
self.tagscache = {}
Benoit Boissinot
add checking for invalid entries in tag files...
r1986 def parsetag(line, context):
if not line:
return
s = l.split(" ", 1)
if len(s) != 2:
Vadim Gelfer
fix parsing of tags. make parse errors useful. add new tag tests....
r2320 self.ui.warn(_("%s: cannot parse entry\n") % context)
Benoit Boissinot
add checking for invalid entries in tag files...
r1986 return
node, key = s
Vadim Gelfer
fix parsing of tags. make parse errors useful. add new tag tests....
r2320 key = key.strip()
Benoit Boissinot
add checking for invalid entries in tag files...
r1986 try:
bin_n = bin(node)
except TypeError:
Vadim Gelfer
fix parsing of tags. make parse errors useful. add new tag tests....
r2320 self.ui.warn(_("%s: node '%s' is not well formed\n") %
(context, node))
Benoit Boissinot
add checking for invalid entries in tag files...
r1986 return
if bin_n not in self.changelog.nodemap:
Vadim Gelfer
fix parsing of tags. make parse errors useful. add new tag tests....
r2320 self.ui.warn(_("%s: tag '%s' refers to unknown node\n") %
(context, key))
Benoit Boissinot
add checking for invalid entries in tag files...
r1986 return
Vadim Gelfer
fix parsing of tags. make parse errors useful. add new tag tests....
r2320 self.tagscache[key] = bin_n
Benoit Boissinot
add checking for invalid entries in tag files...
r1986
Vadim Gelfer
fix parsing of tags. make parse errors useful. add new tag tests....
r2320 # read the tags file from each head, ending with the tip,
Benoit Boissinot
add checking for invalid entries in tag files...
r1986 # and add each tag found to the map, with "newer" ones
# taking precedence
Vadim Gelfer
fix parsing of tags. make parse errors useful. add new tag tests....
r2320 heads = self.heads()
heads.reverse()
Benoit Boissinot
add checking for invalid entries in tag files...
r1986 fl = self.file(".hgtags")
Vadim Gelfer
fix parsing of tags. make parse errors useful. add new tag tests....
r2320 for node in heads:
change = self.changelog.read(node)
rev = self.changelog.rev(node)
fn, ff = self.manifest.find(change[0], '.hgtags')
if fn is None: continue
Benoit Boissinot
add checking for invalid entries in tag files...
r1986 count = 0
Vadim Gelfer
fix parsing of tags. make parse errors useful. add new tag tests....
r2320 for l in fl.read(fn).splitlines():
Benoit Boissinot
add checking for invalid entries in tag files...
r1986 count += 1
Vadim Gelfer
fix parsing of tags. make parse errors useful. add new tag tests....
r2320 parsetag(l, _(".hgtags (rev %d:%s), line %d") %
(rev, short(node), count))
mpm@selenic.com
Break apart hg.py...
r1089 try:
f = self.opener("localtags")
Benoit Boissinot
add checking for invalid entries in tag files...
r1986 count = 0
mpm@selenic.com
Break apart hg.py...
r1089 for l in f:
Benoit Boissinot
add checking for invalid entries in tag files...
r1986 count += 1
Vadim Gelfer
fix parsing of tags. make parse errors useful. add new tag tests....
r2320 parsetag(l, _("localtags, line %d") % count)
mpm@selenic.com
Break apart hg.py...
r1089 except IOError:
pass
self.tagscache['tip'] = self.changelog.tip()
return self.tagscache
def tagslist(self):
'''return a list of tags ordered by revision'''
l = []
for t, n in self.tags().items():
try:
r = self.changelog.rev(n)
except:
r = -2 # sort to the beginning of the list if unknown
Thomas Arendsen Hein
Cleanup of indentation, spacing, newlines, strings and line length
r1615 l.append((r, t, n))
mpm@selenic.com
Break apart hg.py...
r1089 l.sort()
Thomas Arendsen Hein
Cleanup of indentation, spacing, newlines, strings and line length
r1615 return [(t, n) for r, t, n in l]
mpm@selenic.com
Break apart hg.py...
r1089
def nodetags(self, node):
'''return the tags associated with a node'''
if not self.nodetagscache:
self.nodetagscache = {}
Thomas Arendsen Hein
Cleanup of indentation, spacing, newlines, strings and line length
r1615 for t, n in self.tags().items():
self.nodetagscache.setdefault(n, []).append(t)
mpm@selenic.com
Break apart hg.py...
r1089 return self.nodetagscache.get(node, [])
def lookup(self, key):
try:
return self.tags()[key]
except KeyError:
Brendan Cully
Add '.' as a shortcut revision name for the working directory parent.
r2789 if key == '.':
key = self.dirstate.parents()[0]
if key == nullid:
raise repo.RepoError(_("no revision checked out"))
mpm@selenic.com
Break apart hg.py...
r1089 try:
return self.changelog.lookup(key)
except:
Benoit Boissinot
i18n part2: use '_' for all strings who are part of the user interface
r1402 raise repo.RepoError(_("unknown revision '%s'") % key)
mpm@selenic.com
Break apart hg.py...
r1089
def dev(self):
Vadim Gelfer
replace os.stat with os.lstat in some where.
r2448 return os.lstat(self.path).st_dev
mpm@selenic.com
Break apart hg.py...
r1089
def local(self):
mpm@selenic.com
Separate out old-http support...
r1101 return True
mpm@selenic.com
Break apart hg.py...
r1089
def join(self, f):
return os.path.join(self.path, f)
def wjoin(self, f):
return os.path.join(self.root, f)
def file(self, f):
Thomas Arendsen Hein
Cleanup of indentation, spacing, newlines, strings and line length
r1615 if f[0] == '/':
f = f[1:]
mason@suse.com
Implement revlogng....
r2072 return filelog.filelog(self.opener, f, self.revlogversion)
mpm@selenic.com
Break apart hg.py...
r1089
Matt Mackall
Add context helper functions to localrepo
r2564 def changectx(self, changeid):
return context.changectx(self, changeid)
def filectx(self, path, changeid=None, fileid=None):
"""changeid can be a changeset revision, node, or tag.
fileid can be a file revision or node."""
return context.filectx(self, path, changeid, fileid)
mpm@selenic.com
Break apart hg.py...
r1089 def getcwd(self):
return self.dirstate.getcwd()
def wfile(self, f, mode='r'):
return self.wopener(f, mode)
def wread(self, filename):
mpm@selenic.com
Add file encoding/decoding support
r1258 if self.encodepats == None:
l = []
for pat, cmd in self.ui.configitems("encode"):
Benoit Boissinot
fix the call to util.matcher (the args should be (reporoot, cwd, ...))
r1947 mf = util.matcher(self.root, "", [pat], [], [])[1]
mpm@selenic.com
Add file encoding/decoding support
r1258 l.append((mf, cmd))
self.encodepats = l
data = self.wopener(filename, 'r').read()
for mf, cmd in self.encodepats:
if mf(filename):
Benoit Boissinot
i18n part2: use '_' for all strings who are part of the user interface
r1402 self.ui.debug(_("filtering %s through %s\n") % (filename, cmd))
mpm@selenic.com
Add file encoding/decoding support
r1258 data = util.filter(data, cmd)
break
return data
mpm@selenic.com
Break apart hg.py...
r1089
def wwrite(self, filename, data, fd=None):
mpm@selenic.com
Add file encoding/decoding support
r1258 if self.decodepats == None:
l = []
for pat, cmd in self.ui.configitems("decode"):
Benoit Boissinot
fix the call to util.matcher (the args should be (reporoot, cwd, ...))
r1947 mf = util.matcher(self.root, "", [pat], [], [])[1]
mpm@selenic.com
Add file encoding/decoding support
r1258 l.append((mf, cmd))
self.decodepats = l
for mf, cmd in self.decodepats:
if mf(filename):
Benoit Boissinot
i18n part2: use '_' for all strings who are part of the user interface
r1402 self.ui.debug(_("filtering %s through %s\n") % (filename, cmd))
mpm@selenic.com
Add file encoding/decoding support
r1258 data = util.filter(data, cmd)
break
mpm@selenic.com
Break apart hg.py...
r1089 if fd:
return fd.write(data)
return self.wopener(filename, 'w').write(data)
def transaction(self):
mason@suse.com
Automatic nesting into running transactions in the same repository....
r1806 tr = self.transhandle
if tr != None and tr.running():
return tr.nest()
Thomas Arendsen Hein
Renamed localrepo.undo() to rollback() and talk about "rollback information".
r2362 # save dirstate for rollback
mpm@selenic.com
Break apart hg.py...
r1089 try:
ds = self.opener("dirstate").read()
except IOError:
ds = ""
self.opener("journal.dirstate", "w").write(ds)
mason@suse.com
Automatic nesting into running transactions in the same repository....
r1806 tr = transaction.transaction(self.ui.warn, self.opener,
Vadim Gelfer
fix broken environment save/restore when a hook runs....
r1880 self.join("journal"),
mason@suse.com
Automatic nesting into running transactions in the same repository....
r1806 aftertrans(self.path))
self.transhandle = tr
return tr
mpm@selenic.com
Break apart hg.py...
r1089
def recover(self):
Benoit Boissinot
fix warnings from pychecker (unused variables and shadowing)
r1749 l = self.lock()
mpm@selenic.com
Break apart hg.py...
r1089 if os.path.exists(self.join("journal")):
Benoit Boissinot
i18n part2: use '_' for all strings who are part of the user interface
r1402 self.ui.status(_("rolling back interrupted transaction\n"))
Matt Mackall
Automatically run "verify" whenever we run "recover"
r1516 transaction.rollback(self.opener, self.join("journal"))
Benoit Boissinot
revalidate revlog data after locking the repo (issue132)
r1784 self.reload()
Matt Mackall
Automatically run "verify" whenever we run "recover"
r1516 return True
mpm@selenic.com
Break apart hg.py...
r1089 else:
Benoit Boissinot
i18n part2: use '_' for all strings who are part of the user interface
r1402 self.ui.warn(_("no interrupted transaction available\n"))
Matt Mackall
Automatically run "verify" whenever we run "recover"
r1516 return False
mpm@selenic.com
Break apart hg.py...
r1089
Thomas Arendsen Hein
Renamed localrepo.undo() to rollback() and talk about "rollback information".
r2362 def rollback(self, wlock=None):
mason@suse.com
Allow callers to pass in the dirstate lock in most localrepo.py funcs....
r1712 if not wlock:
wlock = self.wlock()
Benoit Boissinot
fix warnings from pychecker (unused variables and shadowing)
r1749 l = self.lock()
mpm@selenic.com
Break apart hg.py...
r1089 if os.path.exists(self.join("undo")):
Benoit Boissinot
i18n part2: use '_' for all strings who are part of the user interface
r1402 self.ui.status(_("rolling back last transaction\n"))
mpm@selenic.com
Break apart hg.py...
r1089 transaction.rollback(self.opener, self.join("undo"))
util.rename(self.join("undo.dirstate"), self.join("dirstate"))
Benoit Boissinot
revalidate revlog data after locking the repo (issue132)
r1784 self.reload()
self.wreload()
mpm@selenic.com
Break apart hg.py...
r1089 else:
Thomas Arendsen Hein
Renamed localrepo.undo() to rollback() and talk about "rollback information".
r2362 self.ui.warn(_("no rollback information available\n"))
mpm@selenic.com
Break apart hg.py...
r1089
Benoit Boissinot
revalidate revlog data after locking the repo (issue132)
r1784 def wreload(self):
self.dirstate.read()
def reload(self):
self.changelog.load()
self.manifest.load()
self.tagscache = None
self.nodetagscache = None
Vadim Gelfer
fix backtrace printed when cannot get lock....
r2016 def do_lock(self, lockname, wait, releasefn=None, acquirefn=None,
desc=None):
mpm@selenic.com
Break apart hg.py...
r1089 try:
Vadim Gelfer
fix backtrace printed when cannot get lock....
r2016 l = lock.lock(self.join(lockname), 0, releasefn, desc=desc)
Benoit Boissinot
add localrepo.wlock for protecting the dirstate...
r1531 except lock.LockHeld, inst:
if not wait:
Vadim Gelfer
fix backtrace printed when cannot get lock....
r2016 raise
self.ui.warn(_("waiting for lock on %s held by %s\n") %
(desc, inst.args[0]))
# default to 600 seconds timeout
l = lock.lock(self.join(lockname),
int(self.ui.config("ui", "timeout") or 600),
releasefn, desc=desc)
Benoit Boissinot
localrepo: refactor the locking functions
r1751 if acquirefn:
acquirefn()
return l
def lock(self, wait=1):
Vadim Gelfer
fix backtrace printed when cannot get lock....
r2016 return self.do_lock("lock", wait, acquirefn=self.reload,
desc=_('repository %s') % self.origroot)
Benoit Boissinot
localrepo: refactor the locking functions
r1751
def wlock(self, wait=1):
Vadim Gelfer
fix backtrace printed when cannot get lock....
r2016 return self.do_lock("wlock", wait, self.dirstate.write,
self.wreload,
desc=_('working directory of %s') % self.origroot)
Benoit Boissinot
add localrepo.wlock for protecting the dirstate...
r1531
Matt Mackall
Refactor excessive merge detection, add test
r1716 def checkfilemerge(self, filename, text, filelog, manifest1, manifest2):
"determine whether a new filenode is needed"
fp1 = manifest1.get(filename, nullid)
fp2 = manifest2.get(filename, nullid)
if fp2 != nullid:
# is one parent an ancestor of the other?
fpa = filelog.ancestor(fp1, fp2)
if fpa == fp1:
fp1, fp2 = fp2, nullid
elif fpa == fp2:
fp2 = nullid
# is the file unmodified from the parent? report existing entry
if fp2 == nullid and text == filelog.read(fp1):
return (fp1, None, None)
return (None, fp1, fp2)
mason@suse.com
Allow callers to pass in the dirstate lock in most localrepo.py funcs....
r1712 def rawcommit(self, files, text, user, date, p1=None, p2=None, wlock=None):
mpm@selenic.com
Break apart hg.py...
r1089 orig_parent = self.dirstate.parents()[0] or nullid
p1 = p1 or self.dirstate.parents()[0] or nullid
p2 = p2 or self.dirstate.parents()[1] or nullid
c1 = self.changelog.read(p1)
c2 = self.changelog.read(p2)
Matt Mackall
Change remaining users of manifest flags
r2840 m1 = self.manifest.read(c1[0]).copy()
mpm@selenic.com
Break apart hg.py...
r1089 m2 = self.manifest.read(c2[0])
changed = []
if orig_parent == p1:
update_dirstate = 1
else:
update_dirstate = 0
mason@suse.com
Allow callers to pass in the dirstate lock in most localrepo.py funcs....
r1712 if not wlock:
wlock = self.wlock()
Benoit Boissinot
fix warnings from pychecker (unused variables and shadowing)
r1749 l = self.lock()
mpm@selenic.com
Break apart hg.py...
r1089 tr = self.transaction()
linkrev = self.changelog.count()
for f in files:
try:
t = self.wread(f)
Matt Mackall
Change remaining users of manifest flags
r2840 m1.set(f, util.is_exec(self.wjoin(f), m1.execf(f)))
mpm@selenic.com
Break apart hg.py...
r1089 r = self.file(f)
Matt Mackall
Refactor excessive merge detection, add test
r1716 (entry, fp1, fp2) = self.checkfilemerge(f, t, r, m1, m2)
if entry:
Matt Mackall
Change remaining users of manifest flags
r2840 m1[f] = entry
Matt Mackall
Refactor excessive merge detection, add test
r1716 continue
mpm@selenic.com
Break apart hg.py...
r1089
Matt Mackall
Change remaining users of manifest flags
r2840 m1[f] = r.add(t, {}, tr, linkrev, fp1, fp2)
mpm@selenic.com
Break apart hg.py...
r1089 changed.append(f)
if update_dirstate:
self.dirstate.update([f], "n")
except IOError:
try:
Matt Mackall
Change remaining users of manifest flags
r2840 del m1[f]
del m1[f]
mpm@selenic.com
Break apart hg.py...
r1089 if update_dirstate:
self.dirstate.forget([f])
except:
# deleted from p2?
pass
Matt Mackall
Remove manifest.readflags
r2841 mnode = self.manifest.add(m1, tr, linkrev, c1[0], c2[0])
mpm@selenic.com
Break apart hg.py...
r1089 user = user or self.ui.username()
n = self.changelog.add(mnode, changed, text, tr, p1, p2, user, date)
tr.close()
if update_dirstate:
self.dirstate.setparents(n, nullid)
Thomas Arendsen Hein
Cleanup of indentation, spacing, newlines, strings and line length
r1615 def commit(self, files=None, text="", user=None, date=None,
john.levon@sun.com
Force editor to appear for an 'hg backout' if a message or logfile aren't...
r2267 match=util.always, force=False, lock=None, wlock=None,
force_editor=False):
mpm@selenic.com
Break apart hg.py...
r1089 commit = []
remove = []
changed = []
if files:
for f in files:
s = self.dirstate.state(f)
if s in 'nmai':
commit.append(f)
elif s == 'r':
remove.append(f)
else:
Benoit Boissinot
i18n part2: use '_' for all strings who are part of the user interface
r1402 self.ui.warn(_("%s not tracked!\n") % f)
mpm@selenic.com
Break apart hg.py...
r1089 else:
Thomas Arendsen Hein
Distinguish removed and deleted files. Tests are not fixed yet....
r1619 modified, added, removed, deleted, unknown = self.changes(match=match)
Thomas Arendsen Hein
Renamed c, a, d, u to modified, added, removed, unknown for users of changes()
r1618 commit = modified + added
remove = removed
mpm@selenic.com
Break apart hg.py...
r1089
p1, p2 = self.dirstate.parents()
c1 = self.changelog.read(p1)
c2 = self.changelog.read(p2)
Matt Mackall
Change remaining users of manifest flags
r2840 m1 = self.manifest.read(c1[0]).copy()
mpm@selenic.com
Break apart hg.py...
r1089 m2 = self.manifest.read(c2[0])
if not commit and not remove and not force and p2 == nullid:
Benoit Boissinot
i18n part2: use '_' for all strings who are part of the user interface
r1402 self.ui.status(_("nothing changed\n"))
mpm@selenic.com
Break apart hg.py...
r1089 return None
Vadim Gelfer
add pretxncommit hook....
r1721 xp1 = hex(p1)
if p2 == nullid: xp2 = ''
else: xp2 = hex(p2)
Vadim Gelfer
fix names of parent changeset ids in hooks....
r1727 self.hook("precommit", throw=True, parent1=xp1, parent2=xp2)
mpm@selenic.com
Break apart hg.py...
r1089
mason@suse.com
Allow callers to pass in the dirstate lock in most localrepo.py funcs....
r1712 if not wlock:
wlock = self.wlock()
mason@suse.com
Allow repo lock to be passed in to localrepo.commit for performance
r1807 if not lock:
lock = self.lock()
mpm@selenic.com
Break apart hg.py...
r1089 tr = self.transaction()
# check in files
new = {}
linkrev = self.changelog.count()
commit.sort()
for f in commit:
self.ui.note(f + "\n")
try:
Matt Mackall
Change remaining users of manifest flags
r2840 m1.set(f, util.is_exec(self.wjoin(f), m1.execf(f)))
mpm@selenic.com
Break apart hg.py...
r1089 t = self.wread(f)
except IOError:
Benoit Boissinot
i18n part2: use '_' for all strings who are part of the user interface
r1402 self.ui.warn(_("trouble committing %s!\n") % f)
mpm@selenic.com
Break apart hg.py...
r1089 raise
mpm@selenic.com
fix some rename/copy bugs...
r1117 r = self.file(f)
mpm@selenic.com
Break apart hg.py...
r1089 meta = {}
cp = self.dirstate.copied(f)
if cp:
meta["copy"] = cp
meta["copyrev"] = hex(m1.get(cp, m2.get(cp, nullid)))
Benoit Boissinot
i18n part2: use '_' for all strings who are part of the user interface
r1402 self.ui.debug(_(" %s: copy %s:%s\n") % (f, cp, meta["copyrev"]))
mpm@selenic.com
fix some rename/copy bugs...
r1117 fp1, fp2 = nullid, nullid
else:
Matt Mackall
Refactor excessive merge detection, add test
r1716 entry, fp1, fp2 = self.checkfilemerge(f, t, r, m1, m2)
if entry:
new[f] = entry
mpm@selenic.com
Break apart hg.py...
r1089 continue
new[f] = r.add(t, meta, tr, linkrev, fp1, fp2)
# remember what we've added so that we can later calculate
# the files to pull from a set of changesets
changed.append(f)
# update manifest
m1.update(new)
for f in remove:
if f in m1:
del m1[f]
Matt Mackall
Remove manifest.readflags
r2841 mn = self.manifest.add(m1, tr, linkrev, c1[0], c2[0],
mpm@selenic.com
Break apart hg.py...
r1089 (new, remove))
# add changeset
new = new.keys()
new.sort()
Thomas Arendsen Hein
Pass correct username as $HGUSER to hgeditor if "commit -u" is used....
r1983 user = user or self.ui.username()
john.levon@sun.com
Force editor to appear for an 'hg backout' if a message or logfile aren't...
r2267 if not text or force_editor:
edittext = []
if text:
edittext.append(text)
edittext.append("")
mpm@selenic.com
Break apart hg.py...
r1089 if p2 != nullid:
Thomas Arendsen Hein
Create default commit message in a more pythonic way.
r1709 edittext.append("HG: branch merge")
edittext.extend(["HG: changed %s" % f for f in changed])
edittext.extend(["HG: removed %s" % f for f in remove])
mpm@selenic.com
Break apart hg.py...
r1089 if not changed and not remove:
Thomas Arendsen Hein
Create default commit message in a more pythonic way.
r1709 edittext.append("HG: no files changed")
edittext.append("")
Thomas Arendsen Hein
Run commit message editor in the repo root (like hooks)....
r1706 # run editor in the repository root
olddir = os.getcwd()
os.chdir(self.root)
Thomas Arendsen Hein
Strip empty lines and trailing spaces around commit messages....
r2301 text = self.ui.edit("\n".join(edittext), user)
Thomas Arendsen Hein
Run commit message editor in the repo root (like hooks)....
r1706 os.chdir(olddir)
mpm@selenic.com
Break apart hg.py...
r1089
Thomas Arendsen Hein
Strip empty lines and trailing spaces around commit messages....
r2301 lines = [line.rstrip() for line in text.rstrip().splitlines()]
while lines and not lines[0]:
del lines[0]
if not lines:
return None
text = '\n'.join(lines)
Benoit Boissinot
add removed files to the changelog file list...
r1645 n = self.changelog.add(mn, changed + remove, text, tr, p1, p2, user, date)
Vadim Gelfer
fix names of parent changeset ids in hooks....
r1727 self.hook('pretxncommit', throw=True, node=hex(n), parent1=xp1,
parent2=xp2)
mpm@selenic.com
Break apart hg.py...
r1089 tr.close()
self.dirstate.setparents(n)
self.dirstate.update(new, "n")
self.dirstate.forget(remove)
Vadim Gelfer
fix names of parent changeset ids in hooks....
r1727 self.hook("commit", node=hex(n), parent1=xp1, parent2=xp2)
mpm@selenic.com
Break apart hg.py...
r1089 return n
Vadim Gelfer
rewrite revert command. fix issues 93, 123, 147....
r2029 def walk(self, node=None, files=[], match=util.always, badmatch=None):
mpm@selenic.com
Break apart hg.py...
r1089 if node:
Benoit Boissinot
fix the cat command...
r1582 fdict = dict.fromkeys(files)
mpm@selenic.com
Break apart hg.py...
r1089 for fn in self.manifest.read(self.changelog.read(node)[0]):
Benoit Boissinot
fix the cat command...
r1582 fdict.pop(fn, None)
if match(fn):
yield 'm', fn
for fn in fdict:
Vadim Gelfer
rewrite revert command. fix issues 93, 123, 147....
r2029 if badmatch and badmatch(fn):
if match(fn):
yield 'b', fn
else:
self.ui.warn(_('%s: No such file in rev %s\n') % (
util.pathto(self.getcwd(), fn), short(node)))
mpm@selenic.com
Break apart hg.py...
r1089 else:
Vadim Gelfer
small changes to revert command....
r2042 for src, fn in self.dirstate.walk(files, match, badmatch=badmatch):
mpm@selenic.com
Break apart hg.py...
r1089 yield src, fn
Vadim Gelfer
status: add -c (clean) and -A (all files) options...
r2661 def status(self, node1=None, node2=None, files=[], match=util.always,
wlock=None, list_ignored=False, list_clean=False):
"""return status of files between two nodes or node and working directory
Thomas Arendsen Hein
Cleaned up localrepo.changes()
r1616
If node1 is None, use the first dirstate parent instead.
If node2 is None, compare node1 with working directory.
"""
mpm@selenic.com
Break apart hg.py...
r1089
def fcmp(fn, mf):
t1 = self.wread(fn)
t2 = self.file(fn).read(mf.get(fn, nullid))
return cmp(t1, t2)
def mfmatches(node):
Thomas Arendsen Hein
Cleaned up localrepo.changes()
r1616 change = self.changelog.read(node)
mf = dict(self.manifest.read(change[0]))
mpm@selenic.com
Break apart hg.py...
r1089 for fn in mf.keys():
if not match(fn):
del mf[fn]
return mf
Vadim Gelfer
status: add -c (clean) and -A (all files) options...
r2661 modified, added, removed, deleted, unknown = [], [], [], [], []
ignored, clean = [], []
Chris Mason
Fix cold cache diff performance...
r2474 compareworking = False
Chris Mason
Fix localrepo.changes() Correctly decide if we are diffing the working dir
r2491 if not node1 or (not node2 and node1 == self.dirstate.parents()[0]):
Chris Mason
Fix cold cache diff performance...
r2474 compareworking = True
if not compareworking:
Alexis S. L. Carvalho
speed up hg log --debug...
r1802 # read the manifest from node1 before the manifest from node2,
# so that we'll hit the manifest cache if we're going through
# all the revisions in parent->child order.
mf1 = mfmatches(node1)
mpm@selenic.com
Break apart hg.py...
r1089 # are we comparing the working directory?
if not node2:
mason@suse.com
Allow callers to pass in the dirstate lock in most localrepo.py funcs....
r1712 if not wlock:
try:
wlock = self.wlock(wait=0)
Benoit Boissinot
use repo.lock when cloning via copy, use lock.LockException when necessary
r1754 except lock.LockException:
mason@suse.com
Allow callers to pass in the dirstate lock in most localrepo.py funcs....
r1712 wlock = None
Vadim Gelfer
status: add -c (clean) and -A (all files) options...
r2661 (lookup, modified, added, removed, deleted, unknown,
ignored, clean) = self.dirstate.status(files, match,
list_ignored, list_clean)
mpm@selenic.com
Break apart hg.py...
r1089
# are we comparing working dir against its parent?
Chris Mason
Fix cold cache diff performance...
r2474 if compareworking:
Thomas Arendsen Hein
Cleaned up localrepo.changes()
r1616 if lookup:
mpm@selenic.com
Break apart hg.py...
r1089 # do a full compare of any files that might have changed
Thomas Arendsen Hein
Cleaned up localrepo.changes()
r1616 mf2 = mfmatches(self.dirstate.parents()[0])
for f in lookup:
mpm@selenic.com
Break apart hg.py...
r1089 if fcmp(f, mf2):
Thomas Arendsen Hein
Cleaned up localrepo.changes()
r1616 modified.append(f)
Benoit Boissinot
reset mtime when two files are equal
r1532 elif wlock is not None:
self.dirstate.update([f], "n")
Thomas Arendsen Hein
Cleaned up localrepo.changes()
r1616 else:
# we are comparing working dir against non-parent
# generate a pseudo-manifest for the working dir
mf2 = mfmatches(self.dirstate.parents()[0])
for f in lookup + modified + added:
mf2[f] = ""
Thomas Arendsen Hein
Make localrepo.changes() internally distinguish between removed and deleted.
r1617 for f in removed:
Thomas Arendsen Hein
Cleaned up localrepo.changes()
r1616 if f in mf2:
del mf2[f]
mpm@selenic.com
Break apart hg.py...
r1089 else:
Thomas Arendsen Hein
Cleaned up localrepo.changes()
r1616 # we are comparing two revisions
mf2 = mfmatches(node2)
mpm@selenic.com
Break apart hg.py...
r1089
Chris Mason
Fix cold cache diff performance...
r2474 if not compareworking:
Thomas Arendsen Hein
Cleaned up localrepo.changes()
r1616 # flush lists from dirstate before comparing manifests
Vadim Gelfer
status: add -c (clean) and -A (all files) options...
r2661 modified, added, clean = [], [], []
mpm@selenic.com
Break apart hg.py...
r1089
Chris Mason
Fix cold cache diff performance...
r2474 # make sure to sort the files so we talk to the disk in a
# reasonable order
mf2keys = mf2.keys()
mf2keys.sort()
for fn in mf2keys:
Thomas Arendsen Hein
Cleaned up localrepo.changes()
r1616 if mf1.has_key(fn):
if mf1[fn] != mf2[fn] and (mf2[fn] != "" or fcmp(fn, mf1)):
modified.append(fn)
Vadim Gelfer
status: add -c (clean) and -A (all files) options...
r2661 elif list_clean:
clean.append(fn)
Thomas Arendsen Hein
Cleaned up localrepo.changes()
r1616 del mf1[fn]
else:
added.append(fn)
mpm@selenic.com
Break apart hg.py...
r1089
Thomas Arendsen Hein
Make localrepo.changes() internally distinguish between removed and deleted.
r1617 removed = mf1.keys()
Thomas Arendsen Hein
Cleaned up localrepo.changes()
r1616 # sort and return results:
Vadim Gelfer
status: add -c (clean) and -A (all files) options...
r2661 for l in modified, added, removed, deleted, unknown, ignored, clean:
mpm@selenic.com
Break apart hg.py...
r1089 l.sort()
Vadim Gelfer
status: add -c (clean) and -A (all files) options...
r2661 return (modified, added, removed, deleted, unknown, ignored, clean)
def changes(self, node1=None, node2=None, files=[], match=util.always,
wlock=None, list_ignored=False, list_clean=False):
'''DEPRECATED - use status instead'''
marduit = self.status(node1, node2, files, match, wlock,
list_ignored, list_clean)
if list_ignored:
return marduit[:-1]
Thomas Arendsen Hein
New option -i/--ignored for 'hg status' to show ignored files....
r2022 else:
Vadim Gelfer
status: add -c (clean) and -A (all files) options...
r2661 return marduit[:-2]
mpm@selenic.com
Break apart hg.py...
r1089
mason@suse.com
Allow callers to pass in the dirstate lock in most localrepo.py funcs....
r1712 def add(self, list, wlock=None):
if not wlock:
wlock = self.wlock()
mpm@selenic.com
Break apart hg.py...
r1089 for f in list:
p = self.wjoin(f)
if not os.path.exists(p):
Benoit Boissinot
i18n part2: use '_' for all strings who are part of the user interface
r1402 self.ui.warn(_("%s does not exist!\n") % f)
mpm@selenic.com
Break apart hg.py...
r1089 elif not os.path.isfile(p):
Thomas Arendsen Hein
Cleanup of indentation, spacing, newlines, strings and line length
r1615 self.ui.warn(_("%s not added: only files supported currently\n")
% f)
mpm@selenic.com
Break apart hg.py...
r1089 elif self.dirstate.state(f) in 'an':
Benoit Boissinot
i18n part2: use '_' for all strings who are part of the user interface
r1402 self.ui.warn(_("%s already tracked!\n") % f)
mpm@selenic.com
Break apart hg.py...
r1089 else:
self.dirstate.update([f], "a")
mason@suse.com
Allow callers to pass in the dirstate lock in most localrepo.py funcs....
r1712 def forget(self, list, wlock=None):
if not wlock:
wlock = self.wlock()
mpm@selenic.com
Break apart hg.py...
r1089 for f in list:
if self.dirstate.state(f) not in 'ai':
Benoit Boissinot
i18n part2: use '_' for all strings who are part of the user interface
r1402 self.ui.warn(_("%s not added!\n") % f)
mpm@selenic.com
Break apart hg.py...
r1089 else:
self.dirstate.forget([f])
mason@suse.com
Allow callers to pass in the dirstate lock in most localrepo.py funcs....
r1712 def remove(self, list, unlink=False, wlock=None):
Benoit Boissinot
refactor some unlink/remove code and make sure we prune empty dir
r1415 if unlink:
for f in list:
try:
util.unlink(self.wjoin(f))
except OSError, inst:
Thomas Arendsen Hein
Cleanup of indentation, spacing, newlines, strings and line length
r1615 if inst.errno != errno.ENOENT:
raise
mason@suse.com
Allow callers to pass in the dirstate lock in most localrepo.py funcs....
r1712 if not wlock:
wlock = self.wlock()
mpm@selenic.com
Break apart hg.py...
r1089 for f in list:
p = self.wjoin(f)
if os.path.exists(p):
Benoit Boissinot
i18n part2: use '_' for all strings who are part of the user interface
r1402 self.ui.warn(_("%s still exists!\n") % f)
mpm@selenic.com
Break apart hg.py...
r1089 elif self.dirstate.state(f) == 'a':
self.dirstate.forget([f])
elif f not in self.dirstate:
Benoit Boissinot
i18n part2: use '_' for all strings who are part of the user interface
r1402 self.ui.warn(_("%s not tracked!\n") % f)
mpm@selenic.com
Break apart hg.py...
r1089 else:
self.dirstate.update([f], "r")
mason@suse.com
Allow callers to pass in the dirstate lock in most localrepo.py funcs....
r1712 def undelete(self, list, wlock=None):
Matt Mackall
Allow reverting a deleted file with two parents...
r1448 p = self.dirstate.parents()[0]
Benoit Boissinot
revert added and removed files to their normal state before reverting...
r1447 mn = self.changelog.read(p)[0]
m = self.manifest.read(mn)
mason@suse.com
Allow callers to pass in the dirstate lock in most localrepo.py funcs....
r1712 if not wlock:
wlock = self.wlock()
Benoit Boissinot
revert added and removed files to their normal state before reverting...
r1447 for f in list:
if self.dirstate.state(f) not in "r":
self.ui.warn("%s not removed!\n" % f)
else:
t = self.file(f).read(m[f])
Benoit Boissinot
util.opener already create the base directory if it doesn't exist
r1477 self.wwrite(f, t)
Matt Mackall
Change remaining users of manifest flags
r2840 util.set_exec(self.wjoin(f), m.execf(f))
Benoit Boissinot
revert added and removed files to their normal state before reverting...
r1447 self.dirstate.update([f], "n")
mason@suse.com
Allow callers to pass in the dirstate lock in most localrepo.py funcs....
r1712 def copy(self, source, dest, wlock=None):
mpm@selenic.com
Break apart hg.py...
r1089 p = self.wjoin(dest)
if not os.path.exists(p):
Benoit Boissinot
i18n part2: use '_' for all strings who are part of the user interface
r1402 self.ui.warn(_("%s does not exist!\n") % dest)
mpm@selenic.com
Break apart hg.py...
r1089 elif not os.path.isfile(p):
Benoit Boissinot
i18n part2: use '_' for all strings who are part of the user interface
r1402 self.ui.warn(_("copy failed: %s is not a file\n") % dest)
mpm@selenic.com
Break apart hg.py...
r1089 else:
mason@suse.com
Allow callers to pass in the dirstate lock in most localrepo.py funcs....
r1712 if not wlock:
wlock = self.wlock()
mpm@selenic.com
Break apart hg.py...
r1089 if self.dirstate.state(dest) == '?':
self.dirstate.update([dest], "a")
self.dirstate.copy(source, dest)
Thomas Arendsen Hein
Fixes to "hg heads -r FOO":...
r1551 def heads(self, start=None):
Benoit Boissinot
add a -r/--rev option to heads to show only heads descendant from rev
r1550 heads = self.changelog.heads(start)
# sort the output in rev descending order
heads = [(-self.changelog.rev(h), h) for h in heads]
heads.sort()
return [n for (r, n) in heads]
mpm@selenic.com
Break apart hg.py...
r1089
# branchlookup returns a dict giving a list of branches for
# each head. A branch is defined as the tag of a node or
# the branch of the node's parents. If a node has multiple
# branch tags, tags are eliminated if they are visible from other
# branch tags.
#
# So, for this graph: a->b->c->d->e
# \ /
# aa -----/
# a has tag 2.6.12
# d has tag 2.6.13
# e would have branch tags for 2.6.12 and 2.6.13. Because the node
# for 2.6.12 can be reached from the node 2.6.13, that is eliminated
# from the list.
#
# It is possible that more than one head will have the same branch tag.
# callers need to check the result for multiple heads under the same
# branch tag if that is a problem for them (ie checkout of a specific
# branch).
#
# passing in a specific branch will limit the depth of the search
# through the parents. It won't limit the branches returned in the
# result though.
def branchlookup(self, heads=None, branch=None):
if not heads:
heads = self.heads()
headt = [ h for h in heads ]
chlog = self.changelog
branches = {}
merges = []
seenmerge = {}
# traverse the tree once for each head, recording in the branches
# dict which tags are visible from this head. The branches
# dict also records which tags are visible from each tag
# while we traverse.
while headt or merges:
if merges:
n, found = merges.pop()
visit = [n]
else:
h = headt.pop()
visit = [h]
found = [h]
seen = {}
while visit:
n = visit.pop()
if n in seen:
continue
pp = chlog.parents(n)
tags = self.nodetags(n)
if tags:
for x in tags:
if x == 'tip':
continue
for f in found:
branches.setdefault(f, {})[n] = 1
branches.setdefault(n, {})[n] = 1
break
if n not in found:
found.append(n)
if branch in tags:
continue
seen[n] = 1
if pp[1] != nullid and n not in seenmerge:
merges.append((pp[1], [x for x in found]))
seenmerge[n] = 1
if pp[0] != nullid:
visit.append(pp[0])
# traverse the branches dict, eliminating branch tags from each
# head that are visible from another branch tag for that head.
out = {}
viscache = {}
for h in heads:
def visible(node):
if node in viscache:
return viscache[node]
ret = {}
visit = [node]
while visit:
x = visit.pop()
if x in viscache:
ret.update(viscache[x])
elif x not in ret:
ret[x] = 1
if x in branches:
visit[len(visit):] = branches[x].keys()
viscache[node] = ret
return ret
if h not in branches:
continue
# O(n^2), but somewhat limited. This only searches the
# tags visible from a specific head, not all the tags in the
# whole repo.
for b in branches[h]:
vis = False
for bb in branches[h].keys():
if b != bb:
if b in visible(bb):
vis = True
break
if not vis:
l = out.setdefault(h, [])
l[len(l):] = self.nodetags(b)
return out
def branches(self, nodes):
Thomas Arendsen Hein
Cleanup of indentation, spacing, newlines, strings and line length
r1615 if not nodes:
nodes = [self.changelog.tip()]
mpm@selenic.com
Break apart hg.py...
r1089 b = []
for n in nodes:
t = n
Benoit Boissinot
n is always 'True', we can only stop the loop with the break statement
r2345 while 1:
mpm@selenic.com
Break apart hg.py...
r1089 p = self.changelog.parents(n)
if p[1] != nullid or p[0] == nullid:
b.append((t, n, p[0], p[1]))
break
n = p[0]
return b
def between(self, pairs):
r = []
for top, bottom in pairs:
n, l, i = top, [], 0
f = 1
while n != bottom:
p = self.changelog.parents(n)[0]
if i == f:
l.append(n)
f = f * 2
n = p
i += 1
r.append(l)
return r
Vadim Gelfer
add -f/--force to pull, incoming, outgoing, to work on unrelated repo....
r1959 def findincoming(self, remote, base=None, heads=None, force=False):
Benoit Boissinot
document and fix findincoming...
r2339 """Return list of roots of the subsets of missing nodes from remote
If base dict is specified, assume that these nodes and their parents
exist on the remote side and that no child of a node of base exists
in both remote and self.
Furthermore base will be updated to include the nodes that exists
in self and remote but no children exists in self and remote.
If a list of heads is specified, return only nodes which are heads
or ancestors of these heads.
All the ancestors of base are in self and in remote.
All the descendants of the list returned are missing in self.
(and so we know that the rest of the nodes are missing in remote, see
outgoing)
"""
mpm@selenic.com
Break apart hg.py...
r1089 m = self.changelog.nodemap
search = []
fetch = {}
seen = {}
seenbranch = {}
if base == None:
base = {}
Matt Mackall
Move empty local repo logic for pull into findincoming...
r2108 if not heads:
heads = remote.heads()
if self.changelog.tip() == nullid:
Benoit Boissinot
document and fix findincoming...
r2339 base[nullid] = 1
Matt Mackall
Move empty local repo logic for pull into findincoming...
r2108 if heads != [nullid]:
return [nullid]
return []
mpm@selenic.com
Break apart hg.py...
r1089 # assume we're closer to the tip than the root
# and start by examining the heads
Benoit Boissinot
i18n part2: use '_' for all strings who are part of the user interface
r1402 self.ui.status(_("searching for changes\n"))
mpm@selenic.com
Break apart hg.py...
r1089
unknown = []
for h in heads:
if h not in m:
unknown.append(h)
else:
base[h] = 1
if not unknown:
Benoit Boissinot
findincoming should return an iterable
r1895 return []
mpm@selenic.com
Break apart hg.py...
r1089
Benoit Boissinot
document and fix findincoming...
r2339 req = dict.fromkeys(unknown)
mpm@selenic.com
Break apart hg.py...
r1089 reqcnt = 0
# search through remote branches
# a 'branch' here is a linear segment of history, with four parts:
# head, root, first parent, second parent
# (a branch always has two parents (or none) by definition)
unknown = remote.branches(unknown)
while unknown:
r = []
while unknown:
n = unknown.pop(0)
if n[0] in seen:
continue
Thomas Arendsen Hein
Cleanup of indentation, spacing, newlines, strings and line length
r1615 self.ui.debug(_("examining %s:%s\n")
% (short(n[0]), short(n[1])))
Benoit Boissinot
document and fix findincoming...
r2339 if n[0] == nullid: # found the end of the branch
pass
elif n in seenbranch:
Benoit Boissinot
i18n part2: use '_' for all strings who are part of the user interface
r1402 self.ui.debug(_("branch already found\n"))
mpm@selenic.com
Break apart hg.py...
r1089 continue
Benoit Boissinot
document and fix findincoming...
r2339 elif n[1] and n[1] in m: # do we know the base?
Benoit Boissinot
i18n part2: use '_' for all strings who are part of the user interface
r1402 self.ui.debug(_("found incomplete branch %s:%s\n")
mpm@selenic.com
Break apart hg.py...
r1089 % (short(n[0]), short(n[1])))
search.append(n) # schedule branch range for scanning
seenbranch[n] = 1
else:
if n[1] not in seen and n[1] not in fetch:
if n[2] in m and n[3] in m:
Benoit Boissinot
i18n part2: use '_' for all strings who are part of the user interface
r1402 self.ui.debug(_("found new changeset %s\n") %
mpm@selenic.com
Break apart hg.py...
r1089 short(n[1]))
fetch[n[1]] = 1 # earliest unknown
Benoit Boissinot
document and fix findincoming...
r2339 for p in n[2:4]:
if p in m:
base[p] = 1 # latest known
mpm@selenic.com
Break apart hg.py...
r1089
Benoit Boissinot
document and fix findincoming...
r2339 for p in n[2:4]:
if p not in req and p not in m:
r.append(p)
req[p] = 1
mpm@selenic.com
Break apart hg.py...
r1089 seen[n[0]] = 1
if r:
reqcnt += 1
Benoit Boissinot
i18n part2: use '_' for all strings who are part of the user interface
r1402 self.ui.debug(_("request %d: %s\n") %
mpm@selenic.com
Break apart hg.py...
r1089 (reqcnt, " ".join(map(short, r))))
for p in range(0, len(r), 10):
for b in remote.branches(r[p:p+10]):
Benoit Boissinot
i18n part2: use '_' for all strings who are part of the user interface
r1402 self.ui.debug(_("received %s:%s\n") %
mpm@selenic.com
Break apart hg.py...
r1089 (short(b[0]), short(b[1])))
Benoit Boissinot
document and fix findincoming...
r2339 unknown.append(b)
mpm@selenic.com
Break apart hg.py...
r1089
# do binary search on the branches we found
while search:
n = search.pop(0)
reqcnt += 1
l = remote.between([(n[0], n[1])])[0]
l.append(n[1])
p = n[0]
f = 1
for i in l:
Benoit Boissinot
i18n part2: use '_' for all strings who are part of the user interface
r1402 self.ui.debug(_("narrowing %d:%d %s\n") % (f, len(l), short(i)))
mpm@selenic.com
Break apart hg.py...
r1089 if i in m:
if f <= 2:
Benoit Boissinot
i18n part2: use '_' for all strings who are part of the user interface
r1402 self.ui.debug(_("found new branch changeset %s\n") %
mpm@selenic.com
Break apart hg.py...
r1089 short(p))
fetch[p] = 1
base[i] = 1
else:
Benoit Boissinot
i18n part2: use '_' for all strings who are part of the user interface
r1402 self.ui.debug(_("narrowed branch search to %s:%s\n")
mpm@selenic.com
Break apart hg.py...
r1089 % (short(p), short(i)))
search.append((p, i))
break
p, f = i, f * 2
# sanity check our fetch list
for f in fetch.keys():
if f in m:
Benoit Boissinot
i18n part2: use '_' for all strings who are part of the user interface
r1402 raise repo.RepoError(_("already have changeset ") + short(f[:4]))
mpm@selenic.com
Break apart hg.py...
r1089
if base.keys() == [nullid]:
Vadim Gelfer
add -f/--force to pull, incoming, outgoing, to work on unrelated repo....
r1959 if force:
self.ui.warn(_("warning: repository is unrelated\n"))
else:
raise util.Abort(_("repository is unrelated"))
mpm@selenic.com
Break apart hg.py...
r1089
Benoit Boissinot
i18n part2: use '_' for all strings who are part of the user interface
r1402 self.ui.note(_("found new changesets starting at ") +
mpm@selenic.com
Break apart hg.py...
r1089 " ".join([short(f) for f in fetch]) + "\n")
Benoit Boissinot
i18n part2: use '_' for all strings who are part of the user interface
r1402 self.ui.debug(_("%d total queries\n") % reqcnt)
mpm@selenic.com
Break apart hg.py...
r1089
return fetch.keys()
Vadim Gelfer
add -f/--force to pull, incoming, outgoing, to work on unrelated repo....
r1959 def findoutgoing(self, remote, base=None, heads=None, force=False):
Thomas Arendsen Hein
Fix hg push and hg push -r sometimes creating new heads without --force....
r2021 """Return list of nodes that are roots of subsets not in remote
If base dict is specified, assume that these nodes and their parents
exist on the remote side.
If a list of heads is specified, return only nodes which are heads
or ancestors of these heads, and return a second element which
contains all remote heads which get new children.
"""
mpm@selenic.com
Break apart hg.py...
r1089 if base == None:
base = {}
Vadim Gelfer
add -f/--force to pull, incoming, outgoing, to work on unrelated repo....
r1959 self.findincoming(remote, base, heads, force=force)
mpm@selenic.com
Break apart hg.py...
r1089
Benoit Boissinot
i18n part2: use '_' for all strings who are part of the user interface
r1402 self.ui.debug(_("common changesets up to ")
mpm@selenic.com
Break apart hg.py...
r1089 + " ".join(map(short, base.keys())) + "\n")
remain = dict.fromkeys(self.changelog.nodemap)
# prune everything remote has from the tree
del remain[nullid]
remove = base.keys()
while remove:
n = remove.pop(0)
if n in remain:
del remain[n]
for p in self.changelog.parents(n):
remove.append(p)
# find every node whose parents have been pruned
subset = []
Thomas Arendsen Hein
Fix hg push and hg push -r sometimes creating new heads without --force....
r2021 # find every remote head that will get new children
updated_heads = {}
mpm@selenic.com
Break apart hg.py...
r1089 for n in remain:
p1, p2 = self.changelog.parents(n)
if p1 not in remain and p2 not in remain:
subset.append(n)
Thomas Arendsen Hein
Fix hg push and hg push -r sometimes creating new heads without --force....
r2021 if heads:
if p1 in heads:
updated_heads[p1] = True
if p2 in heads:
updated_heads[p2] = True
mpm@selenic.com
Break apart hg.py...
r1089
# this is the set of all roots we have to push
Thomas Arendsen Hein
Fix hg push and hg push -r sometimes creating new heads without --force....
r2021 if heads:
return subset, updated_heads.keys()
else:
return subset
mpm@selenic.com
Break apart hg.py...
r1089
Vadim Gelfer
fetch: hold lock and wlock across all operations
r2827 def pull(self, remote, heads=None, force=False, lock=None):
mylock = False
if not lock:
lock = self.lock()
mylock = True
mpm@selenic.com
Break apart hg.py...
r1089
Vadim Gelfer
fetch: hold lock and wlock across all operations
r2827 try:
fetch = self.findincoming(remote, force=force)
if fetch == [nullid]:
self.ui.status(_("requesting all changes\n"))
mpm@selenic.com
Break apart hg.py...
r1089
Vadim Gelfer
fetch: hold lock and wlock across all operations
r2827 if not fetch:
self.ui.status(_("no changes found\n"))
return 0
mpm@selenic.com
Break apart hg.py...
r1089
Vadim Gelfer
fetch: hold lock and wlock across all operations
r2827 if heads is None:
cg = remote.changegroup(fetch, 'pull')
else:
cg = remote.changegroupsubset(fetch, heads, 'pull')
return self.addchangegroup(cg, 'pull', remote.url())
finally:
if mylock:
lock.release()
mpm@selenic.com
Break apart hg.py...
r1089
Benoit Boissinot
add an optional argument to push only the specified revisions (push -r)
r1781 def push(self, remote, force=False, revs=None):
Vadim Gelfer
extend network protocol to stop clients from locking servers...
r2439 # there are two ways to push to remote repo:
#
# addchangegroup assumes local user can lock remote
# repo (local filesystem, old ssh servers).
#
# unbundle assumes local user cannot lock remote repo (new ssh
# servers, http servers).
mpm@selenic.com
Break apart hg.py...
r1089
Vadim Gelfer
add support for streaming clone....
r2612 if remote.capable('unbundle'):
Vadim Gelfer
localrepository.push: propagate return value
r2463 return self.push_unbundle(remote, force, revs)
return self.push_addchangegroup(remote, force, revs)
Vadim Gelfer
extend network protocol to stop clients from locking servers...
r2439
def prepush(self, remote, force, revs):
mpm@selenic.com
Break apart hg.py...
r1089 base = {}
Thomas Arendsen Hein
Fix hg push and hg push -r sometimes creating new heads without --force....
r2021 remote_heads = remote.heads()
inc = self.findincoming(remote, base, remote_heads, force=force)
mpm@selenic.com
Break apart hg.py...
r1089 if not force and inc:
Benoit Boissinot
i18n part2: use '_' for all strings who are part of the user interface
r1402 self.ui.warn(_("abort: unsynced remote changes!\n"))
Thomas Arendsen Hein
Fix hg push and hg push -r sometimes creating new heads without --force....
r2021 self.ui.status(_("(did you forget to sync?"
" use push -f to force)\n"))
Vadim Gelfer
extend network protocol to stop clients from locking servers...
r2439 return None, 1
mpm@selenic.com
Break apart hg.py...
r1089
Thomas Arendsen Hein
Fix hg push and hg push -r sometimes creating new heads without --force....
r2021 update, updated_heads = self.findoutgoing(remote, base, remote_heads)
Benoit Boissinot
add an optional argument to push only the specified revisions (push -r)
r1781 if revs is not None:
msng_cl, bases, heads = self.changelog.nodesbetween(update, revs)
else:
bases, heads = update, self.changelog.heads()
if not bases:
Benoit Boissinot
i18n part2: use '_' for all strings who are part of the user interface
r1402 self.ui.status(_("no changes found\n"))
Vadim Gelfer
extend network protocol to stop clients from locking servers...
r2439 return None, 1
mpm@selenic.com
Break apart hg.py...
r1089 elif not force:
Matt Mackall
Revert push new heads logic...
r2109 # FIXME we don't properly detect creation of new heads
# in the push -r case, assume the user knows what he's doing
if not revs and len(remote_heads) < len(heads) \
and remote_heads != [nullid]:
Benoit Boissinot
i18n part2: use '_' for all strings who are part of the user interface
r1402 self.ui.warn(_("abort: push creates new remote branches!\n"))
self.ui.status(_("(did you forget to merge?"
" use push -f to force)\n"))
Vadim Gelfer
extend network protocol to stop clients from locking servers...
r2439 return None, 1
mpm@selenic.com
Break apart hg.py...
r1089
Benoit Boissinot
add an optional argument to push only the specified revisions (push -r)
r1781 if revs is None:
Benoit Boissinot
merge with crew
r1782 cg = self.changegroup(update, 'push')
Benoit Boissinot
add an optional argument to push only the specified revisions (push -r)
r1781 else:
Benoit Boissinot
merge with crew
r1782 cg = self.changegroupsubset(update, revs, 'push')
Vadim Gelfer
extend network protocol to stop clients from locking servers...
r2439 return cg, remote_heads
def push_addchangegroup(self, remote, force, revs):
lock = remote.lock()
ret = self.prepush(remote, force, revs)
if ret[0] is not None:
cg, remote_heads = ret
Vadim Gelfer
hooks: add url to changegroup, incoming, prechangegroup, pretxnchangegroup hooks...
r2673 return remote.addchangegroup(cg, 'push', self.url())
Vadim Gelfer
extend network protocol to stop clients from locking servers...
r2439 return ret[1]
def push_unbundle(self, remote, force, revs):
# local repo finds heads on server, finds out what revs it
# must push. once revs transferred, if server finds it has
# different heads (someone else won commit/push race), server
# aborts.
ret = self.prepush(remote, force, revs)
if ret[0] is not None:
cg, remote_heads = ret
if force: remote_heads = ['force']
return remote.unbundle(cg, remote_heads, 'push')
return ret[1]
mpm@selenic.com
Break apart hg.py...
r1089
Vadim Gelfer
add preoutgoing and outgoing hooks....
r1736 def changegroupsubset(self, bases, heads, source):
Eric Hopper
Added a lot of comments to changegroupsubset.
r1466 """This function generates a changegroup consisting of all the nodes
that are descendents of any of the bases, and ancestors of any of
the heads.
It is fairly complex as determining which filenodes and which
manifest nodes need to be included for the changeset to be complete
is non-trivial.
Another wrinkle is doing the reverse, figuring out which changeset in
the changegroup a particular filenode or manifestnode belongs to."""
Vadim Gelfer
add preoutgoing and outgoing hooks....
r1736 self.hook('preoutgoing', throw=True, source=source)
Eric Hopper
Added a lot of comments to changegroupsubset.
r1466 # Set up some initial variables
# Make it easy to refer to self.changelog
Eric Hopper
This changes the revlog.group and re-implements the localrepo.changeroup...
r1458 cl = self.changelog
Eric Hopper
Added a lot of comments to changegroupsubset.
r1466 # msng is short for missing - compute the list of changesets in this
# changegroup.
Eric Hopper
Bug fixing in localrepository.changegroupsubset. Bugs found in testing.
r1460 msng_cl_lst, bases, heads = cl.nodesbetween(bases, heads)
Eric Hopper
Added a lot of comments to changegroupsubset.
r1466 # Some bases may turn out to be superfluous, and some heads may be
# too. nodesbetween will return the minimal set of bases and heads
# necessary to re-create the changegroup.
# Known heads are the list of heads that it is assumed the recipient
# of this changegroup will know about.
Eric Hopper
This changes the revlog.group and re-implements the localrepo.changeroup...
r1458 knownheads = {}
Eric Hopper
Added a lot of comments to changegroupsubset.
r1466 # We assume that all parents of bases are known heads.
Eric Hopper
Bug fixing in localrepository.changegroupsubset. Bugs found in testing.
r1460 for n in bases:
Eric Hopper
This changes the revlog.group and re-implements the localrepo.changeroup...
r1458 for p in cl.parents(n):
if p != nullid:
knownheads[p] = 1
knownheads = knownheads.keys()
Eric Hopper
Bug fixing in localrepository.changegroupsubset. Bugs found in testing.
r1460 if knownheads:
Eric Hopper
Added a lot of comments to changegroupsubset.
r1466 # Now that we know what heads are known, we can compute which
# changesets are known. The recipient must know about all
# changesets required to reach the known heads from the null
# changeset.
Eric Hopper
Bug fixing in localrepository.changegroupsubset. Bugs found in testing.
r1460 has_cl_set, junk, junk = cl.nodesbetween(None, knownheads)
Eric Hopper
Added a lot of comments to changegroupsubset.
r1466 junk = None
# Transform the list into an ersatz set.
Eric Hopper
Fixed silly bug involving a non-existent variable.
r1464 has_cl_set = dict.fromkeys(has_cl_set)
Eric Hopper
Bug fixing in localrepository.changegroupsubset. Bugs found in testing.
r1460 else:
Eric Hopper
Added a lot of comments to changegroupsubset.
r1466 # If there were no known heads, the recipient cannot be assumed to
# know about any changesets.
Eric Hopper
Bug fixing in localrepository.changegroupsubset. Bugs found in testing.
r1460 has_cl_set = {}
Eric Hopper
This changes the revlog.group and re-implements the localrepo.changeroup...
r1458
Eric Hopper
Added a lot of comments to changegroupsubset.
r1466 # Make it easy to refer to self.manifest
Eric Hopper
This changes the revlog.group and re-implements the localrepo.changeroup...
r1458 mnfst = self.manifest
Eric Hopper
Added a lot of comments to changegroupsubset.
r1466 # We don't know which manifests are missing yet
Eric Hopper
This changes the revlog.group and re-implements the localrepo.changeroup...
r1458 msng_mnfst_set = {}
Eric Hopper
Added a lot of comments to changegroupsubset.
r1466 # Nor do we know which filenodes are missing.
Eric Hopper
This changes the revlog.group and re-implements the localrepo.changeroup...
r1458 msng_filenode_set = {}
Eric Hopper
Bug fixing in localrepository.changegroupsubset. Bugs found in testing.
r1460 junk = mnfst.index[mnfst.count() - 1] # Get around a bug in lazyindex
junk = None
Eric Hopper
Added a lot of comments to changegroupsubset.
r1466 # A changeset always belongs to itself, so the changenode lookup
# function for a changenode is identity.
Eric Hopper
This changes the revlog.group and re-implements the localrepo.changeroup...
r1458 def identity(x):
return x
Eric Hopper
Added a lot of comments to changegroupsubset.
r1466 # A function generating function. Sets up an environment for the
# inner function.
Eric Hopper
This changes the revlog.group and re-implements the localrepo.changeroup...
r1458 def cmp_by_rev_func(revlog):
Eric Hopper
Added a lot of comments to changegroupsubset.
r1466 # Compare two nodes by their revision number in the environment's
# revision history. Since the revision number both represents the
# most efficient order to read the nodes in, and represents a
# topological sorting of the nodes, this function is often useful.
def cmp_by_rev(a, b):
Eric Hopper
This changes the revlog.group and re-implements the localrepo.changeroup...
r1458 return cmp(revlog.rev(a), revlog.rev(b))
Eric Hopper
Added a lot of comments to changegroupsubset.
r1466 return cmp_by_rev
Eric Hopper
This changes the revlog.group and re-implements the localrepo.changeroup...
r1458
Eric Hopper
Added a lot of comments to changegroupsubset.
r1466 # If we determine that a particular file or manifest node must be a
# node that the recipient of the changegroup will already have, we can
# also assume the recipient will have all the parents. This function
# prunes them from the set of missing nodes.
Eric Hopper
This changes the revlog.group and re-implements the localrepo.changeroup...
r1458 def prune_parents(revlog, hasset, msngset):
haslst = hasset.keys()
haslst.sort(cmp_by_rev_func(revlog))
for node in haslst:
parentlst = [p for p in revlog.parents(node) if p != nullid]
while parentlst:
n = parentlst.pop()
if n not in hasset:
hasset[n] = 1
p = [p for p in revlog.parents(n) if p != nullid]
parentlst.extend(p)
for n in hasset:
msngset.pop(n, None)
Eric Hopper
Added a lot of comments to changegroupsubset.
r1466 # This is a function generating function used to set up an environment
# for the inner function to execute in.
Eric Hopper
This changes the revlog.group and re-implements the localrepo.changeroup...
r1458 def manifest_and_file_collector(changedfileset):
Eric Hopper
Added a lot of comments to changegroupsubset.
r1466 # This is an information gathering function that gathers
# information from each changeset node that goes out as part of
# the changegroup. The information gathered is a list of which
# manifest nodes are potentially required (the recipient may
# already have them) and total list of all files which were
# changed in any changeset in the changegroup.
#
# We also remember the first changenode we saw any manifest
# referenced by so we can later determine which changenode 'owns'
# the manifest.
Eric Hopper
This changes the revlog.group and re-implements the localrepo.changeroup...
r1458 def collect_manifests_and_files(clnode):
c = cl.read(clnode)
for f in c[3]:
# This is to make sure we only have one instance of each
# filename string for each filename.
Eric Hopper
Bug fixing in localrepository.changegroupsubset. Bugs found in testing.
r1460 changedfileset.setdefault(f, f)
msng_mnfst_set.setdefault(c[0], clnode)
Eric Hopper
This changes the revlog.group and re-implements the localrepo.changeroup...
r1458 return collect_manifests_and_files
Eric Hopper
Added a lot of comments to changegroupsubset.
r1466 # Figure out which manifest nodes (of the ones we think might be part
# of the changegroup) the recipient must know about and remove them
# from the changegroup.
Eric Hopper
This changes the revlog.group and re-implements the localrepo.changeroup...
r1458 def prune_manifests():
has_mnfst_set = {}
for n in msng_mnfst_set:
Eric Hopper
Added a lot of comments to changegroupsubset.
r1466 # If a 'missing' manifest thinks it belongs to a changenode
# the recipient is assumed to have, obviously the recipient
# must have that manifest.
Eric Hopper
This changes the revlog.group and re-implements the localrepo.changeroup...
r1458 linknode = cl.node(mnfst.linkrev(n))
if linknode in has_cl_set:
has_mnfst_set[n] = 1
prune_parents(mnfst, has_mnfst_set, msng_mnfst_set)
Eric Hopper
Added a lot of comments to changegroupsubset.
r1466 # Use the information collected in collect_manifests_and_files to say
# which changenode any manifestnode belongs to.
Eric Hopper
This changes the revlog.group and re-implements the localrepo.changeroup...
r1458 def lookup_manifest_link(mnfstnode):
return msng_mnfst_set[mnfstnode]
Eric Hopper
Added a lot of comments to changegroupsubset.
r1466 # A function generating function that sets up the initial environment
# the inner function.
Eric Hopper
This changes the revlog.group and re-implements the localrepo.changeroup...
r1458 def filenode_collector(changedfiles):
Eric Hopper
Optimizing manifest reads in changegroupsubset by using deltas.
r1462 next_rev = [0]
Eric Hopper
Added a lot of comments to changegroupsubset.
r1466 # This gathers information from each manifestnode included in the
# changegroup about which filenodes the manifest node references
# so we can include those in the changegroup too.
#
# It also remembers which changenode each filenode belongs to. It
# does this by assuming the a filenode belongs to the changenode
# the first manifest that references it belongs to.
Eric Hopper
This changes the revlog.group and re-implements the localrepo.changeroup...
r1458 def collect_msng_filenodes(mnfstnode):
Eric Hopper
Optimizing manifest reads in changegroupsubset by using deltas.
r1462 r = mnfst.rev(mnfstnode)
if r == next_rev[0]:
# If the last rev we looked at was the one just previous,
# we only need to see a diff.
delta = mdiff.patchtext(mnfst.delta(mnfstnode))
Eric Hopper
Added a lot of comments to changegroupsubset.
r1466 # For each line in the delta
Eric Hopper
Optimizing manifest reads in changegroupsubset by using deltas.
r1462 for dline in delta.splitlines():
Eric Hopper
Added a lot of comments to changegroupsubset.
r1466 # get the filename and filenode for that line
Eric Hopper
Optimizing manifest reads in changegroupsubset by using deltas.
r1462 f, fnode = dline.split('\0')
fnode = bin(fnode[:40])
f = changedfiles.get(f, None)
Eric Hopper
Added a lot of comments to changegroupsubset.
r1466 # And if the file is in the list of files we care
# about.
Eric Hopper
Optimizing manifest reads in changegroupsubset by using deltas.
r1462 if f is not None:
Eric Hopper
Added a lot of comments to changegroupsubset.
r1466 # Get the changenode this manifest belongs to
clnode = msng_mnfst_set[mnfstnode]
# Create the set of filenodes for the file if
# there isn't one already.
ndset = msng_filenode_set.setdefault(f, {})
# And set the filenode's changelog node to the
# manifest's if it hasn't been set already.
ndset.setdefault(fnode, clnode)
else:
# Otherwise we need a full manifest.
m = mnfst.read(mnfstnode)
# For every file in we care about.
for f in changedfiles:
fnode = m.get(f, None)
# If it's in the manifest
if fnode is not None:
# See comments above.
Eric Hopper
Optimizing manifest reads in changegroupsubset by using deltas.
r1462 clnode = msng_mnfst_set[mnfstnode]
ndset = msng_filenode_set.setdefault(f, {})
ndset.setdefault(fnode, clnode)
Eric Hopper
Added a lot of comments to changegroupsubset.
r1466 # Remember the revision we hope to see next.
Eric Hopper
Optimizing manifest reads in changegroupsubset by using deltas.
r1462 next_rev[0] = r + 1
Eric Hopper
Bug fixing in localrepository.changegroupsubset. Bugs found in testing.
r1460 return collect_msng_filenodes
Eric Hopper
This changes the revlog.group and re-implements the localrepo.changeroup...
r1458
Eric Hopper
Added a lot of comments to changegroupsubset.
r1466 # We have a list of filenodes we think we need for a file, lets remove
# all those we now the recipient must have.
Eric Hopper
This changes the revlog.group and re-implements the localrepo.changeroup...
r1458 def prune_filenodes(f, filerevlog):
msngset = msng_filenode_set[f]
hasset = {}
Eric Hopper
Added a lot of comments to changegroupsubset.
r1466 # If a 'missing' filenode thinks it belongs to a changenode we
# assume the recipient must have, then the recipient must have
# that filenode.
Eric Hopper
This changes the revlog.group and re-implements the localrepo.changeroup...
r1458 for n in msngset:
clnode = cl.node(filerevlog.linkrev(n))
if clnode in has_cl_set:
hasset[n] = 1
prune_parents(filerevlog, hasset, msngset)
Eric Hopper
Added a lot of comments to changegroupsubset.
r1466 # A function generator function that sets up the a context for the
# inner function.
Eric Hopper
This changes the revlog.group and re-implements the localrepo.changeroup...
r1458 def lookup_filenode_link_func(fname):
msngset = msng_filenode_set[fname]
Eric Hopper
Added a lot of comments to changegroupsubset.
r1466 # Lookup the changenode the filenode belongs to.
Eric Hopper
This changes the revlog.group and re-implements the localrepo.changeroup...
r1458 def lookup_filenode_link(fnode):
return msngset[fnode]
return lookup_filenode_link
mpm@selenic.com
Break apart hg.py...
r1089
Eric Hopper
Added a lot of comments to changegroupsubset.
r1466 # Now that we have all theses utility functions to help out and
# logically divide up the task, generate the group.
mpm@selenic.com
Break apart hg.py...
r1089 def gengroup():
Eric Hopper
Added a lot of comments to changegroupsubset.
r1466 # The set of changed files starts empty.
Eric Hopper
This changes the revlog.group and re-implements the localrepo.changeroup...
r1458 changedfiles = {}
Eric Hopper
Added a lot of comments to changegroupsubset.
r1466 # Create a changenode group generator that will call our functions
# back to lookup the owning changenode and collect information.
Eric Hopper
This changes the revlog.group and re-implements the localrepo.changeroup...
r1458 group = cl.group(msng_cl_lst, identity,
manifest_and_file_collector(changedfiles))
for chnk in group:
yield chnk
Eric Hopper
Added a lot of comments to changegroupsubset.
r1466
# The list of manifests has been collected by the generator
# calling our functions back.
Eric Hopper
This changes the revlog.group and re-implements the localrepo.changeroup...
r1458 prune_manifests()
msng_mnfst_lst = msng_mnfst_set.keys()
Eric Hopper
Added a lot of comments to changegroupsubset.
r1466 # Sort the manifestnodes by revision number.
Eric Hopper
This changes the revlog.group and re-implements the localrepo.changeroup...
r1458 msng_mnfst_lst.sort(cmp_by_rev_func(mnfst))
Eric Hopper
Added a lot of comments to changegroupsubset.
r1466 # Create a generator for the manifestnodes that calls our lookup
# and data collection functions back.
Eric Hopper
Bug fixing in localrepository.changegroupsubset. Bugs found in testing.
r1460 group = mnfst.group(msng_mnfst_lst, lookup_manifest_link,
Eric Hopper
This changes the revlog.group and re-implements the localrepo.changeroup...
r1458 filenode_collector(changedfiles))
for chnk in group:
yield chnk
Eric Hopper
Added a lot of comments to changegroupsubset.
r1466
# These are no longer needed, dereference and toss the memory for
# them.
Eric Hopper
This changes the revlog.group and re-implements the localrepo.changeroup...
r1458 msng_mnfst_lst = None
msng_mnfst_set.clear()
Eric Hopper
Added a lot of comments to changegroupsubset.
r1466
Eric Hopper
Optimizing manifest reads in changegroupsubset by using deltas.
r1462 changedfiles = changedfiles.keys()
changedfiles.sort()
Eric Hopper
Added a lot of comments to changegroupsubset.
r1466 # Go through all our files in order sorted by name.
Eric Hopper
This changes the revlog.group and re-implements the localrepo.changeroup...
r1458 for fname in changedfiles:
filerevlog = self.file(fname)
Eric Hopper
Added a lot of comments to changegroupsubset.
r1466 # Toss out the filenodes that the recipient isn't really
# missing.
Eric Hopper
This fixes a bug that Chris Mason found. As for a test case, I can't...
r1630 if msng_filenode_set.has_key(fname):
prune_filenodes(fname, filerevlog)
msng_filenode_lst = msng_filenode_set[fname].keys()
else:
msng_filenode_lst = []
Eric Hopper
Added a lot of comments to changegroupsubset.
r1466 # If any filenodes are left, generate the group for them,
# otherwise don't bother.
Eric Hopper
This changes the revlog.group and re-implements the localrepo.changeroup...
r1458 if len(msng_filenode_lst) > 0:
Thomas Arendsen Hein
make incoming work via ssh (issue139); move chunk code into separate module....
r1981 yield changegroup.genchunk(fname)
Eric Hopper
Added a lot of comments to changegroupsubset.
r1466 # Sort the filenodes by their revision #
Eric Hopper
This changes the revlog.group and re-implements the localrepo.changeroup...
r1458 msng_filenode_lst.sort(cmp_by_rev_func(filerevlog))
Eric Hopper
Added a lot of comments to changegroupsubset.
r1466 # Create a group generator and only pass in a changenode
# lookup function as we need to collect no information
# from filenodes.
Eric Hopper
This changes the revlog.group and re-implements the localrepo.changeroup...
r1458 group = filerevlog.group(msng_filenode_lst,
Eric Hopper
Bug fixing in localrepository.changegroupsubset. Bugs found in testing.
r1460 lookup_filenode_link_func(fname))
Eric Hopper
This changes the revlog.group and re-implements the localrepo.changeroup...
r1458 for chnk in group:
yield chnk
Eric Hopper
This fixes a bug that Chris Mason found. As for a test case, I can't...
r1630 if msng_filenode_set.has_key(fname):
# Don't need this anymore, toss it to free memory.
del msng_filenode_set[fname]
Eric Hopper
Added a lot of comments to changegroupsubset.
r1466 # Signal that no more groups are left.
Thomas Arendsen Hein
make incoming work via ssh (issue139); move chunk code into separate module....
r1981 yield changegroup.closechunk()
mpm@selenic.com
Break apart hg.py...
r1089
Benoit Boissinot
fix a NameError in changegroupsubset
r2150 if msng_cl_lst:
Vincent Danjean
allow to pull from an empty repo without getting a backtrace
r2149 self.hook('outgoing', node=hex(msng_cl_lst[0]), source=source)
Vadim Gelfer
add preoutgoing and outgoing hooks....
r1736
Eric Hopper
This changes the revlog.group and re-implements the localrepo.changeroup...
r1458 return util.chunkbuffer(gengroup())
Vadim Gelfer
add preoutgoing and outgoing hooks....
r1736 def changegroup(self, basenodes, source):
Eric Hopper
Added a lot of comments to changegroupsubset.
r1466 """Generate a changegroup of all nodes that we have that a recipient
doesn't.
This is much easier than the previous function as we can assume that
the recipient has any changenode we aren't sending them."""
Vadim Gelfer
add preoutgoing and outgoing hooks....
r1736
self.hook('preoutgoing', throw=True, source=source)
Eric Hopper
This changes the revlog.group and re-implements the localrepo.changeroup...
r1458 cl = self.changelog
nodes = cl.nodesbetween(basenodes, None)[0]
revset = dict.fromkeys([cl.rev(n) for n in nodes])
def identity(x):
return x
mpm@selenic.com
Break apart hg.py...
r1089
Eric Hopper
This changes the revlog.group and re-implements the localrepo.changeroup...
r1458 def gennodelst(revlog):
for r in xrange(0, revlog.count()):
n = revlog.node(r)
if revlog.linkrev(n) in revset:
yield n
def changed_file_collector(changedfileset):
def collect_changed_files(clnode):
c = cl.read(clnode)
for fname in c[3]:
changedfileset[fname] = 1
return collect_changed_files
def lookuprevlink_func(revlog):
def lookuprevlink(n):
return cl.node(revlog.linkrev(n))
return lookuprevlink
def gengroup():
mpm@selenic.com
Break apart hg.py...
r1089 # construct a list of all changed files
Eric Hopper
This changes the revlog.group and re-implements the localrepo.changeroup...
r1458 changedfiles = {}
for chnk in cl.group(nodes, identity,
changed_file_collector(changedfiles)):
yield chnk
changedfiles = changedfiles.keys()
changedfiles.sort()
mpm@selenic.com
Break apart hg.py...
r1089
Eric Hopper
This changes the revlog.group and re-implements the localrepo.changeroup...
r1458 mnfst = self.manifest
nodeiter = gennodelst(mnfst)
for chnk in mnfst.group(nodeiter, lookuprevlink_func(mnfst)):
yield chnk
mpm@selenic.com
Break apart hg.py...
r1089
Eric Hopper
This changes the revlog.group and re-implements the localrepo.changeroup...
r1458 for fname in changedfiles:
filerevlog = self.file(fname)
nodeiter = gennodelst(filerevlog)
nodeiter = list(nodeiter)
if nodeiter:
Thomas Arendsen Hein
make incoming work via ssh (issue139); move chunk code into separate module....
r1981 yield changegroup.genchunk(fname)
Eric Hopper
This changes the revlog.group and re-implements the localrepo.changeroup...
r1458 lookup = lookuprevlink_func(filerevlog)
for chnk in filerevlog.group(nodeiter, lookup):
yield chnk
mpm@selenic.com
Break apart hg.py...
r1089
Thomas Arendsen Hein
make incoming work via ssh (issue139); move chunk code into separate module....
r1981 yield changegroup.closechunk()
Matt Mackall
Don't die calling outgoing hook if we have no changesets
r2107
if nodes:
self.hook('outgoing', node=hex(nodes[0]), source=source)
mpm@selenic.com
Break apart hg.py...
r1089
Eric Hopper
This changes the revlog.group and re-implements the localrepo.changeroup...
r1458 return util.chunkbuffer(gengroup())
mpm@selenic.com
Break apart hg.py...
r1089
Vadim Gelfer
hooks: add url to changegroup, incoming, prechangegroup, pretxnchangegroup hooks...
r2673 def addchangegroup(self, source, srctype, url):
Vadim Gelfer
add merge command. means same thing as "update -m"....
r2019 """add changegroup to repo.
returns number of heads modified or added + 1."""
mpm@selenic.com
Break apart hg.py...
r1089
def csmap(x):
Benoit Boissinot
i18n part2: use '_' for all strings who are part of the user interface
r1402 self.ui.debug(_("add changeset %s\n") % short(x))
Vadim Gelfer
fix race in localrepo.addchangegroup....
r1998 return cl.count()
mpm@selenic.com
Break apart hg.py...
r1089
def revmap(x):
Vadim Gelfer
fix race in localrepo.addchangegroup....
r1998 return cl.rev(x)
mpm@selenic.com
Break apart hg.py...
r1089
Thomas Arendsen Hein
Cleanup of indentation, spacing, newlines, strings and line length
r1615 if not source:
Vadim Gelfer
add merge command. means same thing as "update -m"....
r2019 return 0
Vadim Gelfer
add prechangegroup and pretxnchangegroup hooks....
r1730
Vadim Gelfer
hooks: add url to changegroup, incoming, prechangegroup, pretxnchangegroup hooks...
r2673 self.hook('prechangegroup', throw=True, source=srctype, url=url)
Vadim Gelfer
add prechangegroup and pretxnchangegroup hooks....
r1730
mpm@selenic.com
Break apart hg.py...
r1089 changesets = files = revisions = 0
tr = self.transaction()
Benoit Boissinot
remove appendfile for the manifest when adding a changegroup...
r2395 # write changelog data to temp files so concurrent readers will not see
# inconsistent view
Thomas Arendsen Hein
Always remove appendopener tmp files (fixes issue235)....
r2232 cl = None
try:
cl = appendfile.appendchangelog(self.opener, self.changelog.version)
Vadim Gelfer
fix race in localrepo.addchangegroup....
r1998
Thomas Arendsen Hein
Always remove appendopener tmp files (fixes issue235)....
r2232 oldheads = len(cl.heads())
mpm@selenic.com
Break apart hg.py...
r1089
Thomas Arendsen Hein
Always remove appendopener tmp files (fixes issue235)....
r2232 # pull off the changeset group
self.ui.status(_("adding changesets\n"))
Benoit Boissinot
use a more reliable way to find what are the new changesets on pull/unbundle...
r2347 cor = cl.count() - 1
Thomas Arendsen Hein
Always remove appendopener tmp files (fixes issue235)....
r2232 chunkiter = changegroup.chunkiter(source)
Thomas Arendsen Hein
Manifest groups may be empty, so don't abort in this case (fixes issue210)....
r2354 if cl.addgroup(chunkiter, csmap, tr, 1) is None:
raise util.Abort(_("received changelog group is empty"))
Benoit Boissinot
use a more reliable way to find what are the new changesets on pull/unbundle...
r2347 cnr = cl.count() - 1
Thomas Arendsen Hein
Always remove appendopener tmp files (fixes issue235)....
r2232 changesets = cnr - cor
mpm@selenic.com
Break apart hg.py...
r1089
Benoit Boissinot
remove appendfile for the manifest when adding a changegroup...
r2395 # pull off the manifest group
self.ui.status(_("adding manifests\n"))
chunkiter = changegroup.chunkiter(source)
# no need to check for empty manifest group here:
# if the result of the merge of 1 and 2 is the same in 3 and 4,
# no new manifest will be created and the manifest group will
# be empty during the pull
self.manifest.addgroup(chunkiter, revmap, tr)
mpm@selenic.com
Break apart hg.py...
r1089
Benoit Boissinot
remove appendfile for the manifest when adding a changegroup...
r2395 # process the files
self.ui.status(_("adding file changes\n"))
while 1:
f = changegroup.getchunk(source)
if not f:
break
self.ui.debug(_("adding %s revisions\n") % f)
fl = self.file(f)
o = fl.count()
chunkiter = changegroup.chunkiter(source)
if fl.addgroup(chunkiter, revmap, tr) is None:
raise util.Abort(_("received file revlog group is empty"))
revisions += fl.count() - o
files += 1
mpm@selenic.com
Break apart hg.py...
r1089
Thomas Arendsen Hein
Always remove appendopener tmp files (fixes issue235)....
r2232 cl.writedata()
finally:
if cl:
cl.cleanup()
Vadim Gelfer
fix race in localrepo.addchangegroup....
r1998
Benoit Boissinot
remove appendfile for the manifest when adding a changegroup...
r2395 # make changelog see real files again
mason@suse.com
Additional appendfile fixes for interleaved data/index files...
r2082 self.changelog = changelog.changelog(self.opener, self.changelog.version)
mason@suse.com
Make the appendfile class inline-data index friendly...
r2075 self.changelog.checkinlinesize(tr)
Vadim Gelfer
fix race in localrepo.addchangegroup....
r1998
mpm@selenic.com
Break apart hg.py...
r1089 newheads = len(self.changelog.heads())
heads = ""
Thomas Arendsen Hein
Report on push/pull if heads are merged, too, like for new heads....
r2424 if oldheads and newheads != oldheads:
heads = _(" (%+d heads)") % (newheads - oldheads)
mpm@selenic.com
Break apart hg.py...
r1089
Benoit Boissinot
i18n part2: use '_' for all strings who are part of the user interface
r1402 self.ui.status(_("added %d changesets"
" with %d changes to %d files%s\n")
% (changesets, revisions, files, heads))
mpm@selenic.com
Break apart hg.py...
r1089
Benoit Boissinot
fix a traceback when unbundling does not add any changesets...
r2259 if changesets > 0:
self.hook('pretxnchangegroup', throw=True,
Vadim Gelfer
hooks: add url to changegroup, incoming, prechangegroup, pretxnchangegroup hooks...
r2673 node=hex(self.changelog.node(cor+1)), source=srctype,
url=url)
Vadim Gelfer
add prechangegroup and pretxnchangegroup hooks....
r1730
mpm@selenic.com
Break apart hg.py...
r1089 tr.close()
Benoit Boissinot
Fix traceback when nothing was added during unbundle...
r1375 if changesets > 0:
Vadim Gelfer
changegroup hooks: add source to hook parameters
r2229 self.hook("changegroup", node=hex(self.changelog.node(cor+1)),
Vadim Gelfer
hooks: add url to changegroup, incoming, prechangegroup, pretxnchangegroup hooks...
r2673 source=srctype, url=url)
mpm@selenic.com
Break apart hg.py...
r1089
Benoit Boissinot
Fix traceback when nothing was added during unbundle...
r1375 for i in range(cor + 1, cnr + 1):
Vadim Gelfer
changegroup hooks: add source to hook parameters
r2229 self.hook("incoming", node=hex(self.changelog.node(i)),
Vadim Gelfer
hooks: add url to changegroup, incoming, prechangegroup, pretxnchangegroup hooks...
r2673 source=srctype, url=url)
mpm@selenic.com
Hook fixups...
r1316
Vadim Gelfer
add merge command. means same thing as "update -m"....
r2019 return newheads - oldheads + 1
mpm@selenic.com
Break apart hg.py...
r1089
Vadim Gelfer
add support for streaming clone....
r2612 def stream_in(self, remote):
Vadim Gelfer
clone: disable stream support on server side by default....
r2621 fp = remote.stream_out()
resp = int(fp.readline())
if resp != 0:
raise util.Abort(_('operation forbidden by server'))
Vadim Gelfer
add support for streaming clone....
r2612 self.ui.status(_('streaming all changes\n'))
total_files, total_bytes = map(int, fp.readline().split(' ', 1))
self.ui.status(_('%d files to transfer, %s of data\n') %
(total_files, util.bytecount(total_bytes)))
start = time.time()
for i in xrange(total_files):
name, size = fp.readline().split('\0', 1)
size = int(size)
self.ui.debug('adding %s (%s)\n' % (name, util.bytecount(size)))
ofp = self.opener(name, 'w')
for chunk in util.filechunkiter(fp, limit=size):
ofp.write(chunk)
ofp.close()
elapsed = time.time() - start
self.ui.status(_('transferred %s in %.1f seconds (%s/sec)\n') %
(util.bytecount(total_bytes), elapsed,
util.bytecount(total_bytes / elapsed)))
self.reload()
return len(self.heads()) + 1
mpm@selenic.com
Break apart hg.py...
r1089
Vadim Gelfer
clone: do not make streaming default. add --stream option instead.
r2613 def clone(self, remote, heads=[], stream=False):
Vadim Gelfer
add support for streaming clone....
r2612 '''clone remote repository.
Matt Mackall
hg verify: more consistency checking between changesets and manifests
r1382
Vadim Gelfer
add support for streaming clone....
r2612 keyword arguments:
heads: list of revs to clone (forces use of pull)
Vadim Gelfer
clone: disable stream support on server side by default....
r2621 stream: use streaming clone if possible'''
mpm@selenic.com
Break apart hg.py...
r1089
Vadim Gelfer
clone: disable stream support on server side by default....
r2621 # now, all clients that can request uncompressed clones can
# read repo formats supported by all servers that can serve
# them.
mpm@selenic.com
Break apart hg.py...
r1089
Vadim Gelfer
add support for streaming clone....
r2612 # if revlog format changes, client will have to check version
Vadim Gelfer
clone: disable stream support on server side by default....
r2621 # and format flags on "stream" capability, and use
# uncompressed only if compatible.
mpm@selenic.com
Break apart hg.py...
r1089
Vadim Gelfer
clone: do not make streaming default. add --stream option instead.
r2613 if stream and not heads and remote.capable('stream'):
Vadim Gelfer
add support for streaming clone....
r2612 return self.stream_in(remote)
return self.pull(remote, heads)
mason@suse.com
Automatic nesting into running transactions in the same repository....
r1806
# used to avoid circular references so destructors work
def aftertrans(base):
p = base
def a():
util.rename(os.path.join(p, "journal"), os.path.join(p, "undo"))
util.rename(os.path.join(p, "journal.dirstate"),
os.path.join(p, "undo.dirstate"))
return a
Vadim Gelfer
clean up hg.py: move repo constructor code into each repo module
r2740 def instance(ui, path, create):
return localrepository(ui, util.drop_scheme('file', path), create)
def islocal(path):
return True