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