blob: 303efabe1bba5df12ddd3c34057d13db668679b3 [file] [log] [blame]
Guido van Rossumadb3a9d1997-05-21 07:24:50 +00001#! /depot/sundry/plat/bin/python1.4
2
Guido van Rossumd7bfa801997-05-21 21:31:39 +00003"""Interactive FAQ project.
4
5XXX TO DO
6
Guido van Rossumf701bf11997-05-21 22:25:56 +00007- customize rcs command pathnames
8- recognize urls and email addresses and turn them into <A> tags
Guido van Rossumd7bfa801997-05-21 21:31:39 +00009- use cookies to keep Name/email the same
10- explanation of editing somewhere
11- various embellishments, GIFs, crosslinks, hints, etc.
12- create new sections
13- rearrange entries
14- delete entries
15- log changes
16- send email on changes
17- optional staging of entries until reviewed?
18- review revision log and older versions
19- freeze entries
20- username/password for editors
21- Change references to other Q's and whole sections
22- Browse should display menu of 7 sections & let you pick
23 (or frontpage should have the option to browse a section or all)
24- support adding annotations, too
25
26"""
Guido van Rossumadb3a9d1997-05-21 07:24:50 +000027
Guido van Rossumf701bf11997-05-21 22:25:56 +000028import cgi, string, os, sys
Guido van Rossumadb3a9d1997-05-21 07:24:50 +000029
30NAMEPAT = "faq??.???.htp"
Guido van Rossumd7bfa801997-05-21 21:31:39 +000031NAMEREG = "^faq\([0-9][0-9]\)\.\([0-9][0-9][0-9]\)\.htp$"
Guido van Rossumadb3a9d1997-05-21 07:24:50 +000032
33class FAQServer:
34
35 def __init__(self):
36 pass
37
38 def main(self):
39 self.form = cgi.FieldStorage()
40 req = self.req or 'frontpage'
41 try:
42 method = getattr(self, 'do_%s' % req)
43 except AttributeError:
44 print "Unrecognized request type", req
45 else:
46 method()
47
Guido van Rossum3c3354c1997-05-21 16:52:18 +000048 KEYS = ['req', 'query', 'name', 'text', 'commit', 'title',
Guido van Rossumd7bfa801997-05-21 21:31:39 +000049 'author', 'email', 'log', 'section', 'number', 'add']
Guido van Rossumadb3a9d1997-05-21 07:24:50 +000050
51 def __getattr__(self, key):
52 if key not in self.KEYS:
53 raise AttributeError
Guido van Rossum3c3354c1997-05-21 16:52:18 +000054 try:
Guido van Rossumf701bf11997-05-21 22:25:56 +000055 value = self.form[key].value
56 setattr(self, key, value)
57 return value
Guido van Rossum3c3354c1997-05-21 16:52:18 +000058 except KeyError:
59 return ''
Guido van Rossumadb3a9d1997-05-21 07:24:50 +000060
61 def do_frontpage(self):
62 print """
63 <TITLE>Python FAQ (alpha 1)</TITLE>
64 <H1>Python FAQ Front Page</H1>
65 <UL>
Guido van Rossumd7bfa801997-05-21 21:31:39 +000066 <LI><A HREF="faq.py?req=index">FAQ index</A>
67 <LI><A HREF="faq.py?req=all">The whole FAQ</A>
Guido van Rossum3c3354c1997-05-21 16:52:18 +000068 <LI><A HREF="faq.py?req=roulette">FAQ roulette</A>
Guido van Rossumd7bfa801997-05-21 21:31:39 +000069 <LI><A HREF="faq.py?req=recent">Recently changed FAQ entries</A>
70 <LI><A HREF="faq.py?req=add">Add a new FAQ entry</A>
Guido van Rossumadb3a9d1997-05-21 07:24:50 +000071 </UL>
72
Guido van Rossumd7bfa801997-05-21 21:31:39 +000073 <H2>Search the FAQ</H2>
74
Guido van Rossumadb3a9d1997-05-21 07:24:50 +000075 <FORM ACTION="faq.py?req=query">
76 <INPUT TYPE=text NAME=query>
77 <INPUT TYPE=submit VALUE="Search">
78 <INPUT TYPE=hidden NAME=req VALUE=query>
79 </FORM>
80
81 Disclaimer: these pages are intended to be edited by anyone.
82 Please exercise discretion when editing, don't be rude, etc.
83 """
84
Guido van Rossumd7bfa801997-05-21 21:31:39 +000085 def do_index(self):
86 print """
87 <TITLE>Python FAQ Index</TITLE>
88 <H1>Python FAQ Index</H1>
89 """
90 names = os.listdir(os.curdir)
91 names.sort()
92 section = None
93 for name in names:
94 headers, text = self.read(name)
95 if headers:
96 title = headers['title']
97 i = string.find(title, '.')
98 nsec = title[:i]
99 if nsec != section:
100 if section:
101 print """
102 <P>
103 <LI><A HREF="faq.py?req=add&amp;section=%s"
104 >Add new entry</A> (at this point)
105 </UL>
106 """ % section
107 section = nsec
108 print "<H2>Section %s</H2>" % section
109 print "<UL>"
110 print '<LI><A HREF="faq.py?req=show&name=%s">%s</A>' % (
111 name, cgi.escape(title))
112 if section:
113 print """
114 <P>
115 <LI><A HREF="faq.py?req=add&amp;section=%s">Add new entry</A>
116 (at this point)
117 </UL>
118 """ % section
119 else:
120 print "No FAQ entries?!?!"
121
122 def do_show(self):
123 name = self.name
124 headers, text = self.read(name)
125 if not headers:
126 print "Invalid file name", name
127 return
128 self.show(name, headers['title'], text, 1)
129
130 def do_all(self):
Guido van Rossumadb3a9d1997-05-21 07:24:50 +0000131 print """
132 <TITLE>Python FAQ</TITLE>
133 <H1>Python FAQ</H1>
134 <HR>
135 """
136 names = os.listdir(os.curdir)
137 names.sort()
Guido van Rossumd7bfa801997-05-21 21:31:39 +0000138 section = None
Guido van Rossumadb3a9d1997-05-21 07:24:50 +0000139 for name in names:
Guido van Rossum3c3354c1997-05-21 16:52:18 +0000140 headers, text = self.read(name)
Guido van Rossumadb3a9d1997-05-21 07:24:50 +0000141 if headers:
Guido van Rossumd7bfa801997-05-21 21:31:39 +0000142 title = headers['title']
143 i = string.find(title, '.')
144 nsec = title[:i]
145 if nsec != section:
146 section = nsec
147 print "<H1>Section %s</H1>" % section
148 print "<HR>"
149 self.show(name, title, text, 1)
Guido van Rossumadb3a9d1997-05-21 07:24:50 +0000150 n = n+1
Guido van Rossumd7bfa801997-05-21 21:31:39 +0000151 if not section:
Guido van Rossumadb3a9d1997-05-21 07:24:50 +0000152 print "No FAQ entries?!?!"
153
154 def do_roulette(self):
155 import whrandom
156 print """
157 <TITLE>Python FAQ Roulette</TITLE>
158 <H1>Python FAQ Roulette</H1>
159 Please check the correctness of the entry below.
160 If you find any problems, please edit the entry.
161 <P>
162 <HR>
163 """
164 names = os.listdir(os.curdir)
165 while names:
166 name = whrandom.choice(names)
Guido van Rossum3c3354c1997-05-21 16:52:18 +0000167 headers, text = self.read(name)
Guido van Rossumadb3a9d1997-05-21 07:24:50 +0000168 if headers:
Guido van Rossum3c3354c1997-05-21 16:52:18 +0000169 self.show(name, headers['title'], text, 1)
Guido van Rossumadb3a9d1997-05-21 07:24:50 +0000170 print '<P><A HREF="faq.py?req=roulette">Show another one</A>'
171 break
172 else:
173 names.remove(name)
174 else:
175 print "No FAQ entries?!?!"
176
Guido van Rossumd7bfa801997-05-21 21:31:39 +0000177 def do_recent(self):
178 import fnmatch, stat
179 names = os.listdir(os.curdir)
180 now = time.time()
181 list = []
182 for name in names:
183 if not fnmatch.fnmatch(name, NAMEPAT):
184 continue
185 try:
186 st = os.stat(name)
187 except os.error:
188 continue
189 tuple = (st[stat.ST_MTIME], name)
190 list.append(tuple)
191 list.sort()
192 list.reverse()
Guido van Rossumadb3a9d1997-05-21 07:24:50 +0000193 print """
Guido van Rossumd7bfa801997-05-21 21:31:39 +0000194 <TITLE>Python FAQ, Most Recently Modified First</TITLE>
195 <H1>Python FAQ, Most Recently Modified First</H1>
196 <HR>
Guido van Rossumadb3a9d1997-05-21 07:24:50 +0000197 """
Guido van Rossumd7bfa801997-05-21 21:31:39 +0000198 n = 0
199 for (mtime, name) in list:
200 headers, text = self.read(name)
201 if headers:
202 self.show(name, headers['title'], text, 1)
203 n = n+1
204 if not n:
205 print "No FAQ entries?!?!"
Guido van Rossumadb3a9d1997-05-21 07:24:50 +0000206
207 def do_query(self):
208 import regex
209 print "<TITLE>Python FAQ Query Results</TITLE>"
210 print "<H1>Python FAQ Query Results</H1>"
211 query = self.query
212 if not query:
213 print "No query string"
214 return
215 p = regex.compile(query, regex.casefold)
216 names = os.listdir(os.curdir)
217 names.sort()
218 print "<HR>"
219 n = 0
220 for name in names:
Guido van Rossum3c3354c1997-05-21 16:52:18 +0000221 headers, text = self.read(name)
Guido van Rossumadb3a9d1997-05-21 07:24:50 +0000222 if headers:
223 title = headers['title']
Guido van Rossum3c3354c1997-05-21 16:52:18 +0000224 if p.search(title) >= 0 or p.search(text) >= 0:
225 self.show(name, title, text, 1)
Guido van Rossumadb3a9d1997-05-21 07:24:50 +0000226 n = n+1
227 if not n:
228 print "No hits."
229
Guido van Rossumd7bfa801997-05-21 21:31:39 +0000230 def do_add(self):
231 section = self.section
232 if not section:
233 print """
234 <TITLE>How to add a new FAQ entry</TITLE>
235 <H1>How to add a new FAQ entry</H1>
236
237 Go to the <A HREF="faq.py?req=index">FAQ index</A>
238 and click on the "Add new entry" link at the end
239 of the section to which you want to add the entry.
240 """
241 return
242 try:
243 nsec = string.atoi(section)
244 except ValueError:
245 print "Bad section number", nsec
246 names = os.listdir(os.curdir)
247 max = 0
248 import regex
249 prog = regex.compile(NAMEREG)
250 for name in names:
251 if prog.match(name) >= 0:
252 s1, s2 = prog.group(1, 2)
253 n1, n2 = string.atoi(s1), string.atoi(s2)
254 if n1 == nsec:
255 if n2 > max:
256 max = n2
257 if not max:
258 print "Can't add new sections yet."
259 return
260 num = max+1
261 name = "faq%02d.%03d.htp" % (nsec, num)
262 self.name = name
263 self.add = "yes"
264 self.number = str(num)
265 self.do_edit()
266
Guido van Rossumadb3a9d1997-05-21 07:24:50 +0000267 def do_edit(self):
268 name = self.name
Guido van Rossum3c3354c1997-05-21 16:52:18 +0000269 headers, text = self.read(name)
Guido van Rossumadb3a9d1997-05-21 07:24:50 +0000270 if not headers:
271 print "Invalid file name", name
272 return
273 print """
274 <TITLE>Python FAQ Edit Form</TITLE>
275 <H1>Python FAQ Edit Form</H1>
276 """
Guido van Rossumadb3a9d1997-05-21 07:24:50 +0000277 title = headers['title']
Guido van Rossum3c3354c1997-05-21 16:52:18 +0000278 print "<FORM METHOD=POST ACTION=faq.py>"
Guido van Rossumd7bfa801997-05-21 21:31:39 +0000279 self.showedit(name, title, text)
280 if self.add:
281 print """
282 <INPUT TYPE=hidden NAME=add VALUE=%s>
283 <INPUT TYPE=hidden NAME=section VALUE=%s>
284 <INPUT TYPE=hidden NAME=number VALUE=%s>
285 """ % (self.add, self.section, self.number)
Guido van Rossumadb3a9d1997-05-21 07:24:50 +0000286 print """
Guido van Rossumadb3a9d1997-05-21 07:24:50 +0000287 <INPUT TYPE=submit VALUE="Review Edit">
288 <INPUT TYPE=hidden NAME=req VALUE=review>
289 <INPUT TYPE=hidden NAME=name VALUE=%s>
290 </FORM>
291 <HR>
292 """ % name
Guido van Rossum3c3354c1997-05-21 16:52:18 +0000293 self.show(name, title, text)
Guido van Rossumadb3a9d1997-05-21 07:24:50 +0000294
295 def do_review(self):
Guido van Rossum3c3354c1997-05-21 16:52:18 +0000296 if self.commit:
297 self.checkin()
298 return
Guido van Rossumadb3a9d1997-05-21 07:24:50 +0000299 name = self.name
300 text = self.text
Guido van Rossumadb3a9d1997-05-21 07:24:50 +0000301 title = self.title
Guido van Rossum3c3354c1997-05-21 16:52:18 +0000302 headers, oldtext = self.read(name)
Guido van Rossumadb3a9d1997-05-21 07:24:50 +0000303 if not headers:
304 print "Invalid file name", name
305 return
306 print """
307 <TITLE>Python FAQ Review Form</TITLE>
308 <H1>Python FAQ Review Form</H1>
Guido van Rossumd7bfa801997-05-21 21:31:39 +0000309 <HR>
Guido van Rossumadb3a9d1997-05-21 07:24:50 +0000310 """
Guido van Rossum3c3354c1997-05-21 16:52:18 +0000311 self.show(name, title, text)
312 print "<FORM METHOD=POST ACTION=faq.py>"
313 if self.log and self.author and '@' in self.email:
314 print """
315 <INPUT TYPE=submit NAME=commit VALUE="Commit">
316 Click this button to commit the change.
317 <P>
318 <HR>
319 <P>
320 """
321 else:
322 print """
323 To commit this change, please enter your name,
324 email and a log message in the form below.
325 <P>
326 <HR>
327 <P>
328 """
Guido van Rossumd7bfa801997-05-21 21:31:39 +0000329 self.showedit(name, title, text)
330 if self.add:
331 print """
332 <INPUT TYPE=hidden NAME=add VALUE=%s>
333 <INPUT TYPE=hidden NAME=section VALUE=%s>
334 <INPUT TYPE=hidden NAME=number VALUE=%s>
335 """ % (self.add, self.section, self.number)
Guido van Rossumadb3a9d1997-05-21 07:24:50 +0000336 print """
Guido van Rossumadb3a9d1997-05-21 07:24:50 +0000337 <BR>
338 <INPUT TYPE=submit VALUE="Review Edit">
339 <INPUT TYPE=hidden NAME=req VALUE=review>
340 <INPUT TYPE=hidden NAME=name VALUE=%s>
341 </FORM>
342 <HR>
343 """ % name
344
Guido van Rossumf701bf11997-05-21 22:25:56 +0000345 def do_info(self):
346 name = self.name
347 headers, text = self.read(name)
348 if not headers:
349 print "Invalid file name", name
350 return
351 print '<PRE>'
352 sys.stdout.flush()
353 os.system("/depot/gnu/plat/bin/rlog -r %s </dev/null 2>&1" % self.name)
354 print '</PRE>'
355 print '<A HREF="faq.py?req=rlog&name=%s">View full rcs log</A>' % name
356
357 def do_rlog(self):
358 name = self.name
359 headers, text = self.read(name)
360 if not headers:
361 print "Invalid file name", name
362 return
363 print '<PRE>'
364 sys.stdout.flush()
365 os.system("/depot/gnu/plat/bin/rlog %s </dev/null 2>&1" % self.name)
366 print '</PRE>'
367
Guido van Rossum3c3354c1997-05-21 16:52:18 +0000368 def checkin(self):
369 import regsub, time, tempfile
370 name = self.name
371
372 headers, oldtext = self.read(name)
373 if not headers:
374 print "Invalid file name", name
375 return
376 text = self.text
377 title = self.title
378 author = self.author
379 email = self.email
380 log = self.log
381 text = regsub.gsub("\r\n", "\n", text)
382 log = regsub.gsub("\r\n", "\n", log)
383 author = string.join(string.split(author))
384 email = string.join(string.split(email))
385 title = string.join(string.split(title))
386 oldtitle = headers['title']
387 oldtitle = string.join(string.split(oldtitle))
388 text = string.strip(text)
389 oldtext = string.strip(oldtext)
390 if text == oldtext and title == oldtitle:
391 print "No changes."
392 # XXX Should exit more ceremoniously
393 return
Guido van Rossumd7bfa801997-05-21 21:31:39 +0000394 # Check that the FAQ entry number didn't change
Guido van Rossum3c3354c1997-05-21 16:52:18 +0000395 if string.split(title)[:1] != string.split(oldtitle)[:1]:
Guido van Rossumd7bfa801997-05-21 21:31:39 +0000396 print "Don't change the FAQ entry number please."
Guido van Rossum3c3354c1997-05-21 16:52:18 +0000397 # XXX Should exit more ceremoniously
398 return
399 remhost = os.environ["REMOTE_HOST"]
400 remaddr = os.environ["REMOTE_ADDR"]
401 try:
402 os.unlink(name + "~")
403 except os.error:
404 pass
405 try:
406 os.rename(name, name + "~")
407 except os.error:
408 pass
409 try:
410 os.unlink(name)
411 except os.error:
412 pass
413 try:
414 f = open(name, "w")
415 except IOError, msg:
416 print "Can't open", name, "for writing:", cgi.escape(str(msg))
417 # XXX Should exit more ceremoniously
418 return
419 now = time.ctime(time.time())
420 f.write("Title: %s\n" % title)
421 f.write("Last-Changed-Date: %s\n" % now)
422 f.write("Last-Changed-Author: %s\n" % author)
423 f.write("Last-Changed-Email: %s\n" % email)
424 f.write("Last-Changed-Remote-Host: %s\n" % remhost)
425 f.write("Last-Changed-Remote-Address: %s\n" % remaddr)
426 keys = headers.keys()
427 keys.sort()
428 keys.remove('title')
429 for key in keys:
430 if key[:13] != 'last-changed-':
431 f.write("%s: %s\n" % (string.capwords(key, '-'),
432 headers[key]))
433 f.write("\n")
434 f.write(text)
435 f.write("\n")
436 f.close()
437
438 tfn = tempfile.mktemp()
439 f = open(tfn, "w")
440 f.write("Last-Changed-Date: %s\n" % now)
441 f.write("Last-Changed-Author: %s\n" % author)
442 f.write("Last-Changed-Email: %s\n" % email)
443 f.write("Last-Changed-Remote-Host: %s\n" % remhost)
444 f.write("Last-Changed-Remote-Address: %s\n" % remaddr)
445 f.write("\n")
446 f.write(log)
447 f.write("\n")
448 f.close()
449
450 p = os.popen("""
451 /depot/gnu/plat/bin/rcs -l %s </dev/null 2>&1
452 /depot/gnu/plat/bin/ci -u %s <%s 2>&1
453 rm -f %s
454 """ % (name, name, tfn, tfn))
455 output = p.read()
456 sts = p.close()
457 if not sts:
458 print """
Guido van Rossumd7bfa801997-05-21 21:31:39 +0000459 <TITLE>Python FAQ Entry Edited</TITLE>
Guido van Rossum3c3354c1997-05-21 16:52:18 +0000460 <H1>Python FAQ Entry Edited</H1>
461 <HR>
462 """
463 self.show(name, title, text, 1)
464 if output:
465 print "<PRE>%s</PRE>" % cgi.escape(output)
466 else:
467 print """
468 <H1>Python FAQ Entry Commit Failed</H1>
469 Exit status 0x%04x
470 """ % sts
471 if output:
472 print "<PRE>%s</PRE>" % cgi.escape(output)
473
Guido van Rossumd7bfa801997-05-21 21:31:39 +0000474 def showedit(self, name, title, text):
Guido van Rossum3c3354c1997-05-21 16:52:18 +0000475 print """
476 Title: <INPUT TYPE=text SIZE=70 NAME=title VALUE="%s"<BR>
477 <TEXTAREA COLS=80 ROWS=20 NAME=text>""" % title
478 print cgi.escape(string.strip(text))
479 print """</TEXTAREA>
480 <BR>
481 Please provide the following information for logging purposes:
482 <BR>
483 <CODE>Name : </CODE><INPUT TYPE=text SIZE=70 NAME=author VALUE="%s"<BR>
484 <CODE>Email: </CODE><INPUT TYPE=text SIZE=70 NAME=email VALUE="%s"<BR>
485 Log message (reason for the change):<BR>
486 <TEXTAREA COLS=80 ROWS=5 NAME=log>\n%s\n</TEXTAREA>
487 """ % (self.author, self.email, self.log)
Guido van Rossumadb3a9d1997-05-21 07:24:50 +0000488
489 def showheaders(self, headers):
490 print "<UL>"
491 keys = map(string.lower, headers.keys())
492 keys.sort()
493 for key in keys:
494 print "<LI><B>%s:</B> %s" % (string.capwords(key, '-'),
495 headers[key] or '')
496 print "</UL>"
497
498 def read(self, name):
499 import fnmatch, rfc822
500 if not fnmatch.fnmatch(name, NAMEPAT):
501 return None, None
Guido van Rossumd7bfa801997-05-21 21:31:39 +0000502 if self.add:
503 try:
504 fname = "faq%02d.%03d.htp" % (string.atoi(self.section),
505 string.atoi(self.number))
506 except ValueError:
507 return None, None
508 if fname != name:
509 return None, None
510 headers = {'title': "%s.%s. " % (self.section, self.number)}
511 text = ""
512 return headers, text
Guido van Rossumadb3a9d1997-05-21 07:24:50 +0000513 f = open(name)
514 headers = rfc822.Message(f)
Guido van Rossum3c3354c1997-05-21 16:52:18 +0000515 text = f.read()
Guido van Rossumadb3a9d1997-05-21 07:24:50 +0000516 f.close()
Guido van Rossum3c3354c1997-05-21 16:52:18 +0000517 return headers, text
Guido van Rossumadb3a9d1997-05-21 07:24:50 +0000518
Guido van Rossum3c3354c1997-05-21 16:52:18 +0000519 def show(self, name, title, text, edit=0):
Guido van Rossumadb3a9d1997-05-21 07:24:50 +0000520 # XXX Should put <A> tags around recognizable URLs
521 # XXX Should also turn "see section N" into hyperlinks
Guido van Rossumd7bfa801997-05-21 21:31:39 +0000522 print "<H2>%s</H2>" % cgi.escape(title)
Guido van Rossumadb3a9d1997-05-21 07:24:50 +0000523 pre = 0
Guido van Rossum3c3354c1997-05-21 16:52:18 +0000524 for line in string.split(text, '\n'):
Guido van Rossumadb3a9d1997-05-21 07:24:50 +0000525 if not string.strip(line):
526 if pre:
527 print '</PRE>'
528 pre = 0
529 else:
530 print '<P>'
531 else:
Guido van Rossum3c3354c1997-05-21 16:52:18 +0000532 if line == string.lstrip(line): # I.e., no leading whitespace
Guido van Rossumadb3a9d1997-05-21 07:24:50 +0000533 if pre:
534 print '</PRE>'
535 pre = 0
536 else:
537 if not pre:
538 print '<PRE>'
539 pre = 1
Guido van Rossumd7bfa801997-05-21 21:31:39 +0000540 print cgi.escape(line)
Guido van Rossumadb3a9d1997-05-21 07:24:50 +0000541 if pre:
542 print '</PRE>'
543 pre = 0
544 print '<P>'
545 if edit:
Guido van Rossumf701bf11997-05-21 22:25:56 +0000546 print """
547 <A HREF="faq.py?req=edit&name=%s">Edit this entry</A> /
548 <A HREF="faq.py?req=info&name=%s" TARGET=_blank>Log info</A>
549 """ % (name, name)
Guido van Rossumadb3a9d1997-05-21 07:24:50 +0000550 print '<P>'
551 print "<HR>"
552
553
554print "Content-type: text/html\n"
Guido van Rossum3c3354c1997-05-21 16:52:18 +0000555dt = 0
Guido van Rossumadb3a9d1997-05-21 07:24:50 +0000556try:
Guido van Rossum3c3354c1997-05-21 16:52:18 +0000557 import time
558 t1 = time.time()
Guido van Rossumadb3a9d1997-05-21 07:24:50 +0000559 x = FAQServer()
560 x.main()
Guido van Rossum3c3354c1997-05-21 16:52:18 +0000561 t2 = time.time()
562 dt = t2-t1
Guido van Rossumadb3a9d1997-05-21 07:24:50 +0000563except:
564 print "<HR>Sorry, an error occurred"
565 cgi.print_exception()
Guido van Rossumd7bfa801997-05-21 21:31:39 +0000566print "<P>(time = %s seconds)" % str(round(dt, 3))