manifest.py
198 lines
| 7.4 KiB
| text/x-python
|
PythonLexer
/ mercurial / manifest.py
mpm@selenic.com
|
r1089 | # manifest.py - manifest revision class for mercurial | ||
# | ||||
Thomas Arendsen Hein
|
r4635 | # Copyright 2005-2007 Matt Mackall <mpm@selenic.com> | ||
mpm@selenic.com
|
r1089 | # | ||
Martin Geisler
|
r8225 | # This software may be used and distributed according to the terms of the | ||
# GNU General Public License version 2, incorporated herein by reference. | ||||
mpm@selenic.com
|
r1089 | |||
Matt Mackall
|
r3891 | from i18n import _ | ||
Peter Arrenbrecht
|
r8390 | import mdiff, parsers, error, revlog | ||
Simon Heimberg
|
r8312 | import array, struct | ||
mpm@selenic.com
|
r1089 | |||
Matt Mackall
|
r2835 | class manifestdict(dict): | ||
Alexis S. L. Carvalho
|
r2857 | def __init__(self, mapping=None, flags=None): | ||
if mapping is None: mapping = {} | ||||
if flags is None: flags = {} | ||||
Matt Mackall
|
r2831 | dict.__init__(self, mapping) | ||
Matt Mackall
|
r2839 | self._flags = flags | ||
Matt Mackall
|
r2834 | def flags(self, f): | ||
Matt Mackall
|
r2839 | return self._flags.get(f, "") | ||
Matt Mackall
|
r6743 | def set(self, f, flags): | ||
self._flags[f] = flags | ||||
Matt Mackall
|
r2831 | def copy(self): | ||
Benoit Boissinot
|
r9416 | return manifestdict(self, dict.copy(self._flags)) | ||
Matt Mackall
|
r2831 | |||
Matt Mackall
|
r7634 | class manifest(revlog.revlog): | ||
Matt Mackall
|
r4258 | def __init__(self, opener): | ||
Benoit Boissinot
|
r9414 | self._mancache = None | ||
Matt Mackall
|
r7634 | revlog.revlog.__init__(self, opener, "00manifest.i") | ||
mpm@selenic.com
|
r1089 | |||
Matt Mackall
|
r4995 | def parse(self, lines): | ||
mfdict = manifestdict() | ||||
Bryan O'Sullivan
|
r6389 | parsers.parse_manifest(mfdict, mfdict._flags, lines) | ||
Matt Mackall
|
r4995 | return mfdict | ||
Brendan Cully
|
r3196 | |||
def readdelta(self, node): | ||||
Matt Mackall
|
r7362 | r = self.rev(node) | ||
return self.parse(mdiff.patchtext(self.revdiff(r - 1, r))) | ||||
Thomas Arendsen Hein
|
r3223 | |||
mpm@selenic.com
|
r1089 | def read(self, node): | ||
Matt Mackall
|
r7634 | if node == revlog.nullid: | ||
return manifestdict() # don't upset local cache | ||||
Benoit Boissinot
|
r9414 | if self._mancache and self._mancache[0] == node: | ||
return self._mancache[1] | ||||
mpm@selenic.com
|
r1089 | text = self.revision(node) | ||
Benoit Boissinot
|
r9414 | arraytext = array.array('c', text) | ||
Matt Mackall
|
r4995 | mapping = self.parse(text) | ||
Benoit Boissinot
|
r9414 | self._mancache = (node, mapping, arraytext) | ||
Matt Mackall
|
r2835 | return mapping | ||
mpm@selenic.com
|
r1089 | |||
Vadim Gelfer
|
r2320 | def _search(self, m, s, lo=0, hi=None): | ||
'''return a tuple (start, end) that says where to find s within m. | ||||
If the string is found m[start:end] are the line containing | ||||
that string. If start == end the string was not found and | ||||
they indicate the proper sorted insertion point. This was | ||||
taken from bisect_left, and modified to find line start/end as | ||||
it goes along. | ||||
m should be a buffer or a string | ||||
s is a string''' | ||||
def advance(i, c): | ||||
while i < lenm and m[i] != c: | ||||
i += 1 | ||||
return i | ||||
Patrick Mezard
|
r7405 | if not s: | ||
return (lo, lo) | ||||
Vadim Gelfer
|
r2320 | lenm = len(m) | ||
if not hi: | ||||
hi = lenm | ||||
while lo < hi: | ||||
mid = (lo + hi) // 2 | ||||
start = mid | ||||
while start > 0 and m[start-1] != '\n': | ||||
start -= 1 | ||||
end = advance(start, '\0') | ||||
if m[start:end] < s: | ||||
# we know that after the null there are 40 bytes of sha1 | ||||
# this translates to the bisect lo = mid + 1 | ||||
lo = advance(end + 40, '\n') + 1 | ||||
else: | ||||
# this translates to the bisect hi = mid | ||||
hi = start | ||||
end = advance(lo, '\0') | ||||
found = m[lo:end] | ||||
if cmp(s, found) == 0: | ||||
# we know that after the null there are 40 bytes of sha1 | ||||
end = advance(end + 40, '\n') | ||||
return (lo, end+1) | ||||
else: | ||||
return (lo, lo) | ||||
def find(self, node, f): | ||||
'''look up entry for a single file efficiently. | ||||
Alexis S. L. Carvalho
|
r4159 | return (node, flags) pair if found, (None, None) if not.''' | ||
Benoit Boissinot
|
r9414 | if self._mancache and self._mancache[0] == node: | ||
return self._mancache[1].get(f), self._mancache[1].flags(f) | ||||
Vadim Gelfer
|
r2320 | text = self.revision(node) | ||
start, end = self._search(text, f) | ||||
if start == end: | ||||
return None, None | ||||
l = text[start:end] | ||||
f, n = l.split('\0') | ||||
Matt Mackall
|
r7634 | return revlog.bin(n[:40]), n[40:-1] | ||
Vadim Gelfer
|
r2320 | |||
Matt Mackall
|
r2841 | def add(self, map, transaction, link, p1=None, p2=None, | ||
mpm@selenic.com
|
r1089 | changed=None): | ||
# apply the changes collected during the bisect loop to our addlist | ||||
mason@suse.com
|
r1534 | # return a delta suitable for addrevision | ||
def addlistdelta(addlist, x): | ||||
# start from the bottom up | ||||
mpm@selenic.com
|
r1089 | # so changes to the offsets don't mess things up. | ||
Benoit Boissinot
|
r9413 | for start, end, content in reversed(x): | ||
if content: | ||||
addlist[start:end] = array.array('c', content) | ||||
mpm@selenic.com
|
r1089 | else: | ||
del addlist[start:end] | ||||
Benoit Boissinot
|
r9413 | return "".join(struct.pack(">lll", start, end, len(content)) + content | ||
for start, end, content in x) | ||||
mpm@selenic.com
|
r1089 | |||
Matt Mackall
|
r6765 | def checkforbidden(l): | ||
for f in l: | ||||
if '\n' in f or '\r' in f: | ||||
Matt Mackall
|
r7633 | raise error.RevlogError( | ||
Greg Ward
|
r8077 | _("'\\n' and '\\r' disallowed in filenames: %r") % f) | ||
Benoit Boissinot
|
r3607 | |||
Benoit Boissinot
|
r9414 | # if we're using the cache, make sure it is valid and | ||
mpm@selenic.com
|
r1089 | # parented by the same node we're diffing against | ||
Benoit Boissinot
|
r9414 | if not (changed and self._mancache and p1 and self._mancache[0] == p1): | ||
Matt Mackall
|
r8209 | files = sorted(map) | ||
Matt Mackall
|
r6765 | checkforbidden(files) | ||
Benoit Boissinot
|
r3607 | |||
Matt Mackall
|
r1651 | # if this is changed to support newlines in filenames, | ||
# be sure to check the templates/ dir again (especially *-raw.tmpl) | ||||
Matt Mackall
|
r7634 | hex, flags = revlog.hex, map.flags | ||
Benoit Boissinot
|
r9420 | text = ''.join("%s\000%s%s\n" % (f, hex(map[f]), flags(f)) | ||
for f in files) | ||||
arraytext = array.array('c', text) | ||||
mpm@selenic.com
|
r1089 | cachedelta = None | ||
else: | ||||
Benoit Boissinot
|
r9415 | added, removed = changed | ||
Benoit Boissinot
|
r9414 | addlist = self._mancache[2] | ||
mpm@selenic.com
|
r1089 | |||
Benoit Boissinot
|
r9415 | checkforbidden(added) | ||
mpm@selenic.com
|
r1089 | # combine the changed lists into one list for sorting | ||
Benoit Boissinot
|
r9415 | work = [(x, False) for x in added] | ||
work.extend((x, True) for x in removed) | ||||
# this could use heapq.merge() (from python2.6+) or equivalent | ||||
# since the lists are already sorted | ||||
mpm@selenic.com
|
r1089 | work.sort() | ||
delta = [] | ||||
mason@suse.com
|
r1534 | dstart = None | ||
dend = None | ||||
dline = [""] | ||||
start = 0 | ||||
# zero copy representation of addlist as a buffer | ||||
addbuf = buffer(addlist) | ||||
mpm@selenic.com
|
r1089 | |||
mason@suse.com
|
r1534 | # start with a readonly loop that finds the offset of | ||
# each line and creates the deltas | ||||
Benoit Boissinot
|
r9415 | for f, todelete in work: | ||
mpm@selenic.com
|
r1089 | # bs will either be the index of the item or the insert point | ||
Vadim Gelfer
|
r2320 | start, end = self._search(addbuf, f, start) | ||
Benoit Boissinot
|
r9415 | if not todelete: | ||
Matt Mackall
|
r7634 | l = "%s\000%s%s\n" % (f, revlog.hex(map[f]), map.flags(f)) | ||
mpm@selenic.com
|
r1089 | else: | ||
Benoit Boissinot
|
r9415 | if start == end: | ||
# item we want to delete was not found, error out | ||||
raise AssertionError( | ||||
_("failed to remove %s from manifest") % f) | ||||
mason@suse.com
|
r1534 | l = "" | ||
if dstart != None and dstart <= start and dend >= start: | ||||
if dend < end: | ||||
dend = end | ||||
if l: | ||||
dline.append(l) | ||||
mpm@selenic.com
|
r1089 | else: | ||
mason@suse.com
|
r1534 | if dstart != None: | ||
delta.append([dstart, dend, "".join(dline)]) | ||||
dstart = start | ||||
dend = end | ||||
dline = [l] | ||||
mpm@selenic.com
|
r1089 | |||
mason@suse.com
|
r1534 | if dstart != None: | ||
delta.append([dstart, dend, "".join(dline)]) | ||||
# apply the delta to the addlist, and get a delta for addrevision | ||||
cachedelta = addlistdelta(addlist, delta) | ||||
mpm@selenic.com
|
r1089 | |||
mason@suse.com
|
r1534 | # the delta is only valid if we've been processing the tip revision | ||
Benoit Boissinot
|
r9414 | if p1 != self.tip(): | ||
mason@suse.com
|
r1534 | cachedelta = None | ||
Benoit Boissinot
|
r9414 | arraytext = addlist | ||
Benoit Boissinot
|
r9420 | text = buffer(arraytext) | ||
mason@suse.com
|
r1534 | |||
Benoit Boissinot
|
r9420 | n = self.addrevision(text, transaction, link, p1, p2, cachedelta) | ||
Benoit Boissinot
|
r9414 | self._mancache = (n, map, arraytext) | ||
mpm@selenic.com
|
r1089 | |||
return n | ||||