1997-05-21 04:24:50 -03:00
|
|
|
#! /depot/sundry/plat/bin/python1.4
|
|
|
|
|
|
|
|
"""Interactive FAQ project."""
|
|
|
|
|
|
|
|
import cgi, string, os
|
|
|
|
|
|
|
|
NAMEPAT = "faq??.???.htp"
|
|
|
|
|
|
|
|
class FAQServer:
|
|
|
|
|
|
|
|
def __init__(self):
|
|
|
|
pass
|
|
|
|
|
|
|
|
def main(self):
|
|
|
|
self.form = cgi.FieldStorage()
|
|
|
|
req = self.req or 'frontpage'
|
|
|
|
try:
|
|
|
|
method = getattr(self, 'do_%s' % req)
|
|
|
|
except AttributeError:
|
|
|
|
print "Unrecognized request type", req
|
|
|
|
else:
|
|
|
|
method()
|
|
|
|
|
1997-05-21 13:52:18 -03:00
|
|
|
KEYS = ['req', 'query', 'name', 'text', 'commit', 'title',
|
|
|
|
'author', 'email', 'log']
|
1997-05-21 04:24:50 -03:00
|
|
|
|
|
|
|
def __getattr__(self, key):
|
|
|
|
if key not in self.KEYS:
|
|
|
|
raise AttributeError
|
1997-05-21 13:52:18 -03:00
|
|
|
try:
|
1997-05-21 04:24:50 -03:00
|
|
|
item = self.form[key]
|
|
|
|
return item.value
|
1997-05-21 13:52:18 -03:00
|
|
|
except KeyError:
|
|
|
|
return ''
|
1997-05-21 04:24:50 -03:00
|
|
|
|
|
|
|
def do_frontpage(self):
|
|
|
|
print """
|
|
|
|
<TITLE>Python FAQ (alpha 1)</TITLE>
|
|
|
|
<H1>Python FAQ Front Page</H1>
|
|
|
|
<UL>
|
|
|
|
<LI><A HREF="faq.py?req=search">Search the FAQ</A>
|
|
|
|
<LI><A HREF="faq.py?req=browse">Browse the FAQ</A>
|
1997-05-21 13:52:18 -03:00
|
|
|
<LI><A HREF="faq.py?req=submit">Submit a new FAQ entry</A> (not yet)
|
|
|
|
<LI><A HREF="faq.py?req=roulette">FAQ roulette</A>
|
1997-05-21 04:24:50 -03:00
|
|
|
</UL>
|
|
|
|
|
|
|
|
<FORM ACTION="faq.py?req=query">
|
|
|
|
<INPUT TYPE=text NAME=query>
|
|
|
|
<INPUT TYPE=submit VALUE="Search">
|
|
|
|
<INPUT TYPE=hidden NAME=req VALUE=query>
|
|
|
|
</FORM>
|
|
|
|
|
|
|
|
Disclaimer: these pages are intended to be edited by anyone.
|
|
|
|
Please exercise discretion when editing, don't be rude, etc.
|
|
|
|
"""
|
|
|
|
|
|
|
|
def do_browse(self):
|
|
|
|
print """
|
|
|
|
<TITLE>Python FAQ</TITLE>
|
|
|
|
<H1>Python FAQ</H1>
|
|
|
|
<HR>
|
|
|
|
"""
|
|
|
|
names = os.listdir(os.curdir)
|
|
|
|
names.sort()
|
|
|
|
n = 0
|
|
|
|
for name in names:
|
1997-05-21 13:52:18 -03:00
|
|
|
headers, text = self.read(name)
|
1997-05-21 04:24:50 -03:00
|
|
|
if headers:
|
1997-05-21 13:52:18 -03:00
|
|
|
self.show(name, headers['title'], text, 1)
|
1997-05-21 04:24:50 -03:00
|
|
|
n = n+1
|
|
|
|
if not n:
|
|
|
|
print "No FAQ entries?!?!"
|
|
|
|
|
|
|
|
def do_roulette(self):
|
|
|
|
import whrandom
|
|
|
|
print """
|
|
|
|
<TITLE>Python FAQ Roulette</TITLE>
|
|
|
|
<H1>Python FAQ Roulette</H1>
|
|
|
|
Please check the correctness of the entry below.
|
|
|
|
If you find any problems, please edit the entry.
|
|
|
|
<P>
|
|
|
|
<HR>
|
|
|
|
"""
|
|
|
|
names = os.listdir(os.curdir)
|
|
|
|
while names:
|
|
|
|
name = whrandom.choice(names)
|
1997-05-21 13:52:18 -03:00
|
|
|
headers, text = self.read(name)
|
1997-05-21 04:24:50 -03:00
|
|
|
if headers:
|
1997-05-21 13:52:18 -03:00
|
|
|
self.show(name, headers['title'], text, 1)
|
1997-05-21 04:24:50 -03:00
|
|
|
print '<P><A HREF="faq.py?req=roulette">Show another one</A>'
|
|
|
|
break
|
|
|
|
else:
|
|
|
|
names.remove(name)
|
|
|
|
else:
|
|
|
|
print "No FAQ entries?!?!"
|
|
|
|
|
|
|
|
def do_search(self):
|
|
|
|
print """
|
|
|
|
<TITLE>Search the Python FAQ</TITLE>
|
|
|
|
<H1>Search the Python FAQ</H1>
|
|
|
|
|
|
|
|
<FORM ACTION="faq.py?req=query">
|
|
|
|
<INPUT TYPE=text NAME=query>
|
|
|
|
<INPUT TYPE=submit VALUE="Search">
|
|
|
|
<INPUT TYPE=hidden NAME=req VALUE=query>
|
|
|
|
</FORM>
|
|
|
|
"""
|
|
|
|
|
|
|
|
def do_query(self):
|
|
|
|
import regex
|
|
|
|
print "<TITLE>Python FAQ Query Results</TITLE>"
|
|
|
|
print "<H1>Python FAQ Query Results</H1>"
|
|
|
|
query = self.query
|
|
|
|
if not query:
|
|
|
|
print "No query string"
|
|
|
|
return
|
|
|
|
p = regex.compile(query, regex.casefold)
|
|
|
|
names = os.listdir(os.curdir)
|
|
|
|
names.sort()
|
|
|
|
print "<HR>"
|
|
|
|
n = 0
|
|
|
|
for name in names:
|
1997-05-21 13:52:18 -03:00
|
|
|
headers, text = self.read(name)
|
1997-05-21 04:24:50 -03:00
|
|
|
if headers:
|
|
|
|
title = headers['title']
|
1997-05-21 13:52:18 -03:00
|
|
|
if p.search(title) >= 0 or p.search(text) >= 0:
|
|
|
|
self.show(name, title, text, 1)
|
1997-05-21 04:24:50 -03:00
|
|
|
n = n+1
|
|
|
|
if not n:
|
|
|
|
print "No hits."
|
|
|
|
|
|
|
|
def do_edit(self):
|
|
|
|
name = self.name
|
1997-05-21 13:52:18 -03:00
|
|
|
headers, text = self.read(name)
|
1997-05-21 04:24:50 -03:00
|
|
|
if not headers:
|
|
|
|
print "Invalid file name", name
|
|
|
|
return
|
|
|
|
print """
|
|
|
|
<TITLE>Python FAQ Edit Form</TITLE>
|
|
|
|
<H1>Python FAQ Edit Form</H1>
|
|
|
|
"""
|
|
|
|
title = headers['title']
|
1997-05-21 13:52:18 -03:00
|
|
|
print "<FORM METHOD=POST ACTION=faq.py>"
|
|
|
|
self.showedit(name, headers, text)
|
1997-05-21 04:24:50 -03:00
|
|
|
print """
|
|
|
|
<INPUT TYPE=submit VALUE="Review Edit">
|
|
|
|
<INPUT TYPE=hidden NAME=req VALUE=review>
|
|
|
|
<INPUT TYPE=hidden NAME=name VALUE=%s>
|
|
|
|
</FORM>
|
|
|
|
<HR>
|
|
|
|
""" % name
|
1997-05-21 13:52:18 -03:00
|
|
|
self.show(name, title, text)
|
1997-05-21 04:24:50 -03:00
|
|
|
|
|
|
|
def do_review(self):
|
1997-05-21 13:52:18 -03:00
|
|
|
if self.commit:
|
|
|
|
self.checkin()
|
|
|
|
return
|
1997-05-21 04:24:50 -03:00
|
|
|
name = self.name
|
|
|
|
text = self.text
|
|
|
|
title = self.title
|
1997-05-21 13:52:18 -03:00
|
|
|
headers, oldtext = self.read(name)
|
1997-05-21 04:24:50 -03:00
|
|
|
if not headers:
|
|
|
|
print "Invalid file name", name
|
|
|
|
return
|
|
|
|
print """
|
|
|
|
<TITLE>Python FAQ Review Form</TITLE>
|
|
|
|
<H1>Python FAQ Review Form</H1>
|
|
|
|
"""
|
1997-05-21 13:52:18 -03:00
|
|
|
self.show(name, title, text)
|
|
|
|
print "<FORM METHOD=POST ACTION=faq.py>"
|
|
|
|
if self.log and self.author and '@' in self.email:
|
|
|
|
print """
|
|
|
|
<INPUT TYPE=submit NAME=commit VALUE="Commit">
|
|
|
|
Click this button to commit the change.
|
|
|
|
<P>
|
|
|
|
<HR>
|
|
|
|
<P>
|
|
|
|
"""
|
|
|
|
else:
|
|
|
|
print """
|
|
|
|
To commit this change, please enter your name,
|
|
|
|
email and a log message in the form below.
|
|
|
|
<P>
|
|
|
|
<HR>
|
|
|
|
<P>
|
|
|
|
"""
|
|
|
|
self.showedit(name, headers, text)
|
1997-05-21 04:24:50 -03:00
|
|
|
print """
|
|
|
|
<BR>
|
|
|
|
<INPUT TYPE=submit VALUE="Review Edit">
|
|
|
|
<INPUT TYPE=hidden NAME=req VALUE=review>
|
|
|
|
<INPUT TYPE=hidden NAME=name VALUE=%s>
|
|
|
|
</FORM>
|
|
|
|
<HR>
|
|
|
|
""" % name
|
|
|
|
|
1997-05-21 13:52:18 -03:00
|
|
|
def checkin(self):
|
|
|
|
import regsub, time, tempfile
|
|
|
|
name = self.name
|
|
|
|
|
|
|
|
headers, oldtext = self.read(name)
|
|
|
|
if not headers:
|
|
|
|
print "Invalid file name", name
|
|
|
|
return
|
|
|
|
text = self.text
|
|
|
|
title = self.title
|
|
|
|
author = self.author
|
|
|
|
email = self.email
|
|
|
|
log = self.log
|
|
|
|
text = regsub.gsub("\r\n", "\n", text)
|
|
|
|
log = regsub.gsub("\r\n", "\n", log)
|
|
|
|
author = string.join(string.split(author))
|
|
|
|
email = string.join(string.split(email))
|
|
|
|
title = string.join(string.split(title))
|
|
|
|
oldtitle = headers['title']
|
|
|
|
oldtitle = string.join(string.split(oldtitle))
|
|
|
|
text = string.strip(text)
|
|
|
|
oldtext = string.strip(oldtext)
|
|
|
|
if text == oldtext and title == oldtitle:
|
|
|
|
print "No changes."
|
|
|
|
# XXX Should exit more ceremoniously
|
|
|
|
return
|
|
|
|
# Check that the question number didn't change
|
|
|
|
if string.split(title)[:1] != string.split(oldtitle)[:1]:
|
|
|
|
print "Don't change the question number please."
|
|
|
|
# XXX Should exit more ceremoniously
|
|
|
|
return
|
|
|
|
remhost = os.environ["REMOTE_HOST"]
|
|
|
|
remaddr = os.environ["REMOTE_ADDR"]
|
|
|
|
try:
|
|
|
|
os.unlink(name + "~")
|
|
|
|
except os.error:
|
|
|
|
pass
|
|
|
|
try:
|
|
|
|
os.rename(name, name + "~")
|
|
|
|
except os.error:
|
|
|
|
pass
|
|
|
|
try:
|
|
|
|
os.unlink(name)
|
|
|
|
except os.error:
|
|
|
|
pass
|
|
|
|
try:
|
|
|
|
f = open(name, "w")
|
|
|
|
except IOError, msg:
|
|
|
|
print "Can't open", name, "for writing:", cgi.escape(str(msg))
|
|
|
|
# XXX Should exit more ceremoniously
|
|
|
|
return
|
|
|
|
now = time.ctime(time.time())
|
|
|
|
f.write("Title: %s\n" % title)
|
|
|
|
f.write("Last-Changed-Date: %s\n" % now)
|
|
|
|
f.write("Last-Changed-Author: %s\n" % author)
|
|
|
|
f.write("Last-Changed-Email: %s\n" % email)
|
|
|
|
f.write("Last-Changed-Remote-Host: %s\n" % remhost)
|
|
|
|
f.write("Last-Changed-Remote-Address: %s\n" % remaddr)
|
|
|
|
keys = headers.keys()
|
|
|
|
keys.sort()
|
|
|
|
keys.remove('title')
|
|
|
|
for key in keys:
|
|
|
|
if key[:13] != 'last-changed-':
|
|
|
|
f.write("%s: %s\n" % (string.capwords(key, '-'),
|
|
|
|
headers[key]))
|
|
|
|
f.write("\n")
|
|
|
|
f.write(text)
|
|
|
|
f.write("\n")
|
|
|
|
f.close()
|
|
|
|
|
|
|
|
tfn = tempfile.mktemp()
|
|
|
|
f = open(tfn, "w")
|
|
|
|
f.write("Last-Changed-Date: %s\n" % now)
|
|
|
|
f.write("Last-Changed-Author: %s\n" % author)
|
|
|
|
f.write("Last-Changed-Email: %s\n" % email)
|
|
|
|
f.write("Last-Changed-Remote-Host: %s\n" % remhost)
|
|
|
|
f.write("Last-Changed-Remote-Address: %s\n" % remaddr)
|
|
|
|
f.write("\n")
|
|
|
|
f.write(log)
|
|
|
|
f.write("\n")
|
|
|
|
f.close()
|
|
|
|
|
|
|
|
p = os.popen("""
|
|
|
|
/depot/gnu/plat/bin/rcs -l %s </dev/null 2>&1
|
|
|
|
/depot/gnu/plat/bin/ci -u %s <%s 2>&1
|
|
|
|
rm -f %s
|
|
|
|
""" % (name, name, tfn, tfn))
|
|
|
|
output = p.read()
|
|
|
|
sts = p.close()
|
|
|
|
if not sts:
|
|
|
|
print """
|
|
|
|
<H1>Python FAQ Entry Edited</H1>
|
|
|
|
<HR>
|
|
|
|
"""
|
|
|
|
self.show(name, title, text, 1)
|
|
|
|
if output:
|
|
|
|
print "<PRE>%s</PRE>" % cgi.escape(output)
|
|
|
|
else:
|
|
|
|
print """
|
|
|
|
<H1>Python FAQ Entry Commit Failed</H1>
|
|
|
|
Exit status 0x%04x
|
|
|
|
""" % sts
|
|
|
|
if output:
|
|
|
|
print "<PRE>%s</PRE>" % cgi.escape(output)
|
|
|
|
|
|
|
|
def showedit(self, name, headers, text):
|
|
|
|
title = headers['title']
|
|
|
|
print """
|
|
|
|
Title: <INPUT TYPE=text SIZE=70 NAME=title VALUE="%s"<BR>
|
|
|
|
<TEXTAREA COLS=80 ROWS=20 NAME=text>""" % title
|
|
|
|
print cgi.escape(string.strip(text))
|
|
|
|
print """</TEXTAREA>
|
|
|
|
<BR>
|
|
|
|
Please provide the following information for logging purposes:
|
|
|
|
<BR>
|
|
|
|
<CODE>Name : </CODE><INPUT TYPE=text SIZE=70 NAME=author VALUE="%s"<BR>
|
|
|
|
<CODE>Email: </CODE><INPUT TYPE=text SIZE=70 NAME=email VALUE="%s"<BR>
|
|
|
|
Log message (reason for the change):<BR>
|
|
|
|
<TEXTAREA COLS=80 ROWS=5 NAME=log>\n%s\n</TEXTAREA>
|
|
|
|
""" % (self.author, self.email, self.log)
|
1997-05-21 04:24:50 -03:00
|
|
|
|
|
|
|
def showheaders(self, headers):
|
|
|
|
print "<UL>"
|
|
|
|
keys = map(string.lower, headers.keys())
|
|
|
|
keys.sort()
|
|
|
|
for key in keys:
|
|
|
|
print "<LI><B>%s:</B> %s" % (string.capwords(key, '-'),
|
|
|
|
headers[key] or '')
|
|
|
|
print "</UL>"
|
|
|
|
|
|
|
|
def read(self, name):
|
|
|
|
import fnmatch, rfc822
|
|
|
|
if not fnmatch.fnmatch(name, NAMEPAT):
|
|
|
|
return None, None
|
|
|
|
f = open(name)
|
|
|
|
headers = rfc822.Message(f)
|
1997-05-21 13:52:18 -03:00
|
|
|
text = f.read()
|
1997-05-21 04:24:50 -03:00
|
|
|
f.close()
|
1997-05-21 13:52:18 -03:00
|
|
|
return headers, text
|
1997-05-21 04:24:50 -03:00
|
|
|
|
1997-05-21 13:52:18 -03:00
|
|
|
def show(self, name, title, text, edit=0):
|
1997-05-21 04:24:50 -03:00
|
|
|
# XXX Should put <A> tags around recognizable URLs
|
|
|
|
# XXX Should also turn "see section N" into hyperlinks
|
|
|
|
print "<H2>%s</H2>" % title
|
|
|
|
pre = 0
|
1997-05-21 13:52:18 -03:00
|
|
|
for line in string.split(text, '\n'):
|
1997-05-21 04:24:50 -03:00
|
|
|
if not string.strip(line):
|
|
|
|
if pre:
|
|
|
|
print '</PRE>'
|
|
|
|
pre = 0
|
|
|
|
else:
|
|
|
|
print '<P>'
|
|
|
|
else:
|
1997-05-21 13:52:18 -03:00
|
|
|
if line == string.lstrip(line): # I.e., no leading whitespace
|
1997-05-21 04:24:50 -03:00
|
|
|
if pre:
|
|
|
|
print '</PRE>'
|
|
|
|
pre = 0
|
|
|
|
else:
|
|
|
|
if not pre:
|
|
|
|
print '<PRE>'
|
|
|
|
pre = 1
|
|
|
|
print line
|
|
|
|
if pre:
|
|
|
|
print '</PRE>'
|
|
|
|
pre = 0
|
|
|
|
print '<P>'
|
|
|
|
if edit:
|
|
|
|
print '<A HREF="faq.py?req=edit&name=%s">Edit this entry</A>' %name
|
|
|
|
print '<P>'
|
|
|
|
print "<HR>"
|
|
|
|
|
|
|
|
|
|
|
|
print "Content-type: text/html\n"
|
1997-05-21 13:52:18 -03:00
|
|
|
dt = 0
|
1997-05-21 04:24:50 -03:00
|
|
|
try:
|
1997-05-21 13:52:18 -03:00
|
|
|
import time
|
|
|
|
t1 = time.time()
|
1997-05-21 04:24:50 -03:00
|
|
|
x = FAQServer()
|
|
|
|
x.main()
|
1997-05-21 13:52:18 -03:00
|
|
|
t2 = time.time()
|
|
|
|
dt = t2-t1
|
1997-05-21 04:24:50 -03:00
|
|
|
except:
|
|
|
|
print "<HR>Sorry, an error occurred"
|
|
|
|
cgi.print_exception()
|
1997-05-21 13:52:18 -03:00
|
|
|
print "<!-- dt = %s -->" % str(round(dt, 3))
|