##// END OF EJS Templates
tests: fix doctest stability over Python versions...
Mads Kiilerich -
r18379:e0c4f4ba default
parent child Browse files
Show More
@@ -1,348 +1,349 b''
1 1 # changelog.py - changelog class for mercurial
2 2 #
3 3 # Copyright 2005-2007 Matt Mackall <mpm@selenic.com>
4 4 #
5 5 # This software may be used and distributed according to the terms of the
6 6 # GNU General Public License version 2 or any later version.
7 7
8 8 from node import bin, hex, nullid
9 9 from i18n import _
10 10 import util, error, revlog, encoding
11 11
12 12 _defaultextra = {'branch': 'default'}
13 13
14 14 def _string_escape(text):
15 15 """
16 16 >>> d = {'nl': chr(10), 'bs': chr(92), 'cr': chr(13), 'nul': chr(0)}
17 17 >>> s = "ab%(nl)scd%(bs)s%(bs)sn%(nul)sab%(cr)scd%(bs)s%(nl)s" % d
18 18 >>> s
19 19 'ab\\ncd\\\\\\\\n\\x00ab\\rcd\\\\\\n'
20 20 >>> res = _string_escape(s)
21 21 >>> s == res.decode('string_escape')
22 22 True
23 23 """
24 24 # subset of the string_escape codec
25 25 text = text.replace('\\', '\\\\').replace('\n', '\\n').replace('\r', '\\r')
26 26 return text.replace('\0', '\\0')
27 27
28 28 def decodeextra(text):
29 29 """
30 >>> from pprint import pprint as pp
31 >>> pp(decodeextra(encodeextra({'foo': 'bar', 'baz': chr(0) + '2'})))
32 {'baz': '\\x002', 'branch': 'default', 'foo': 'bar'}
33 >>> pp(decodeextra(encodeextra({'foo': 'bar',
34 ... 'baz': chr(92) + chr(0) + '2'})))
35 {'baz': '\\\\\\x002', 'branch': 'default', 'foo': 'bar'}
30 >>> sorted(decodeextra(encodeextra({'foo': 'bar', 'baz': chr(0) + '2'})
31 ... ).iteritems())
32 [('baz', '\\x002'), ('branch', 'default'), ('foo', 'bar')]
33 >>> sorted(decodeextra(encodeextra({'foo': 'bar',
34 ... 'baz': chr(92) + chr(0) + '2'})
35 ... ).iteritems())
36 [('baz', '\\\\\\x002'), ('branch', 'default'), ('foo', 'bar')]
36 37 """
37 38 extra = _defaultextra.copy()
38 39 for l in text.split('\0'):
39 40 if l:
40 41 if '\\0' in l:
41 42 # fix up \0 without getting into trouble with \\0
42 43 l = l.replace('\\\\', '\\\\\n')
43 44 l = l.replace('\\0', '\0')
44 45 l = l.replace('\n', '')
45 46 k, v = l.decode('string_escape').split(':', 1)
46 47 extra[k] = v
47 48 return extra
48 49
49 50 def encodeextra(d):
50 51 # keys must be sorted to produce a deterministic changelog entry
51 52 items = [_string_escape('%s:%s' % (k, d[k])) for k in sorted(d)]
52 53 return "\0".join(items)
53 54
54 55 def stripdesc(desc):
55 56 """strip trailing whitespace and leading and trailing empty lines"""
56 57 return '\n'.join([l.rstrip() for l in desc.splitlines()]).strip('\n')
57 58
58 59 class appender(object):
59 60 '''the changelog index must be updated last on disk, so we use this class
60 61 to delay writes to it'''
61 62 def __init__(self, fp, buf):
62 63 self.data = buf
63 64 self.fp = fp
64 65 self.offset = fp.tell()
65 66 self.size = util.fstat(fp).st_size
66 67
67 68 def end(self):
68 69 return self.size + len("".join(self.data))
69 70 def tell(self):
70 71 return self.offset
71 72 def flush(self):
72 73 pass
73 74 def close(self):
74 75 self.fp.close()
75 76
76 77 def seek(self, offset, whence=0):
77 78 '''virtual file offset spans real file and data'''
78 79 if whence == 0:
79 80 self.offset = offset
80 81 elif whence == 1:
81 82 self.offset += offset
82 83 elif whence == 2:
83 84 self.offset = self.end() + offset
84 85 if self.offset < self.size:
85 86 self.fp.seek(self.offset)
86 87
87 88 def read(self, count=-1):
88 89 '''only trick here is reads that span real file and data'''
89 90 ret = ""
90 91 if self.offset < self.size:
91 92 s = self.fp.read(count)
92 93 ret = s
93 94 self.offset += len(s)
94 95 if count > 0:
95 96 count -= len(s)
96 97 if count != 0:
97 98 doff = self.offset - self.size
98 99 self.data.insert(0, "".join(self.data))
99 100 del self.data[1:]
100 101 s = self.data[0][doff:doff + count]
101 102 self.offset += len(s)
102 103 ret += s
103 104 return ret
104 105
105 106 def write(self, s):
106 107 self.data.append(str(s))
107 108 self.offset += len(s)
108 109
109 110 def delayopener(opener, target, divert, buf):
110 111 def o(name, mode='r'):
111 112 if name != target:
112 113 return opener(name, mode)
113 114 if divert:
114 115 return opener(name + ".a", mode.replace('a', 'w'))
115 116 # otherwise, divert to memory
116 117 return appender(opener(name, mode), buf)
117 118 return o
118 119
119 120 class changelog(revlog.revlog):
120 121 def __init__(self, opener):
121 122 revlog.revlog.__init__(self, opener, "00changelog.i")
122 123 if self._initempty:
123 124 # changelogs don't benefit from generaldelta
124 125 self.version &= ~revlog.REVLOGGENERALDELTA
125 126 self._generaldelta = False
126 127 self._realopener = opener
127 128 self._delayed = False
128 129 self._divert = False
129 130 self.filteredrevs = frozenset()
130 131
131 132 def tip(self):
132 133 """filtered version of revlog.tip"""
133 134 for i in xrange(len(self) -1, -2, -1):
134 135 if i not in self.filteredrevs:
135 136 return self.node(i)
136 137
137 138 def __iter__(self):
138 139 """filtered version of revlog.__iter__"""
139 140 if len(self.filteredrevs) == 0:
140 141 return revlog.revlog.__iter__(self)
141 142
142 143 def filterediter():
143 144 for i in xrange(len(self)):
144 145 if i not in self.filteredrevs:
145 146 yield i
146 147
147 148 return filterediter()
148 149
149 150 def revs(self, start=0, stop=None):
150 151 """filtered version of revlog.revs"""
151 152 for i in super(changelog, self).revs(start, stop):
152 153 if i not in self.filteredrevs:
153 154 yield i
154 155
155 156 @util.propertycache
156 157 def nodemap(self):
157 158 # XXX need filtering too
158 159 self.rev(self.node(0))
159 160 return self._nodecache
160 161
161 162 def hasnode(self, node):
162 163 """filtered version of revlog.hasnode"""
163 164 try:
164 165 i = self.rev(node)
165 166 return i not in self.filteredrevs
166 167 except KeyError:
167 168 return False
168 169
169 170 def headrevs(self):
170 171 if self.filteredrevs:
171 172 # XXX we should fix and use the C version
172 173 return self._headrevs()
173 174 return super(changelog, self).headrevs()
174 175
175 176 def strip(self, *args, **kwargs):
176 177 # XXX make something better than assert
177 178 # We can't expect proper strip behavior if we are filtered.
178 179 assert not self.filteredrevs
179 180 super(changelog, self).strip(*args, **kwargs)
180 181
181 182 def rev(self, node):
182 183 """filtered version of revlog.rev"""
183 184 r = super(changelog, self).rev(node)
184 185 if r in self.filteredrevs:
185 186 raise error.LookupError(node, self.indexfile, _('no node'))
186 187 return r
187 188
188 189 def node(self, rev):
189 190 """filtered version of revlog.node"""
190 191 if rev in self.filteredrevs:
191 192 raise IndexError(rev)
192 193 return super(changelog, self).node(rev)
193 194
194 195 def linkrev(self, rev):
195 196 """filtered version of revlog.linkrev"""
196 197 if rev in self.filteredrevs:
197 198 raise IndexError(rev)
198 199 return super(changelog, self).linkrev(rev)
199 200
200 201 def parentrevs(self, rev):
201 202 """filtered version of revlog.parentrevs"""
202 203 if rev in self.filteredrevs:
203 204 raise IndexError(rev)
204 205 return super(changelog, self).parentrevs(rev)
205 206
206 207 def flags(self, rev):
207 208 """filtered version of revlog.flags"""
208 209 if rev in self.filteredrevs:
209 210 raise IndexError(rev)
210 211 return super(changelog, self).flags(rev)
211 212
212 213 def delayupdate(self):
213 214 "delay visibility of index updates to other readers"
214 215 self._delayed = True
215 216 self._divert = (len(self) == 0)
216 217 self._delaybuf = []
217 218 self.opener = delayopener(self._realopener, self.indexfile,
218 219 self._divert, self._delaybuf)
219 220
220 221 def finalize(self, tr):
221 222 "finalize index updates"
222 223 self._delayed = False
223 224 self.opener = self._realopener
224 225 # move redirected index data back into place
225 226 if self._divert:
226 227 nfile = self.opener(self.indexfile + ".a")
227 228 n = nfile.name
228 229 nfile.close()
229 230 util.rename(n, n[:-2])
230 231 elif self._delaybuf:
231 232 fp = self.opener(self.indexfile, 'a')
232 233 fp.write("".join(self._delaybuf))
233 234 fp.close()
234 235 self._delaybuf = []
235 236 # split when we're done
236 237 self.checkinlinesize(tr)
237 238
238 239 def readpending(self, file):
239 240 r = revlog.revlog(self.opener, file)
240 241 self.index = r.index
241 242 self.nodemap = r.nodemap
242 243 self._nodecache = r._nodecache
243 244 self._chunkcache = r._chunkcache
244 245
245 246 def writepending(self):
246 247 "create a file containing the unfinalized state for pretxnchangegroup"
247 248 if self._delaybuf:
248 249 # make a temporary copy of the index
249 250 fp1 = self._realopener(self.indexfile)
250 251 fp2 = self._realopener(self.indexfile + ".a", "w")
251 252 fp2.write(fp1.read())
252 253 # add pending data
253 254 fp2.write("".join(self._delaybuf))
254 255 fp2.close()
255 256 # switch modes so finalize can simply rename
256 257 self._delaybuf = []
257 258 self._divert = True
258 259
259 260 if self._divert:
260 261 return True
261 262
262 263 return False
263 264
264 265 def checkinlinesize(self, tr, fp=None):
265 266 if not self._delayed:
266 267 revlog.revlog.checkinlinesize(self, tr, fp)
267 268
268 269 def read(self, node):
269 270 """
270 271 format used:
271 272 nodeid\n : manifest node in ascii
272 273 user\n : user, no \n or \r allowed
273 274 time tz extra\n : date (time is int or float, timezone is int)
274 275 : extra is metadata, encoded and separated by '\0'
275 276 : older versions ignore it
276 277 files\n\n : files modified by the cset, no \n or \r allowed
277 278 (.*) : comment (free text, ideally utf-8)
278 279
279 280 changelog v0 doesn't use extra
280 281 """
281 282 text = self.revision(node)
282 283 if not text:
283 284 return (nullid, "", (0, 0), [], "", _defaultextra)
284 285 last = text.index("\n\n")
285 286 desc = encoding.tolocal(text[last + 2:])
286 287 l = text[:last].split('\n')
287 288 manifest = bin(l[0])
288 289 user = encoding.tolocal(l[1])
289 290
290 291 tdata = l[2].split(' ', 2)
291 292 if len(tdata) != 3:
292 293 time = float(tdata[0])
293 294 try:
294 295 # various tools did silly things with the time zone field.
295 296 timezone = int(tdata[1])
296 297 except ValueError:
297 298 timezone = 0
298 299 extra = _defaultextra
299 300 else:
300 301 time, timezone = float(tdata[0]), int(tdata[1])
301 302 extra = decodeextra(tdata[2])
302 303
303 304 files = l[3:]
304 305 return (manifest, user, (time, timezone), files, desc, extra)
305 306
306 307 def add(self, manifest, files, desc, transaction, p1, p2,
307 308 user, date=None, extra=None):
308 309 # Convert to UTF-8 encoded bytestrings as the very first
309 310 # thing: calling any method on a localstr object will turn it
310 311 # into a str object and the cached UTF-8 string is thus lost.
311 312 user, desc = encoding.fromlocal(user), encoding.fromlocal(desc)
312 313
313 314 user = user.strip()
314 315 # An empty username or a username with a "\n" will make the
315 316 # revision text contain two "\n\n" sequences -> corrupt
316 317 # repository since read cannot unpack the revision.
317 318 if not user:
318 319 raise error.RevlogError(_("empty username"))
319 320 if "\n" in user:
320 321 raise error.RevlogError(_("username %s contains a newline")
321 322 % repr(user))
322 323
323 324 desc = stripdesc(desc)
324 325
325 326 if date:
326 327 parseddate = "%d %d" % util.parsedate(date)
327 328 else:
328 329 parseddate = "%d %d" % util.makedate()
329 330 if extra:
330 331 branch = extra.get("branch")
331 332 if branch in ("default", ""):
332 333 del extra["branch"]
333 334 elif branch in (".", "null", "tip"):
334 335 raise error.RevlogError(_('the name \'%s\' is reserved')
335 336 % branch)
336 337 if extra:
337 338 extra = encodeextra(extra)
338 339 parseddate = "%s %s" % (parseddate, extra)
339 340 l = [hex(manifest), user, parseddate] + sorted(files) + ["", desc]
340 341 text = "\n".join(l)
341 342 return self.addrevision(text, transaction, len(self), p1, p2)
342 343
343 344 def branch(self, rev):
344 345 """return the branch of a revision
345 346
346 347 This function exists because creating a changectx object
347 348 just to access this is costly."""
348 349 return encoding.tolocal(self.read(rev)[5].get("branch"))
General Comments 0
You need to be logged in to leave comments. Login now