Commit b4a8084b authored by Guido van Rossum's avatar Guido van Rossum

Added menu entry on deleting FAQ entries.

Added a generic error handler function.
Added cookie support to preserve author's name and email.
parent 5e931d4e
...@@ -6,7 +6,7 @@ this file as a string constant. ...@@ -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: ...@@ -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: ...@@ -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: ...@@ -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: ...@@ -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: ...@@ -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: ...@@ -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: ...@@ -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: ...@@ -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: ...@@ -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: ...@@ -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: ...@@ -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: ...@@ -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: ...@@ -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: ...@@ -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: ...@@ -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: ...@@ -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: ...@@ -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: ...@@ -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))
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment