Mercurial > public > mercurial-scm > hg-stable
annotate mercurial/hgweb.py @ 173:8da1df932c16
hgweb: make navigation of changesets a bit nicer
author | mpm@selenic.com |
---|---|
date | Thu, 26 May 2005 22:47:43 -0800 |
parents | e9b1147db448 |
children | f25944662097 |
rev | line source |
---|---|
131 | 1 #!/usr/bin/env python |
2 # | |
132 | 3 # hgweb.py - 0.2 - 21 May 2005 - (c) 2005 Jake Edge <jake@edge2.net> |
131 | 4 # - web interface to a mercurial repository |
5 # | |
6 # This software may be used and distributed according to the terms | |
7 # of the GNU General Public License, incorporated herein by reference. | |
8 | |
9 # useful for debugging | |
10 import cgitb | |
11 cgitb.enable() | |
12 | |
13 import os, cgi, time, re, difflib, sys, zlib | |
138 | 14 from mercurial.hg import * |
15 | |
157
2653740d8118
Install the templates where they can be found by hgweb.py
mpm@selenic.com
parents:
156
diff
changeset
|
16 def templatepath(): |
2653740d8118
Install the templates where they can be found by hgweb.py
mpm@selenic.com
parents:
156
diff
changeset
|
17 for f in "templates/map", "../templates/map": |
2653740d8118
Install the templates where they can be found by hgweb.py
mpm@selenic.com
parents:
156
diff
changeset
|
18 p = os.path.join(os.path.dirname(__file__), f) |
2653740d8118
Install the templates where they can be found by hgweb.py
mpm@selenic.com
parents:
156
diff
changeset
|
19 if os.path.isfile(p): return p |
2653740d8118
Install the templates where they can be found by hgweb.py
mpm@selenic.com
parents:
156
diff
changeset
|
20 |
138 | 21 def age(t): |
22 def plural(t, c): | |
23 if c == 1: return t | |
24 return t + "s" | |
25 def fmt(t, c): | |
26 return "%d %s" % (c, plural(t, c)) | |
27 | |
28 now = time.time() | |
29 delta = max(1, int(now - t)) | |
30 | |
31 scales = [["second", 1], | |
32 ["minute", 60], | |
33 ["hour", 3600], | |
34 ["day", 3600 * 24], | |
35 ["week", 3600 * 24 * 7], | |
36 ["month", 3600 * 24 * 30], | |
37 ["year", 3600 * 24 * 365]] | |
38 | |
39 scales.reverse() | |
40 | |
41 for t, s in scales: | |
42 n = delta / s | |
43 if n >= 1: return fmt(t, n) | |
131 | 44 |
45 def nl2br(text): | |
138 | 46 return text.replace('\n', '<br/>') |
131 | 47 |
48 def obfuscate(text): | |
138 | 49 return ''.join([ '&#%d' % ord(c) for c in text ]) |
50 | |
51 def up(p): | |
52 if p[0] != "/": p = "/" + p | |
53 if p[-1] == "/": p = p[:-1] | |
54 up = os.path.dirname(p) | |
55 if up == "/": | |
56 return "/" | |
57 return up + "/" | |
131 | 58 |
59 def httphdr(type): | |
60 print 'Content-type: %s\n' % type | |
61 | |
135 | 62 def write(*things): |
63 for thing in things: | |
64 if hasattr(thing, "__iter__"): | |
65 for part in thing: | |
66 write(part) | |
67 else: | |
68 sys.stdout.write(str(thing)) | |
69 | |
138 | 70 def template(tmpl, **map): |
71 while tmpl: | |
72 m = re.search(r"#([a-zA-Z0-9]+)#", tmpl) | |
73 if m: | |
74 yield tmpl[:m.start(0)] | |
75 v = map.get(m.group(1), "") | |
76 yield callable(v) and v() or v | |
77 tmpl = tmpl[m.end(0):] | |
78 else: | |
79 yield tmpl | |
80 return | |
81 | |
82 class templater: | |
83 def __init__(self, mapfile): | |
84 self.cache = {} | |
85 self.map = {} | |
86 self.base = os.path.dirname(mapfile) | |
87 | |
88 for l in file(mapfile): | |
89 m = re.match(r'(\S+)\s*=\s*"(.*)"$', l) | |
133
fb84d3e71042
added template support for some hgweb output, also, template files for
jake@edge2.net
parents:
132
diff
changeset
|
90 if m: |
138 | 91 self.cache[m.group(1)] = m.group(2) |
92 else: | |
93 m = re.match(r'(\S+)\s*=\s*(\S+)', l) | |
94 if m: | |
95 self.map[m.group(1)] = os.path.join(self.base, m.group(2)) | |
133
fb84d3e71042
added template support for some hgweb output, also, template files for
jake@edge2.net
parents:
132
diff
changeset
|
96 else: |
138 | 97 raise "unknown map entry '%s'" % l |
133
fb84d3e71042
added template support for some hgweb output, also, template files for
jake@edge2.net
parents:
132
diff
changeset
|
98 |
138 | 99 def __call__(self, t, **map): |
100 try: | |
101 tmpl = self.cache[t] | |
102 except KeyError: | |
103 tmpl = self.cache[t] = file(self.map[t]).read() | |
104 return template(tmpl, **map) | |
105 | |
106 class hgweb: | |
107 maxchanges = 20 | |
108 maxfiles = 10 | |
133
fb84d3e71042
added template support for some hgweb output, also, template files for
jake@edge2.net
parents:
132
diff
changeset
|
109 |
157
2653740d8118
Install the templates where they can be found by hgweb.py
mpm@selenic.com
parents:
156
diff
changeset
|
110 def __init__(self, path, name, templatemap = ""): |
2653740d8118
Install the templates where they can be found by hgweb.py
mpm@selenic.com
parents:
156
diff
changeset
|
111 templatemap = templatemap or templatepath() |
2653740d8118
Install the templates where they can be found by hgweb.py
mpm@selenic.com
parents:
156
diff
changeset
|
112 |
138 | 113 self.reponame = name |
114 self.repo = repository(ui(), path) | |
115 self.t = templater(templatemap) | |
131 | 116 |
138 | 117 def date(self, cs): |
118 return time.asctime(time.gmtime(float(cs[2].split(' ')[0]))) | |
119 | |
120 def listfiles(self, files, mf): | |
121 for f in files[:self.maxfiles]: | |
122 yield self.t("filenodelink", node = hex(mf[f]), file = f) | |
123 if len(files) > self.maxfiles: | |
124 yield self.t("fileellipses") | |
125 | |
126 def listfilediffs(self, files, changeset): | |
127 for f in files[:self.maxfiles]: | |
128 yield self.t("filedifflink", node = hex(changeset), file = f) | |
129 if len(files) > self.maxfiles: | |
130 yield self.t("fileellipses") | |
131 | |
156 | 132 def parent(self, t1, node=nullid, rev=-1, **args): |
142 | 133 if node != hex(nullid): |
156 | 134 yield self.t(t1, node = node, rev = rev, **args) |
142 | 135 |
138 | 136 def diff(self, node1, node2, files): |
137 def filterfiles(list, files): | |
138 l = [ x for x in list if x in files ] | |
139 | |
140 for f in files: | |
141 if f[-1] != os.sep: f += os.sep | |
142 l += [ x for x in list if x.startswith(f) ] | |
143 return l | |
131 | 144 |
172
e9b1147db448
hgweb: alternating colors for multifile diffs
mpm@selenic.com
parents:
168
diff
changeset
|
145 parity = [0] |
e9b1147db448
hgweb: alternating colors for multifile diffs
mpm@selenic.com
parents:
168
diff
changeset
|
146 def diffblock(diff, f, fn): |
e9b1147db448
hgweb: alternating colors for multifile diffs
mpm@selenic.com
parents:
168
diff
changeset
|
147 yield self.t("diffblock", |
e9b1147db448
hgweb: alternating colors for multifile diffs
mpm@selenic.com
parents:
168
diff
changeset
|
148 lines = prettyprintlines(diff), |
e9b1147db448
hgweb: alternating colors for multifile diffs
mpm@selenic.com
parents:
168
diff
changeset
|
149 parity = parity[0], |
e9b1147db448
hgweb: alternating colors for multifile diffs
mpm@selenic.com
parents:
168
diff
changeset
|
150 file = f, |
e9b1147db448
hgweb: alternating colors for multifile diffs
mpm@selenic.com
parents:
168
diff
changeset
|
151 filenode = hex(fn)) |
e9b1147db448
hgweb: alternating colors for multifile diffs
mpm@selenic.com
parents:
168
diff
changeset
|
152 parity[0] = 1 - parity[0] |
e9b1147db448
hgweb: alternating colors for multifile diffs
mpm@selenic.com
parents:
168
diff
changeset
|
153 |
e9b1147db448
hgweb: alternating colors for multifile diffs
mpm@selenic.com
parents:
168
diff
changeset
|
154 def prettyprintlines(diff): |
138 | 155 for l in diff.splitlines(1): |
156 line = cgi.escape(l) | |
157 if line.startswith('+'): | |
158 yield self.t("difflineplus", line = line) | |
159 elif line.startswith('-'): | |
160 yield self.t("difflineminus", line = line) | |
161 elif line.startswith('@'): | |
162 yield self.t("difflineat", line = line) | |
163 else: | |
164 yield self.t("diffline", line = line) | |
131 | 165 |
138 | 166 r = self.repo |
167 cl = r.changelog | |
168 mf = r.manifest | |
169 change1 = cl.read(node1) | |
170 change2 = cl.read(node2) | |
171 mmap1 = mf.read(change1[0]) | |
172 mmap2 = mf.read(change2[0]) | |
173 date1 = self.date(change1) | |
174 date2 = self.date(change2) | |
131 | 175 |
138 | 176 c, a, d = r.diffrevs(node1, node2) |
177 c, a, d = map(lambda x: filterfiles(x, files), (c, a, d)) | |
131 | 178 |
138 | 179 for f in c: |
180 to = r.file(f).read(mmap1[f]) | |
181 tn = r.file(f).read(mmap2[f]) | |
172
e9b1147db448
hgweb: alternating colors for multifile diffs
mpm@selenic.com
parents:
168
diff
changeset
|
182 yield diffblock(mdiff.unidiff(to, date1, tn, date2, f), f, tn) |
138 | 183 for f in a: |
184 to = "" | |
185 tn = r.file(f).read(mmap2[f]) | |
172
e9b1147db448
hgweb: alternating colors for multifile diffs
mpm@selenic.com
parents:
168
diff
changeset
|
186 yield diffblock(mdiff.unidiff(to, date1, tn, date2, f), f, tn) |
138 | 187 for f in d: |
188 to = r.file(f).read(mmap1[f]) | |
189 tn = "" | |
172
e9b1147db448
hgweb: alternating colors for multifile diffs
mpm@selenic.com
parents:
168
diff
changeset
|
190 yield diffblock(mdiff.unidiff(to, date1, tn, date2, f), f, tn) |
131 | 191 |
142 | 192 def header(self): |
193 yield self.t("header", repo = self.reponame) | |
194 | |
195 def footer(self): | |
196 yield self.t("footer", repo = self.reponame) | |
197 | |
138 | 198 def changelog(self, pos=None): |
199 def changenav(): | |
200 def seq(factor = 1): | |
201 yield 1 * factor | |
173
8da1df932c16
hgweb: make navigation of changesets a bit nicer
mpm@selenic.com
parents:
172
diff
changeset
|
202 yield 3 * factor |
8da1df932c16
hgweb: make navigation of changesets a bit nicer
mpm@selenic.com
parents:
172
diff
changeset
|
203 #yield 5 * factor |
138 | 204 for f in seq(factor * 10): |
205 yield f | |
131 | 206 |
173
8da1df932c16
hgweb: make navigation of changesets a bit nicer
mpm@selenic.com
parents:
172
diff
changeset
|
207 l = [] |
8da1df932c16
hgweb: make navigation of changesets a bit nicer
mpm@selenic.com
parents:
172
diff
changeset
|
208 for f in seq(): |
8da1df932c16
hgweb: make navigation of changesets a bit nicer
mpm@selenic.com
parents:
172
diff
changeset
|
209 if f < self.maxchanges / 2: continue |
8da1df932c16
hgweb: make navigation of changesets a bit nicer
mpm@selenic.com
parents:
172
diff
changeset
|
210 if f > count: break |
8da1df932c16
hgweb: make navigation of changesets a bit nicer
mpm@selenic.com
parents:
172
diff
changeset
|
211 r = "%d" % f |
8da1df932c16
hgweb: make navigation of changesets a bit nicer
mpm@selenic.com
parents:
172
diff
changeset
|
212 if pos + f < count - (f/2): l.append(("+" + r, pos + f)) |
8da1df932c16
hgweb: make navigation of changesets a bit nicer
mpm@selenic.com
parents:
172
diff
changeset
|
213 if pos - f >= 0 + (f/2): l.insert(0, ("-" + r, pos - f)) |
8da1df932c16
hgweb: make navigation of changesets a bit nicer
mpm@selenic.com
parents:
172
diff
changeset
|
214 |
8da1df932c16
hgweb: make navigation of changesets a bit nicer
mpm@selenic.com
parents:
172
diff
changeset
|
215 yield self.t("naventry", rev = 0, label="(0)") |
138 | 216 |
173
8da1df932c16
hgweb: make navigation of changesets a bit nicer
mpm@selenic.com
parents:
172
diff
changeset
|
217 for label, rev in l: |
8da1df932c16
hgweb: make navigation of changesets a bit nicer
mpm@selenic.com
parents:
172
diff
changeset
|
218 yield self.t("naventry", label = label, rev = rev) |
8da1df932c16
hgweb: make navigation of changesets a bit nicer
mpm@selenic.com
parents:
172
diff
changeset
|
219 |
8da1df932c16
hgweb: make navigation of changesets a bit nicer
mpm@selenic.com
parents:
172
diff
changeset
|
220 yield self.t("naventry", rev = count - 1, label="tip") |
131 | 221 |
138 | 222 def changelist(): |
142 | 223 parity = (start - end) & 1 |
138 | 224 cl = self.repo.changelog |
225 l = [] # build a list in forward order for efficiency | |
226 for i in range(start, end + 1): | |
227 n = cl.node(i) | |
228 changes = cl.read(n) | |
229 hn = hex(n) | |
230 p1, p2 = cl.parents(n) | |
231 t = float(changes[2].split(' ')[0]) | |
131 | 232 |
138 | 233 l.insert(0, self.t( |
234 'changelogentry', | |
142 | 235 parity = parity, |
138 | 236 author = obfuscate(changes[1]), |
237 shortdesc = cgi.escape(changes[4].splitlines()[0]), | |
238 age = age(t), | |
142 | 239 parent1 = self.parent("changelogparent", |
240 hex(p1), cl.rev(p1)), | |
241 parent2 = self.parent("changelogparent", | |
242 hex(p2), cl.rev(p2)), | |
138 | 243 p1 = hex(p1), p2 = hex(p2), |
244 p1rev = cl.rev(p1), p2rev = cl.rev(p2), | |
245 manifest = hex(changes[0]), | |
246 desc = nl2br(cgi.escape(changes[4])), | |
247 date = time.asctime(time.gmtime(t)), | |
248 files = self.listfilediffs(changes[3], n), | |
249 rev = i, | |
250 node = hn)) | |
142 | 251 parity = 1 - parity |
138 | 252 |
253 yield l | |
131 | 254 |
168 | 255 cl = self.repo.changelog |
256 mf = cl.read(cl.tip())[0] | |
257 count = cl.count() | |
138 | 258 pos = pos or count - 1 |
259 end = min(pos, count - 1) | |
260 start = max(0, pos - self.maxchanges) | |
261 end = min(count - 1, start + self.maxchanges) | |
262 | |
142 | 263 yield self.t('changelog', |
264 header = self.header(), | |
265 footer = self.footer(), | |
266 repo = self.reponame, | |
267 changenav = changenav, | |
168 | 268 manifest = hex(mf), |
142 | 269 rev = pos, changesets = count, entries = changelist) |
131 | 270 |
138 | 271 def changeset(self, nodeid): |
272 n = bin(nodeid) | |
273 cl = self.repo.changelog | |
274 changes = cl.read(n) | |
275 p1, p2 = cl.parents(n) | |
276 p1rev, p2rev = cl.rev(p1), cl.rev(p2) | |
277 t = float(changes[2].split(' ')[0]) | |
278 | |
133
fb84d3e71042
added template support for some hgweb output, also, template files for
jake@edge2.net
parents:
132
diff
changeset
|
279 files = [] |
138 | 280 mf = self.repo.manifest.read(changes[0]) |
131 | 281 for f in changes[3]: |
138 | 282 files.append(self.t("filenodelink", |
283 filenode = hex(mf[f]), file = f)) | |
284 | |
285 def diff(): | |
286 yield self.diff(p1, n, changes[3]) | |
131 | 287 |
138 | 288 yield self.t('changeset', |
142 | 289 header = self.header(), |
290 footer = self.footer(), | |
291 repo = self.reponame, | |
138 | 292 diff = diff, |
293 rev = cl.rev(n), | |
294 node = nodeid, | |
295 shortdesc = cgi.escape(changes[4].splitlines()[0]), | |
142 | 296 parent1 = self.parent("changesetparent", |
297 hex(p1), cl.rev(p1)), | |
298 parent2 = self.parent("changesetparent", | |
299 hex(p2), cl.rev(p2)), | |
138 | 300 p1 = hex(p1), p2 = hex(p2), |
301 p1rev = cl.rev(p1), p2rev = cl.rev(p2), | |
302 manifest = hex(changes[0]), | |
303 author = obfuscate(changes[1]), | |
304 desc = nl2br(cgi.escape(changes[4])), | |
305 date = time.asctime(time.gmtime(t)), | |
306 files = files) | |
131 | 307 |
138 | 308 def filelog(self, f, filenode): |
309 cl = self.repo.changelog | |
310 fl = self.repo.file(f) | |
311 count = fl.count() | |
312 | |
313 def entries(): | |
314 l = [] | |
142 | 315 parity = (count - 1) & 1 |
316 | |
138 | 317 for i in range(count): |
318 | |
319 n = fl.node(i) | |
320 lr = fl.linkrev(n) | |
321 cn = cl.node(lr) | |
322 cs = cl.read(cl.node(lr)) | |
323 p1, p2 = fl.parents(n) | |
324 t = float(cs[2].split(' ')[0]) | |
133
fb84d3e71042
added template support for some hgweb output, also, template files for
jake@edge2.net
parents:
132
diff
changeset
|
325 |
138 | 326 l.insert(0, self.t("filelogentry", |
142 | 327 parity = parity, |
138 | 328 filenode = hex(n), |
329 filerev = i, | |
330 file = f, | |
331 node = hex(cn), | |
332 author = obfuscate(cs[1]), | |
333 age = age(t), | |
334 date = time.asctime(time.gmtime(t)), | |
335 shortdesc = cgi.escape(cs[4].splitlines()[0]), | |
336 p1 = hex(p1), p2 = hex(p2), | |
337 p1rev = fl.rev(p1), p2rev = fl.rev(p2))) | |
142 | 338 parity = 1 - parity |
138 | 339 |
340 yield l | |
341 | |
342 yield self.t("filelog", | |
142 | 343 header = self.header(), |
344 footer = self.footer(), | |
345 repo = self.reponame, | |
138 | 346 file = f, |
347 filenode = filenode, | |
348 entries = entries) | |
131 | 349 |
138 | 350 def filerevision(self, f, node): |
351 fl = self.repo.file(f) | |
352 n = bin(node) | |
353 text = cgi.escape(fl.read(n)) | |
354 changerev = fl.linkrev(n) | |
355 cl = self.repo.changelog | |
356 cn = cl.node(changerev) | |
357 cs = cl.read(cn) | |
358 p1, p2 = fl.parents(n) | |
359 t = float(cs[2].split(' ')[0]) | |
360 mfn = cs[0] | |
142 | 361 |
362 def lines(): | |
363 for l, t in enumerate(text.splitlines(1)): | |
364 yield self.t("fileline", | |
365 line = t, | |
366 linenumber = "% 6d" % (l + 1), | |
367 parity = l & 1) | |
138 | 368 |
369 yield self.t("filerevision", file = f, | |
142 | 370 header = self.header(), |
371 footer = self.footer(), | |
372 repo = self.reponame, | |
138 | 373 filenode = node, |
374 path = up(f), | |
142 | 375 text = lines(), |
138 | 376 rev = changerev, |
377 node = hex(cn), | |
378 manifest = hex(mfn), | |
379 author = obfuscate(cs[1]), | |
380 age = age(t), | |
381 date = time.asctime(time.gmtime(t)), | |
382 shortdesc = cgi.escape(cs[4].splitlines()[0]), | |
142 | 383 parent1 = self.parent("filerevparent", |
156 | 384 hex(p1), fl.rev(p1), file=f), |
142 | 385 parent2 = self.parent("filerevparent", |
156 | 386 hex(p2), fl.rev(p2), file=f), |
138 | 387 p1 = hex(p1), p2 = hex(p2), |
388 p1rev = fl.rev(p1), p2rev = fl.rev(p2)) | |
389 | |
131 | 390 |
138 | 391 def fileannotate(self, f, node): |
392 bcache = {} | |
393 ncache = {} | |
394 fl = self.repo.file(f) | |
395 n = bin(node) | |
396 changerev = fl.linkrev(n) | |
397 | |
398 cl = self.repo.changelog | |
399 cn = cl.node(changerev) | |
400 cs = cl.read(cn) | |
401 p1, p2 = fl.parents(n) | |
402 t = float(cs[2].split(' ')[0]) | |
403 mfn = cs[0] | |
131 | 404 |
138 | 405 def annotate(): |
142 | 406 parity = 1 |
407 last = None | |
138 | 408 for r, l in fl.annotate(n): |
409 try: | |
410 cnode = ncache[r] | |
411 except KeyError: | |
412 cnode = ncache[r] = self.repo.changelog.node(r) | |
413 | |
414 try: | |
415 name = bcache[r] | |
416 except KeyError: | |
417 cl = self.repo.changelog.read(cnode) | |
418 name = cl[1] | |
419 f = name.find('@') | |
420 if f >= 0: | |
421 name = name[:f] | |
422 bcache[r] = name | |
131 | 423 |
142 | 424 if last != cnode: |
425 parity = 1 - parity | |
426 last = cnode | |
427 | |
138 | 428 yield self.t("annotateline", |
142 | 429 parity = parity, |
138 | 430 node = hex(cnode), |
431 rev = r, | |
432 author = name, | |
433 file = f, | |
434 line = cgi.escape(l)) | |
435 | |
436 yield self.t("fileannotate", | |
142 | 437 header = self.header(), |
438 footer = self.footer(), | |
439 repo = self.reponame, | |
138 | 440 file = f, |
441 filenode = node, | |
442 annotate = annotate, | |
443 path = up(f), | |
444 rev = changerev, | |
445 node = hex(cn), | |
446 manifest = hex(mfn), | |
447 author = obfuscate(cs[1]), | |
448 age = age(t), | |
449 date = time.asctime(time.gmtime(t)), | |
450 shortdesc = cgi.escape(cs[4].splitlines()[0]), | |
156 | 451 parent1 = self.parent("fileannotateparent", |
452 hex(p1), fl.rev(p1), file=f), | |
453 parent2 = self.parent("fileannotateparent", | |
454 hex(p2), fl.rev(p2), file=f), | |
138 | 455 p1 = hex(p1), p2 = hex(p2), |
456 p1rev = fl.rev(p1), p2rev = fl.rev(p2)) | |
136 | 457 |
138 | 458 def manifest(self, mnode, path): |
459 mf = self.repo.manifest.read(bin(mnode)) | |
460 rev = self.repo.manifest.rev(bin(mnode)) | |
461 node = self.repo.changelog.node(rev) | |
462 | |
463 files = {} | |
142 | 464 |
138 | 465 p = path[1:] |
466 l = len(p) | |
131 | 467 |
138 | 468 for f,n in mf.items(): |
469 if f[:l] != p: | |
470 continue | |
471 remain = f[l:] | |
472 if "/" in remain: | |
473 short = remain[:remain.find("/") + 1] # bleah | |
142 | 474 files[short] = (f, None) |
138 | 475 else: |
476 short = os.path.basename(remain) | |
477 files[short] = (f, n) | |
131 | 478 |
138 | 479 def filelist(): |
142 | 480 parity = 0 |
138 | 481 fl = files.keys() |
482 fl.sort() | |
483 for f in fl: | |
484 full, fnode = files[f] | |
142 | 485 if fnode: |
486 yield self.t("manifestfileentry", | |
487 file = full, | |
488 manifest = mnode, | |
489 filenode = hex(fnode), | |
490 parity = parity, | |
491 basename = f) | |
492 else: | |
493 yield self.t("manifestdirentry", | |
494 parity = parity, | |
495 path = os.path.join(path, f), | |
496 manifest = mnode, basename = f[:-1]) | |
497 parity = 1 - parity | |
138 | 498 |
499 yield self.t("manifest", | |
142 | 500 header = self.header(), |
501 footer = self.footer(), | |
502 repo = self.reponame, | |
138 | 503 manifest = mnode, |
504 rev = rev, | |
505 node = hex(node), | |
506 path = path, | |
507 up = up(path), | |
142 | 508 entries = filelist) |
131 | 509 |
168 | 510 def tags(self): |
511 cl = self.repo.changelog | |
512 mf = cl.read(cl.tip())[0] | |
513 | |
514 self.repo.lookup(0) # prime the cache | |
515 i = self.repo.tags.items() | |
516 i.sort() | |
517 | |
518 def entries(): | |
519 parity = 0 | |
520 for k,n in i: | |
521 yield self.t("tagentry", | |
522 parity = parity, | |
523 tag = k, | |
524 node = hex(n)) | |
525 parity = 1 - parity | |
526 | |
527 yield self.t("tags", | |
528 header = self.header(), | |
529 footer = self.footer(), | |
530 repo = self.reponame, | |
531 manifest = hex(mf), | |
532 entries = entries) | |
533 | |
138 | 534 def filediff(self, file, changeset): |
535 n = bin(changeset) | |
536 cl = self.repo.changelog | |
537 p1 = cl.parents(n)[0] | |
538 cs = cl.read(n) | |
539 mf = self.repo.manifest.read(cs[0]) | |
540 | |
541 def diff(): | |
542 yield self.diff(p1, n, file) | |
131 | 543 |
138 | 544 yield self.t("filediff", |
142 | 545 header = self.header(), |
546 footer = self.footer(), | |
547 repo = self.reponame, | |
138 | 548 file = file, |
549 filenode = hex(mf[file]), | |
550 node = changeset, | |
551 rev = self.repo.changelog.rev(n), | |
552 p1 = hex(p1), | |
553 p1rev = self.repo.changelog.rev(p1), | |
554 diff = diff) | |
555 | |
556 # add tags to things | |
557 # tags -> list of changesets corresponding to tags | |
558 # find tag, changeset, file | |
131 | 559 |
132 | 560 def run(self): |
561 args = cgi.parse() | |
562 | |
138 | 563 if not args.has_key('cmd') or args['cmd'][0] == 'changelog': |
564 hi = self.repo.changelog.count() | |
153
e8a360cd5a9f
changed pos to rev for changelog cmd, changed & to ;
jake@edge2.net
parents:
142
diff
changeset
|
565 if args.has_key('rev'): |
165 | 566 hi = args['rev'][0] |
166
39624c47060f
hgweb: don't blow up on search for unknown keys
mpm@selenic.com
parents:
165
diff
changeset
|
567 try: |
39624c47060f
hgweb: don't blow up on search for unknown keys
mpm@selenic.com
parents:
165
diff
changeset
|
568 hi = self.repo.changelog.rev(self.repo.lookup(hi)) |
39624c47060f
hgweb: don't blow up on search for unknown keys
mpm@selenic.com
parents:
165
diff
changeset
|
569 except KeyError: |
39624c47060f
hgweb: don't blow up on search for unknown keys
mpm@selenic.com
parents:
165
diff
changeset
|
570 hi = self.repo.changelog.count() |
131 | 571 |
138 | 572 write(self.changelog(hi)) |
132 | 573 |
138 | 574 elif args['cmd'][0] == 'changeset': |
575 write(self.changeset(args['node'][0])) | |
576 | |
577 elif args['cmd'][0] == 'manifest': | |
578 write(self.manifest(args['manifest'][0], args['path'][0])) | |
579 | |
168 | 580 elif args['cmd'][0] == 'tags': |
581 write(self.tags()) | |
582 | |
138 | 583 elif args['cmd'][0] == 'filediff': |
584 write(self.filediff(args['file'][0], args['node'][0])) | |
131 | 585 |
132 | 586 elif args['cmd'][0] == 'file': |
138 | 587 write(self.filerevision(args['file'][0], args['filenode'][0])) |
131 | 588 |
138 | 589 elif args['cmd'][0] == 'annotate': |
590 write(self.fileannotate(args['file'][0], args['filenode'][0])) | |
131 | 591 |
138 | 592 elif args['cmd'][0] == 'filelog': |
593 write(self.filelog(args['file'][0], args['filenode'][0])) | |
136 | 594 |
132 | 595 elif args['cmd'][0] == 'branches': |
596 httphdr("text/plain") | |
597 nodes = [] | |
598 if args.has_key('nodes'): | |
138 | 599 nodes = map(bin, args['nodes'][0].split(" ")) |
600 for b in self.repo.branches(nodes): | |
601 sys.stdout.write(" ".join(map(hex, b)) + "\n") | |
131 | 602 |
132 | 603 elif args['cmd'][0] == 'between': |
604 httphdr("text/plain") | |
605 nodes = [] | |
606 if args.has_key('pairs'): | |
138 | 607 pairs = [ map(bin, p.split("-")) |
132 | 608 for p in args['pairs'][0].split(" ") ] |
138 | 609 for b in self.repo.between(pairs): |
610 sys.stdout.write(" ".join(map(hex, b)) + "\n") | |
132 | 611 |
612 elif args['cmd'][0] == 'changegroup': | |
613 httphdr("application/hg-changegroup") | |
614 nodes = [] | |
615 if args.has_key('roots'): | |
138 | 616 nodes = map(bin, args['roots'][0].split(" ")) |
131 | 617 |
132 | 618 z = zlib.compressobj() |
138 | 619 for chunk in self.repo.changegroup(nodes): |
132 | 620 sys.stdout.write(z.compress(chunk)) |
621 | |
622 sys.stdout.write(z.flush()) | |
131 | 623 |
132 | 624 else: |
138 | 625 write(self.t("error")) |
131 | 626 |
158
be7103467d2e
Add 'hg serve' command for stand-alone server
mpm@selenic.com
parents:
157
diff
changeset
|
627 def server(path, name, templates, address, port): |
be7103467d2e
Add 'hg serve' command for stand-alone server
mpm@selenic.com
parents:
157
diff
changeset
|
628 |
be7103467d2e
Add 'hg serve' command for stand-alone server
mpm@selenic.com
parents:
157
diff
changeset
|
629 import BaseHTTPServer |
be7103467d2e
Add 'hg serve' command for stand-alone server
mpm@selenic.com
parents:
157
diff
changeset
|
630 import sys, os |
be7103467d2e
Add 'hg serve' command for stand-alone server
mpm@selenic.com
parents:
157
diff
changeset
|
631 |
be7103467d2e
Add 'hg serve' command for stand-alone server
mpm@selenic.com
parents:
157
diff
changeset
|
632 class hgwebhandler(BaseHTTPServer.BaseHTTPRequestHandler): |
be7103467d2e
Add 'hg serve' command for stand-alone server
mpm@selenic.com
parents:
157
diff
changeset
|
633 def do_POST(self): |
be7103467d2e
Add 'hg serve' command for stand-alone server
mpm@selenic.com
parents:
157
diff
changeset
|
634 self.do_hgweb() |
be7103467d2e
Add 'hg serve' command for stand-alone server
mpm@selenic.com
parents:
157
diff
changeset
|
635 |
be7103467d2e
Add 'hg serve' command for stand-alone server
mpm@selenic.com
parents:
157
diff
changeset
|
636 def do_GET(self): |
be7103467d2e
Add 'hg serve' command for stand-alone server
mpm@selenic.com
parents:
157
diff
changeset
|
637 self.do_hgweb() |
be7103467d2e
Add 'hg serve' command for stand-alone server
mpm@selenic.com
parents:
157
diff
changeset
|
638 |
be7103467d2e
Add 'hg serve' command for stand-alone server
mpm@selenic.com
parents:
157
diff
changeset
|
639 def do_hgweb(self): |
be7103467d2e
Add 'hg serve' command for stand-alone server
mpm@selenic.com
parents:
157
diff
changeset
|
640 query = "" |
be7103467d2e
Add 'hg serve' command for stand-alone server
mpm@selenic.com
parents:
157
diff
changeset
|
641 p = self.path.find("?") |
be7103467d2e
Add 'hg serve' command for stand-alone server
mpm@selenic.com
parents:
157
diff
changeset
|
642 if p: |
be7103467d2e
Add 'hg serve' command for stand-alone server
mpm@selenic.com
parents:
157
diff
changeset
|
643 query = self.path[p + 1:] |
be7103467d2e
Add 'hg serve' command for stand-alone server
mpm@selenic.com
parents:
157
diff
changeset
|
644 query = query.replace('+', ' ') |
be7103467d2e
Add 'hg serve' command for stand-alone server
mpm@selenic.com
parents:
157
diff
changeset
|
645 |
be7103467d2e
Add 'hg serve' command for stand-alone server
mpm@selenic.com
parents:
157
diff
changeset
|
646 env = {} |
be7103467d2e
Add 'hg serve' command for stand-alone server
mpm@selenic.com
parents:
157
diff
changeset
|
647 env['GATEWAY_INTERFACE'] = 'CGI/1.1' |
be7103467d2e
Add 'hg serve' command for stand-alone server
mpm@selenic.com
parents:
157
diff
changeset
|
648 env['REQUEST_METHOD'] = self.command |
be7103467d2e
Add 'hg serve' command for stand-alone server
mpm@selenic.com
parents:
157
diff
changeset
|
649 if query: |
be7103467d2e
Add 'hg serve' command for stand-alone server
mpm@selenic.com
parents:
157
diff
changeset
|
650 env['QUERY_STRING'] = query |
be7103467d2e
Add 'hg serve' command for stand-alone server
mpm@selenic.com
parents:
157
diff
changeset
|
651 host = self.address_string() |
be7103467d2e
Add 'hg serve' command for stand-alone server
mpm@selenic.com
parents:
157
diff
changeset
|
652 if host != self.client_address[0]: |
be7103467d2e
Add 'hg serve' command for stand-alone server
mpm@selenic.com
parents:
157
diff
changeset
|
653 env['REMOTE_HOST'] = host |
be7103467d2e
Add 'hg serve' command for stand-alone server
mpm@selenic.com
parents:
157
diff
changeset
|
654 env['REMOTE_ADDR'] = self.client_address[0] |
be7103467d2e
Add 'hg serve' command for stand-alone server
mpm@selenic.com
parents:
157
diff
changeset
|
655 |
be7103467d2e
Add 'hg serve' command for stand-alone server
mpm@selenic.com
parents:
157
diff
changeset
|
656 if self.headers.typeheader is None: |
be7103467d2e
Add 'hg serve' command for stand-alone server
mpm@selenic.com
parents:
157
diff
changeset
|
657 env['CONTENT_TYPE'] = self.headers.type |
be7103467d2e
Add 'hg serve' command for stand-alone server
mpm@selenic.com
parents:
157
diff
changeset
|
658 else: |
be7103467d2e
Add 'hg serve' command for stand-alone server
mpm@selenic.com
parents:
157
diff
changeset
|
659 env['CONTENT_TYPE'] = self.headers.typeheader |
be7103467d2e
Add 'hg serve' command for stand-alone server
mpm@selenic.com
parents:
157
diff
changeset
|
660 length = self.headers.getheader('content-length') |
be7103467d2e
Add 'hg serve' command for stand-alone server
mpm@selenic.com
parents:
157
diff
changeset
|
661 if length: |
be7103467d2e
Add 'hg serve' command for stand-alone server
mpm@selenic.com
parents:
157
diff
changeset
|
662 env['CONTENT_LENGTH'] = length |
be7103467d2e
Add 'hg serve' command for stand-alone server
mpm@selenic.com
parents:
157
diff
changeset
|
663 accept = [] |
be7103467d2e
Add 'hg serve' command for stand-alone server
mpm@selenic.com
parents:
157
diff
changeset
|
664 for line in self.headers.getallmatchingheaders('accept'): |
be7103467d2e
Add 'hg serve' command for stand-alone server
mpm@selenic.com
parents:
157
diff
changeset
|
665 if line[:1] in "\t\n\r ": |
be7103467d2e
Add 'hg serve' command for stand-alone server
mpm@selenic.com
parents:
157
diff
changeset
|
666 accept.append(line.strip()) |
be7103467d2e
Add 'hg serve' command for stand-alone server
mpm@selenic.com
parents:
157
diff
changeset
|
667 else: |
be7103467d2e
Add 'hg serve' command for stand-alone server
mpm@selenic.com
parents:
157
diff
changeset
|
668 accept = accept + line[7:].split(',') |
be7103467d2e
Add 'hg serve' command for stand-alone server
mpm@selenic.com
parents:
157
diff
changeset
|
669 env['HTTP_ACCEPT'] = ','.join(accept) |
be7103467d2e
Add 'hg serve' command for stand-alone server
mpm@selenic.com
parents:
157
diff
changeset
|
670 |
be7103467d2e
Add 'hg serve' command for stand-alone server
mpm@selenic.com
parents:
157
diff
changeset
|
671 os.environ.update(env) |
be7103467d2e
Add 'hg serve' command for stand-alone server
mpm@selenic.com
parents:
157
diff
changeset
|
672 |
be7103467d2e
Add 'hg serve' command for stand-alone server
mpm@selenic.com
parents:
157
diff
changeset
|
673 save = sys.argv, sys.stdin, sys.stdout, sys.stderr |
be7103467d2e
Add 'hg serve' command for stand-alone server
mpm@selenic.com
parents:
157
diff
changeset
|
674 try: |
be7103467d2e
Add 'hg serve' command for stand-alone server
mpm@selenic.com
parents:
157
diff
changeset
|
675 sys.stdin = self.rfile |
be7103467d2e
Add 'hg serve' command for stand-alone server
mpm@selenic.com
parents:
157
diff
changeset
|
676 sys.stdout = self.wfile |
be7103467d2e
Add 'hg serve' command for stand-alone server
mpm@selenic.com
parents:
157
diff
changeset
|
677 sys.argv = ["hgweb.py"] |
be7103467d2e
Add 'hg serve' command for stand-alone server
mpm@selenic.com
parents:
157
diff
changeset
|
678 if '=' not in query: |
be7103467d2e
Add 'hg serve' command for stand-alone server
mpm@selenic.com
parents:
157
diff
changeset
|
679 sys.argv.append(query) |
be7103467d2e
Add 'hg serve' command for stand-alone server
mpm@selenic.com
parents:
157
diff
changeset
|
680 self.send_response(200, "Script output follows") |
be7103467d2e
Add 'hg serve' command for stand-alone server
mpm@selenic.com
parents:
157
diff
changeset
|
681 hg.run() |
be7103467d2e
Add 'hg serve' command for stand-alone server
mpm@selenic.com
parents:
157
diff
changeset
|
682 finally: |
be7103467d2e
Add 'hg serve' command for stand-alone server
mpm@selenic.com
parents:
157
diff
changeset
|
683 sys.argv, sys.stdin, sys.stdout, sys.stderr = save |
be7103467d2e
Add 'hg serve' command for stand-alone server
mpm@selenic.com
parents:
157
diff
changeset
|
684 |
be7103467d2e
Add 'hg serve' command for stand-alone server
mpm@selenic.com
parents:
157
diff
changeset
|
685 hg = hgweb(path, name, templates) |
be7103467d2e
Add 'hg serve' command for stand-alone server
mpm@selenic.com
parents:
157
diff
changeset
|
686 httpd = BaseHTTPServer.HTTPServer((address, port), hgwebhandler) |
be7103467d2e
Add 'hg serve' command for stand-alone server
mpm@selenic.com
parents:
157
diff
changeset
|
687 httpd.serve_forever() |