changelog.py
225 lines
| 7.6 KiB
| text/x-python
|
PythonLexer
/ mercurial / changelog.py
mpm@selenic.com
|
r1095 | # changelog.py - changelog class for mercurial | ||
mpm@selenic.com
|
r1089 | # | ||
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 | |||
Joel Rosdahl
|
r6211 | from node import bin, hex, nullid | ||
Benoit Boissinot
|
r7035 | from i18n import _ | ||
Matt Mackall
|
r7948 | import util, error, revlog, encoding | ||
mpm@selenic.com
|
r1089 | |||
Benoit Boissinot
|
r3232 | def _string_escape(text): | ||
""" | ||||
>>> d = {'nl': chr(10), 'bs': chr(92), 'cr': chr(13), 'nul': chr(0)} | ||||
>>> s = "ab%(nl)scd%(bs)s%(bs)sn%(nul)sab%(cr)scd%(bs)s%(nl)s" % d | ||||
>>> s | ||||
'ab\\ncd\\\\\\\\n\\x00ab\\rcd\\\\\\n' | ||||
>>> res = _string_escape(s) | ||||
Matt Mackall
|
r5745 | >>> s == res.decode('string_escape') | ||
Benoit Boissinot
|
r3232 | True | ||
""" | ||||
# subset of the string_escape codec | ||||
text = text.replace('\\', '\\\\').replace('\n', '\\n').replace('\r', '\\r') | ||||
return text.replace('\0', '\\0') | ||||
Martin Geisler
|
r8443 | def decodeextra(text): | ||
extra = {} | ||||
for l in text.split('\0'): | ||||
if l: | ||||
k, v = l.decode('string_escape').split(':', 1) | ||||
extra[k] = v | ||||
return extra | ||||
def encodeextra(d): | ||||
# keys must be sorted to produce a deterministic changelog entry | ||||
items = [_string_escape('%s:%s' % (k, d[k])) for k in sorted(d)] | ||||
return "\0".join(items) | ||||
Matt Mackall
|
r4261 | class appender: | ||
timeless
|
r7807 | '''the changelog index must be updated last on disk, so we use this class | ||
Matt Mackall
|
r4261 | to delay writes to it''' | ||
def __init__(self, fp, buf): | ||||
self.data = buf | ||||
self.fp = fp | ||||
self.offset = fp.tell() | ||||
self.size = util.fstat(fp).st_size | ||||
def end(self): | ||||
return self.size + len("".join(self.data)) | ||||
def tell(self): | ||||
return self.offset | ||||
def flush(self): | ||||
pass | ||||
def close(self): | ||||
Benoit Boissinot
|
r4961 | self.fp.close() | ||
Matt Mackall
|
r4261 | |||
def seek(self, offset, whence=0): | ||||
'''virtual file offset spans real file and data''' | ||||
if whence == 0: | ||||
self.offset = offset | ||||
elif whence == 1: | ||||
self.offset += offset | ||||
elif whence == 2: | ||||
self.offset = self.end() + offset | ||||
if self.offset < self.size: | ||||
self.fp.seek(self.offset) | ||||
def read(self, count=-1): | ||||
'''only trick here is reads that span real file and data''' | ||||
ret = "" | ||||
if self.offset < self.size: | ||||
s = self.fp.read(count) | ||||
ret = s | ||||
self.offset += len(s) | ||||
if count > 0: | ||||
count -= len(s) | ||||
if count != 0: | ||||
doff = self.offset - self.size | ||||
self.data.insert(0, "".join(self.data)) | ||||
del self.data[1:] | ||||
s = self.data[0][doff:doff+count] | ||||
self.offset += len(s) | ||||
ret += s | ||||
return ret | ||||
def write(self, s): | ||||
Matt Mackall
|
r5450 | self.data.append(str(s)) | ||
Matt Mackall
|
r4261 | self.offset += len(s) | ||
Matt Mackall
|
r7634 | class changelog(revlog.revlog): | ||
Matt Mackall
|
r4258 | def __init__(self, opener): | ||
Matt Mackall
|
r7634 | revlog.revlog.__init__(self, opener, "00changelog.i") | ||
mpm@selenic.com
|
r1089 | |||
Matt Mackall
|
r4261 | def delayupdate(self): | ||
"delay visibility of index updates to other readers" | ||||
self._realopener = self.opener | ||||
Matt Mackall
|
r4269 | self.opener = self._delayopener | ||
Matt Mackall
|
r6750 | self._delaycount = len(self) | ||
Matt Mackall
|
r4261 | self._delaybuf = [] | ||
Matt Mackall
|
r4269 | self._delayname = None | ||
Matt Mackall
|
r4261 | |||
def finalize(self, tr): | ||||
"finalize index updates" | ||||
self.opener = self._realopener | ||||
Matt Mackall
|
r4269 | # move redirected index data back into place | ||
if self._delayname: | ||||
util.rename(self._delayname + ".a", self._delayname) | ||||
elif self._delaybuf: | ||||
Matt Mackall
|
r4261 | fp = self.opener(self.indexfile, 'a') | ||
fp.write("".join(self._delaybuf)) | ||||
fp.close() | ||||
Matt Mackall
|
r7787 | self._delaybuf = [] | ||
Matt Mackall
|
r4269 | # split when we're done | ||
Matt Mackall
|
r4261 | self.checkinlinesize(tr) | ||
Matt Mackall
|
r4269 | def _delayopener(self, name, mode='r'): | ||
Matt Mackall
|
r4261 | fp = self._realopener(name, mode) | ||
Matt Mackall
|
r4269 | # only divert the index | ||
Matt Mackall
|
r4261 | if not name == self.indexfile: | ||
return fp | ||||
Matt Mackall
|
r4269 | # if we're doing an initial clone, divert to another file | ||
if self._delaycount == 0: | ||||
self._delayname = fp.name | ||||
Matt Mackall
|
r6750 | if not len(self): | ||
Benoit Boissinot
|
r6259 | # make sure to truncate the file | ||
mode = mode.replace('a', 'w') | ||||
Matt Mackall
|
r4269 | return self._realopener(name + ".a", mode) | ||
# otherwise, divert to memory | ||||
Matt Mackall
|
r4261 | return appender(fp, self._delaybuf) | ||
Matt Mackall
|
r7787 | def readpending(self, file): | ||
r = revlog.revlog(self.opener, file) | ||||
self.index = r.index | ||||
self.nodemap = r.nodemap | ||||
self._chunkcache = r._chunkcache | ||||
def writepending(self): | ||||
"create a file containing the unfinalized state for pretxnchangegroup" | ||||
if self._delaybuf: | ||||
# make a temporary copy of the index | ||||
fp1 = self._realopener(self.indexfile) | ||||
fp2 = self._realopener(self.indexfile + ".a", "w") | ||||
fp2.write(fp1.read()) | ||||
# add pending data | ||||
fp2.write("".join(self._delaybuf)) | ||||
fp2.close() | ||||
# switch modes so finalize can simply rename | ||||
self._delaybuf = [] | ||||
self._delayname = fp1.name | ||||
if self._delayname: | ||||
return True | ||||
return False | ||||
Matt Mackall
|
r4261 | def checkinlinesize(self, tr, fp=None): | ||
Matt Mackall
|
r4269 | if self.opener == self._delayopener: | ||
Matt Mackall
|
r4261 | return | ||
Matt Mackall
|
r7634 | return revlog.revlog.checkinlinesize(self, tr, fp) | ||
Matt Mackall
|
r4261 | |||
Matt Mackall
|
r5744 | def read(self, node): | ||
Benoit Boissinot
|
r3077 | """ | ||
format used: | ||||
Benoit Boissinot
|
r3233 | nodeid\n : manifest node in ascii | ||
user\n : user, no \n or \r allowed | ||||
time tz extra\n : date (time is int or float, timezone is int) | ||||
: extra is metadatas, encoded and separated by '\0' | ||||
: older versions ignore it | ||||
files\n\n : files modified by the cset, no \n or \r allowed | ||||
(.*) : comment (free text, ideally utf-8) | ||||
changelog v0 doesn't use extra | ||||
Benoit Boissinot
|
r3077 | """ | ||
Matt Mackall
|
r5744 | text = self.revision(node) | ||
mpm@selenic.com
|
r1089 | if not text: | ||
Alexis S. L. Carvalho
|
r4176 | return (nullid, "", (0, 0), [], "", {'branch': 'default'}) | ||
mpm@selenic.com
|
r1089 | last = text.index("\n\n") | ||
Matt Mackall
|
r7948 | desc = encoding.tolocal(text[last + 2:]) | ||
Benoit Boissinot
|
r3233 | l = text[:last].split('\n') | ||
mpm@selenic.com
|
r1089 | manifest = bin(l[0]) | ||
Matt Mackall
|
r7948 | user = encoding.tolocal(l[1]) | ||
Benoit Boissinot
|
r3233 | |||
extra_data = l[2].split(' ', 2) | ||||
if len(extra_data) != 3: | ||||
time = float(extra_data.pop(0)) | ||||
try: | ||||
# various tools did silly things with the time zone field. | ||||
timezone = int(extra_data[0]) | ||||
except: | ||||
timezone = 0 | ||||
extra = {} | ||||
else: | ||||
time, timezone, extra = extra_data | ||||
time, timezone = float(time), int(timezone) | ||||
Martin Geisler
|
r8443 | extra = decodeextra(extra) | ||
Alexis S. L. Carvalho
|
r4176 | if not extra.get('branch'): | ||
extra['branch'] = 'default' | ||||
mpm@selenic.com
|
r1089 | files = l[3:] | ||
Benoit Boissinot
|
r3233 | return (manifest, user, (time, timezone), files, desc, extra) | ||
mpm@selenic.com
|
r1089 | |||
Martin Geisler
|
r8422 | def add(self, manifest, files, desc, transaction, p1, p2, | ||
user, date=None, extra={}): | ||||
Benoit Boissinot
|
r7035 | user = user.strip() | ||
Martin Geisler
|
r8424 | # An empty username or a username with a "\n" will make the | ||
# revision text contain two "\n\n" sequences -> corrupt | ||||
# repository since read cannot unpack the revision. | ||||
if not user: | ||||
raise error.RevlogError(_("empty username")) | ||||
Benoit Boissinot
|
r7035 | if "\n" in user: | ||
Matt Mackall
|
r7633 | raise error.RevlogError(_("username %s contains a newline") | ||
% repr(user)) | ||||
Matt Mackall
|
r7948 | user, desc = encoding.fromlocal(user), encoding.fromlocal(desc) | ||
Matt Mackall
|
r3771 | |||
Bryan O'Sullivan
|
r1195 | if date: | ||
Benoit Boissinot
|
r2523 | parseddate = "%d %d" % util.parsedate(date) | ||
Bryan O'Sullivan
|
r1195 | else: | ||
Jose M. Prieto
|
r2522 | parseddate = "%d %d" % util.makedate() | ||
Alexis S. L. Carvalho
|
r4176 | if extra and extra.get("branch") in ("default", ""): | ||
del extra["branch"] | ||||
Benoit Boissinot
|
r3233 | if extra: | ||
Martin Geisler
|
r8443 | extra = encodeextra(extra) | ||
Benoit Boissinot
|
r3233 | parseddate = "%s %s" % (parseddate, extra) | ||
Matt Mackall
|
r8209 | l = [hex(manifest), user, parseddate] + sorted(files) + ["", desc] | ||
mpm@selenic.com
|
r1089 | text = "\n".join(l) | ||
Matt Mackall
|
r6750 | return self.addrevision(text, transaction, len(self), p1, p2) | ||