mirror of
https://github.com/python/cpython.git
synced 2025-09-14 04:37:29 +00:00
Added menu entry on deleting FAQ entries.
Added a generic error handler function. Added cookie support to preserve author's name and email.
This commit is contained in:
parent
64099e955d
commit
af5be958e3
1 changed files with 89 additions and 33 deletions
|
@ -6,7 +6,7 @@ this file as a string constant.
|
||||||
|
|
||||||
XXX TO DO
|
XXX TO DO
|
||||||
|
|
||||||
- generic error handler
|
- next/prev/index links in do_show?
|
||||||
- should have files containing section headers
|
- should have files containing section headers
|
||||||
- customize rcs command pathnames
|
- customize rcs command pathnames
|
||||||
- recognize urls and email addresses and turn them into <A> tags
|
- recognize urls and email addresses and turn them into <A> tags
|
||||||
|
@ -60,11 +60,12 @@ class FAQServer:
|
||||||
item = form[key]
|
item = form[key]
|
||||||
except TypeError, msg:
|
except TypeError, msg:
|
||||||
raise KeyError, msg, sys.exc_traceback
|
raise KeyError, msg, sys.exc_traceback
|
||||||
value = self.form[key].value
|
|
||||||
setattr(self, key, value)
|
|
||||||
return value
|
|
||||||
except KeyError:
|
except KeyError:
|
||||||
return ''
|
return ''
|
||||||
|
value = self.form[key].value
|
||||||
|
value = string.strip(value)
|
||||||
|
setattr(self, key, value)
|
||||||
|
return value
|
||||||
|
|
||||||
def do_frontpage(self):
|
def do_frontpage(self):
|
||||||
self.prologue("Python FAQ (alpha) Front Page")
|
self.prologue("Python FAQ (alpha) Front Page")
|
||||||
|
@ -75,6 +76,7 @@ class FAQServer:
|
||||||
<LI><A HREF="faq.py?req=roulette">FAQ roulette</A>
|
<LI><A HREF="faq.py?req=roulette">FAQ roulette</A>
|
||||||
<LI><A HREF="faq.py?req=recent">Recently changed FAQ entries</A>
|
<LI><A HREF="faq.py?req=recent">Recently changed FAQ entries</A>
|
||||||
<LI><A HREF="faq.py?req=add">Add a new FAQ entry</A>
|
<LI><A HREF="faq.py?req=add">Add a new FAQ entry</A>
|
||||||
|
<LI><A HREF="faq.py?req=delete">Delete a FAQ entry</A>
|
||||||
</UL>
|
</UL>
|
||||||
|
|
||||||
<H2>Search the FAQ</H2>
|
<H2>Search the FAQ</H2>
|
||||||
|
@ -124,10 +126,12 @@ class FAQServer:
|
||||||
print "No FAQ entries?!?!"
|
print "No FAQ entries?!?!"
|
||||||
|
|
||||||
def do_show(self):
|
def do_show(self):
|
||||||
|
self.prologue("Python FAQ Entry")
|
||||||
|
print "<HR>"
|
||||||
name = self.name
|
name = self.name
|
||||||
headers, text = self.read(name)
|
headers, text = self.read(name)
|
||||||
if not headers:
|
if not headers:
|
||||||
print "Invalid file name", name
|
self.error("Invalid file name", name)
|
||||||
return
|
return
|
||||||
self.show(name, headers['title'], text)
|
self.show(name, headers['title'], text)
|
||||||
|
|
||||||
|
@ -201,12 +205,12 @@ class FAQServer:
|
||||||
print "No FAQ entries?!?!"
|
print "No FAQ entries?!?!"
|
||||||
|
|
||||||
def do_query(self):
|
def do_query(self):
|
||||||
import regex
|
|
||||||
self.prologue("Python FAQ Query Results")
|
|
||||||
query = self.query
|
query = self.query
|
||||||
if not query:
|
if not query:
|
||||||
print "No query string"
|
self.error("No query string")
|
||||||
return
|
return
|
||||||
|
import regex
|
||||||
|
self.prologue("Python FAQ Query Results")
|
||||||
p = regex.compile(query, regex.casefold)
|
p = regex.compile(query, regex.casefold)
|
||||||
names = os.listdir(os.curdir)
|
names = os.listdir(os.curdir)
|
||||||
names.sort()
|
names.sort()
|
||||||
|
@ -248,7 +252,7 @@ class FAQServer:
|
||||||
if n2 > max:
|
if n2 > max:
|
||||||
max = n2
|
max = n2
|
||||||
if not max:
|
if not max:
|
||||||
print "Can't add new sections yet."
|
self.error("Can't add new sections yet.")
|
||||||
return
|
return
|
||||||
num = max+1
|
num = max+1
|
||||||
name = "faq%02d.%03d.htp" % (nsec, num)
|
name = "faq%02d.%03d.htp" % (nsec, num)
|
||||||
|
@ -257,11 +261,23 @@ class FAQServer:
|
||||||
self.number = str(num)
|
self.number = str(num)
|
||||||
self.do_edit()
|
self.do_edit()
|
||||||
|
|
||||||
|
def do_delete(self):
|
||||||
|
self.prologue("How to delete a FAQ entry")
|
||||||
|
print """
|
||||||
|
At the moment, there's no direct way to delete entries.
|
||||||
|
This is because the entry numbers are also their
|
||||||
|
unique identifiers -- it's a bad idea to renumber entries.
|
||||||
|
<P>
|
||||||
|
If you really think an entry needs to be deleted,
|
||||||
|
change the title to "(deleted)" and make the body
|
||||||
|
empty (keep the entry number in the title though).
|
||||||
|
"""
|
||||||
|
|
||||||
def do_edit(self):
|
def do_edit(self):
|
||||||
name = self.name
|
name = self.name
|
||||||
headers, text = self.read(name)
|
headers, text = self.read(name)
|
||||||
if not headers:
|
if not headers:
|
||||||
print "Invalid file name", name
|
self.error("Invalid file name", name)
|
||||||
return
|
return
|
||||||
self.prologue("Python FAQ Edit Form")
|
self.prologue("Python FAQ Edit Form")
|
||||||
title = headers['title']
|
title = headers['title']
|
||||||
|
@ -293,8 +309,10 @@ class FAQServer:
|
||||||
title = self.title
|
title = self.title
|
||||||
headers, oldtext = self.read(name)
|
headers, oldtext = self.read(name)
|
||||||
if not headers:
|
if not headers:
|
||||||
print "Invalid file name", name
|
self.error("Invalid file name", name)
|
||||||
return
|
return
|
||||||
|
if self.author and '@' in self.email:
|
||||||
|
self.set_cookie(self.author, self.email)
|
||||||
self.prologue("Python FAQ Review Form")
|
self.prologue("Python FAQ Review Form")
|
||||||
print "<HR>"
|
print "<HR>"
|
||||||
self.show(name, title, text, edit=0)
|
self.show(name, title, text, edit=0)
|
||||||
|
@ -336,7 +354,7 @@ class FAQServer:
|
||||||
name = self.name
|
name = self.name
|
||||||
headers, text = self.read(name)
|
headers, text = self.read(name)
|
||||||
if not headers:
|
if not headers:
|
||||||
print "Invalid file name", name
|
self.error("Invalid file name", name)
|
||||||
return
|
return
|
||||||
print '<PRE>'
|
print '<PRE>'
|
||||||
sys.stdout.flush()
|
sys.stdout.flush()
|
||||||
|
@ -348,7 +366,7 @@ class FAQServer:
|
||||||
name = self.name
|
name = self.name
|
||||||
headers, text = self.read(name)
|
headers, text = self.read(name)
|
||||||
if not headers:
|
if not headers:
|
||||||
print "Invalid file name", name
|
self.error("Invalid file name", name)
|
||||||
return
|
return
|
||||||
print '<PRE>'
|
print '<PRE>'
|
||||||
sys.stdout.flush()
|
sys.stdout.flush()
|
||||||
|
@ -361,15 +379,15 @@ class FAQServer:
|
||||||
|
|
||||||
headers, oldtext = self.read(name)
|
headers, oldtext = self.read(name)
|
||||||
if not headers:
|
if not headers:
|
||||||
print "Invalid file name", name
|
self.error("Invalid file name", name)
|
||||||
return
|
return
|
||||||
version = self.version
|
version = self.version
|
||||||
curversion = self.getversion(name)
|
curversion = self.getversion(name)
|
||||||
if version != curversion:
|
if version != curversion:
|
||||||
print "Version conflict."
|
self.error("Version conflict.",
|
||||||
print "You edited version %s but current version is %s." % (
|
"You edited version %s but current version is %s." % (
|
||||||
version, curversion)
|
version, curversion),
|
||||||
print '<A HREF="faq.py?req=show&name=%s">Reload.</A>' % name
|
'<A HREF="faq.py?req=show&name=%s">Reload.</A>' % name)
|
||||||
return
|
return
|
||||||
text = self.text
|
text = self.text
|
||||||
title = self.title
|
title = self.title
|
||||||
|
@ -386,13 +404,11 @@ class FAQServer:
|
||||||
text = string.strip(text)
|
text = string.strip(text)
|
||||||
oldtext = string.strip(oldtext)
|
oldtext = string.strip(oldtext)
|
||||||
if text == oldtext and title == oldtitle:
|
if text == oldtext and title == oldtitle:
|
||||||
print "No changes."
|
self.error("No changes.")
|
||||||
# XXX Should exit more ceremoniously
|
|
||||||
return
|
return
|
||||||
# Check that the FAQ entry number didn't change
|
# Check that the FAQ entry number didn't change
|
||||||
if string.split(title)[:1] != string.split(oldtitle)[:1]:
|
if string.split(title)[:1] != string.split(oldtitle)[:1]:
|
||||||
print "Don't change the FAQ entry number please."
|
self.error("Don't change the FAQ entry number please.")
|
||||||
# XXX Should exit more ceremoniously
|
|
||||||
return
|
return
|
||||||
remhost = os.environ["REMOTE_HOST"]
|
remhost = os.environ["REMOTE_HOST"]
|
||||||
remaddr = os.environ["REMOTE_ADDR"]
|
remaddr = os.environ["REMOTE_ADDR"]
|
||||||
|
@ -411,8 +427,7 @@ class FAQServer:
|
||||||
try:
|
try:
|
||||||
f = open(name, "w")
|
f = open(name, "w")
|
||||||
except IOError, msg:
|
except IOError, msg:
|
||||||
print "Can't open", name, "for writing:", cgi.escape(str(msg))
|
self.error("Can't open", name, "for writing:", cgi.escape(str(msg)))
|
||||||
# XXX Should exit more ceremoniously
|
|
||||||
return
|
return
|
||||||
now = time.ctime(time.time())
|
now = time.ctime(time.time())
|
||||||
f.write("Title: %s\n" % title)
|
f.write("Title: %s\n" % title)
|
||||||
|
@ -453,22 +468,56 @@ class FAQServer:
|
||||||
output = p.read()
|
output = p.read()
|
||||||
sts = p.close()
|
sts = p.close()
|
||||||
if not sts:
|
if not sts:
|
||||||
|
self.set_cookie(author, email)
|
||||||
self.prologue("Python FAQ Entry Edited")
|
self.prologue("Python FAQ Entry Edited")
|
||||||
print "<HR>"
|
print "<HR>"
|
||||||
self.show(name, title, text)
|
self.show(name, title, text)
|
||||||
if output:
|
if output:
|
||||||
print "<PRE>%s</PRE>" % cgi.escape(output)
|
print "<PRE>%s</PRE>" % cgi.escape(output)
|
||||||
else:
|
else:
|
||||||
print """
|
self.error("Python FAQ Entry Commit Failed",
|
||||||
<H1>Python FAQ Entry Commit Failed</H1>
|
"Exit status 0x%04x" % sts)
|
||||||
Exit status 0x%04x
|
|
||||||
""" % sts
|
|
||||||
if output:
|
if output:
|
||||||
print "<PRE>%s</PRE>" % cgi.escape(output)
|
print "<PRE>%s</PRE>" % cgi.escape(output)
|
||||||
print '<HR>'
|
print '<HR>'
|
||||||
print '<A HREF="faq.py?req=show&name=%s">Reload this entry.</A>' % name
|
print '<A HREF="faq.py?req=show&name=%s">Reload this entry.</A>' % name
|
||||||
|
|
||||||
|
def set_cookie(self, author, email):
|
||||||
|
name = "Python-FAQ-ID"
|
||||||
|
value = "%s;%s" % (author, email)
|
||||||
|
import urllib
|
||||||
|
value = urllib.quote(value)
|
||||||
|
print "Set-Cookie: %s=%s; path=/cgi-bin/;" % (name, value),
|
||||||
|
print "domain=%s;" % os.environ['HTTP_HOST'],
|
||||||
|
print "expires=Sat, 01-Jan-2000 00:00:00 GMT"
|
||||||
|
|
||||||
|
def get_cookie(self):
|
||||||
|
if not os.environ.has_key('HTTP_COOKIE'):
|
||||||
|
return "", ""
|
||||||
|
raw = os.environ['HTTP_COOKIE']
|
||||||
|
words = string.split(raw, ';')
|
||||||
|
cookies = {}
|
||||||
|
for word in words:
|
||||||
|
i = string.find(word, '=')
|
||||||
|
if i >= 0:
|
||||||
|
key, value = word[:i], word[i+1:]
|
||||||
|
cookies[key] = value
|
||||||
|
if not cookies.has_key('Python-FAQ-ID'):
|
||||||
|
return "", ""
|
||||||
|
value = cookies['Python-FAQ-ID']
|
||||||
|
import urllib
|
||||||
|
value = urllib.unquote(value)
|
||||||
|
i = string.rfind(value, ';')
|
||||||
|
author, email = value[:i], value[i+1:]
|
||||||
|
return author, email
|
||||||
|
|
||||||
def showedit(self, name, title, text):
|
def showedit(self, name, title, text):
|
||||||
|
author = self.author
|
||||||
|
email = self.email
|
||||||
|
if not author or not email:
|
||||||
|
a, e = self.get_cookie()
|
||||||
|
author = author or a
|
||||||
|
email = email or e
|
||||||
print """
|
print """
|
||||||
Title: <INPUT TYPE=text SIZE=70 NAME=title VALUE="%s"><BR>
|
Title: <INPUT TYPE=text SIZE=70 NAME=title VALUE="%s"><BR>
|
||||||
<TEXTAREA COLS=80 ROWS=20 NAME=text>""" % title
|
<TEXTAREA COLS=80 ROWS=20 NAME=text>""" % title
|
||||||
|
@ -482,8 +531,8 @@ class FAQServer:
|
||||||
<CODE>Email: </CODE><INPUT TYPE=text SIZE=40 NAME=email VALUE="%s">
|
<CODE>Email: </CODE><INPUT TYPE=text SIZE=40 NAME=email VALUE="%s">
|
||||||
<BR>
|
<BR>
|
||||||
Log message (reason for the change):<BR>
|
Log message (reason for the change):<BR>
|
||||||
<TEXTAREA COLS=80 ROWS=5 NAME=log>\n%s\n</TEXTAREA>
|
<TEXTAREA COLS=80 ROWS=5 NAME=log>%s\n</TEXTAREA>
|
||||||
""" % (self.author, self.email, self.log)
|
""" % (author, email, self.log)
|
||||||
|
|
||||||
def showheaders(self, headers):
|
def showheaders(self, headers):
|
||||||
print "<UL>"
|
print "<UL>"
|
||||||
|
@ -577,7 +626,7 @@ class FAQServer:
|
||||||
|
|
||||||
def prologue(self, title):
|
def prologue(self, title):
|
||||||
title = cgi.escape(title)
|
title = cgi.escape(title)
|
||||||
print '''\
|
print '''
|
||||||
<HTML>
|
<HTML>
|
||||||
<HEAD>
|
<HEAD>
|
||||||
<TITLE>%s</TITLE>
|
<TITLE>%s</TITLE>
|
||||||
|
@ -590,6 +639,13 @@ class FAQServer:
|
||||||
<H1>%s</H1>
|
<H1>%s</H1>
|
||||||
''' % (title, title)
|
''' % (title, title)
|
||||||
|
|
||||||
|
def error(self, *messages):
|
||||||
|
self.prologue("Python FAQ error")
|
||||||
|
print "Sorry, an error occurred:<BR>"
|
||||||
|
for message in messages:
|
||||||
|
print message,
|
||||||
|
print
|
||||||
|
|
||||||
def epilogue(self):
|
def epilogue(self):
|
||||||
print '''
|
print '''
|
||||||
<P>
|
<P>
|
||||||
|
@ -601,7 +657,7 @@ class FAQServer:
|
||||||
</HTML>
|
</HTML>
|
||||||
'''
|
'''
|
||||||
|
|
||||||
print "Content-type: text/html\n"
|
print "Content-type: text/html"
|
||||||
dt = 0
|
dt = 0
|
||||||
try:
|
try:
|
||||||
import time
|
import time
|
||||||
|
@ -612,7 +668,7 @@ try:
|
||||||
t2 = time.time()
|
t2 = time.time()
|
||||||
dt = t2-t1
|
dt = t2-t1
|
||||||
except:
|
except:
|
||||||
print "<HR>Sorry, an error occurred"
|
print "\n<HR>Sorry, an error occurred"
|
||||||
cgi.print_exception()
|
cgi.print_exception()
|
||||||
print "<P>(running time = %s seconds)" % str(round(dt, 3))
|
print "<P>(running time = %s seconds)" % str(round(dt, 3))
|
||||||
|
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue