]> granicus.if.org Git - python/commitdiff
Initial revision
authorGuido van Rossum <guido@python.org>
Mon, 26 May 1997 00:07:18 +0000 (00:07 +0000)
committerGuido van Rossum <guido@python.org>
Mon, 26 May 1997 00:07:18 +0000 (00:07 +0000)
Tools/faqwiz/faqconf.py [new file with mode: 0644]
Tools/faqwiz/faqwiz.py [new file with mode: 0644]

diff --git a/Tools/faqwiz/faqconf.py b/Tools/faqwiz/faqconf.py
new file mode 100644 (file)
index 0000000..de82e22
--- /dev/null
@@ -0,0 +1,382 @@
+# Miscellaneous customization constants
+PASSWORD = "Spam"                      # Edit password.  Change this!
+FAQCGI = 'faqw.py'                     # Relative URL of the FAQ cgi script
+FAQNAME = "Python FAQ"                 # Name of the FAQ
+OWNERNAME = "GvR"                      # Name for feedback
+OWNEREMAIL = "guido@python.org"                # Email for feedback
+HOMEURL = "http://www.python.org"      # Related home page
+HOMENAME = "Python home"               # Name of related home page
+MAXHITS = 10                           # Max #hits to be shown directly
+COOKIE_NAME = "Python-FAQ-Wizard"      # Name used for Netscape cookie
+COOKIE_LIFETIME = 4 *7 * 24 * 3600     # Cookie expiration in seconds
+
+# RCS commands
+RCSBINDIR = "/depot/gnu/plat/bin/"     # Directory containing RCS commands
+SH_RLOG = RCSBINDIR + "rlog %(file)s </dev/null 2>&1"
+SH_RLOG_H = RCSBINDIR + "rlog -h %(file)s </dev/null 2>&1"
+SH_RDIFF = RCSBINDIR + "rcsdiff -r%(prev)s -r%(rev)s %(file)s </dev/null 2>&1"
+SH_LOCK = RCSBINDIR + "rcs -l %(file)s </dev/null 2>&1"
+SH_CHECKIN =  RCSBINDIR + "ci -u %(file)s <%(tfn)s 2>&1"
+
+# Titles for various output pages
+T_HOME = FAQNAME + " Wizard 0.2 (alpha)"
+T_ERROR = "Sorry, an error occurred"
+T_ROULETTE = FAQNAME + " Roulette"
+T_ALL = "The Whole " + FAQNAME
+T_INDEX = FAQNAME + " Index"
+T_SEARCH = FAQNAME + " Search Results"
+T_RECENT = "Recently Changed %s Entries" % FAQNAME
+T_SHOW = FAQNAME + " Entry"
+T_LOG = "RCS log for %s entry" % FAQNAME
+T_DIFF = "RCS diff for %s entry" % FAQNAME
+T_ADD = "How to add an entry to the " + FAQNAME
+T_DELETE = "How to delete an entry from the " + FAQNAME
+T_EDIT = FAQNAME + " Edit Wizard"
+T_REVIEW = T_EDIT + " - Review Changes"
+T_COMMITTED = T_EDIT + " - Changes Committed"
+T_COMMITFAILED = T_EDIT + " - Commit Failed"
+T_CANTCOMMIT = T_EDIT + " - Commit Rejected"
+T_HELP = T_EDIT + " - Help"
+
+# Titles of FAQ sections
+SECTION_TITLES = {
+    1: "General information and availability",
+    2: "Python in the real world",
+    3: "Building Python and Other Known Bugs",
+    4: "Programming in Python",
+    5: "Extending Python",
+    6: "Python's design",
+    7: "Using Python on non-UNIX platforms",
+}
+
+# Generic prologue and epilogue
+
+PROLOGUE = '''
+<HTML>
+<HEAD>
+<TITLE>%(title)s</TITLE>
+</HEAD>
+
+<BODY BACKGROUND="http://www.python.org/pics/RedShort.gif"
+      BGCOLOR="#FFFFFF"
+      TEXT="#000000"
+      LINK="#AA0000"
+      VLINK="#906A6A">
+<H1>%(title)s</H1>
+'''
+
+EPILOGUE = '''
+<HR>
+<A HREF="%(HOMEURL)s">%(HOMENAME)s</A> /
+<A HREF="%(FAQCGI)s?req=home">%(FAQNAME)s Wizard</A> /
+Feedback to <A HREF="mailto:%(OWNEREMAIL)s">%(OWNERNAME)s</A>
+
+</BODY>
+</HTML>
+'''
+
+# Home page
+
+HOME = """
+<FORM ACTION="%(FAQCGI)s">
+    <INPUT TYPE=text NAME=query>
+    <INPUT TYPE=submit VALUE="Search"><BR>
+    (Case insensitive regular expressions.)
+    <INPUT TYPE=hidden NAME=req VALUE=search>
+</FORM>
+
+<UL>
+<LI><A HREF="%(FAQCGI)s?req=index">FAQ index</A>
+<LI><A HREF="%(FAQCGI)s?req=all">The whole FAQ</A>
+<LI><A HREF="%(FAQCGI)s?req=recent">Recently changed FAQ entries</A>
+<LI><A HREF="%(FAQCGI)s?req=roulette">FAQ roulette</A>
+</UL>
+"""
+
+# Index formatting
+
+INDEX_SECTION = """
+<P>
+<HR>
+<H2>%(sec)d. %(title)s</H2>
+<UL>
+"""
+
+INDEX_ENDSECTION = """
+</UL>
+"""
+
+INDEX_ENTRY = """\
+<LI><A HREF="%(FAQCGI)s?req=show&file=%(file)s">%(title)s</A><BR>
+"""
+
+# Entry formatting
+
+ENTRY_FOOTER = """
+<A HREF="%(FAQCGI)s?req=edit&file=%(file)s">Edit this entry</A> /
+<A HREF="%(FAQCGI)s?req=log&file=%(file)s">Log info</A>
+"""
+
+ENTRY_LOGINFO = """
+/ Last changed on %(last_changed_date)s by
+<A HREF="mailto:%(last_changed_email)s">%(last_changed_author)s</A>
+"""
+
+# Search
+
+NO_HITS = """
+No hits.
+"""
+
+ONE_HIT = """
+Your search matched the following entry:
+"""
+
+FEW_HITS = """
+Your search matched the following %(count)d entries:
+"""
+
+MANY_HITS = """
+Your search matched more than %(MAXHITS)d entries.
+The %(count)d matching entries are presented here ordered by section:
+"""
+
+# RCS log and diff
+
+LOG = """
+Click on a revision line to see the diff between that revision and the
+previous one.
+"""
+
+DIFFLINK = """\
+<A HREF="%(FAQCGI)s?req=diff&file=%(file)s&rev=%(rev)s">%(line)s</A>
+"""
+
+# Recently changed entries
+
+NO_RECENT = """
+<HR>
+No %(FAQNAME)s entries were changed in the last %(period)s.
+"""
+
+ONE_RECENT = """
+<HR>
+View entries changed in the last:
+<UL>
+<LI><A HREF="%(FAQCGI)s?req=recent&days=1">24 hours</A>
+<LI><A HREF="%(FAQCGI)s?req=recent&days=2">2 days</A>
+<LI><A HREF="%(FAQCGI)s?req=recent&days=3">3 days</A>
+<LI><A HREF="%(FAQCGI)s?req=recent&days=7">week</A>
+<LI><A HREF="%(FAQCGI)s?req=recent&days=28">4 weeks</A>
+<LI><A HREF="%(FAQCGI)s?req=recent&days=365250">millennium</A>
+</UL>
+The following %(FAQNAME)s entry was changed in the last %(period)s:
+"""
+
+SOME_RECENT = """
+<HR>
+View entries changed in the last:
+<UL>
+<LI><A HREF="%(FAQCGI)s?req=recent&days=1">24 hours</A>
+<LI><A HREF="%(FAQCGI)s?req=recent&days=2">2 days</A>
+<LI><A HREF="%(FAQCGI)s?req=recent&days=3">3 days</A>
+<LI><A HREF="%(FAQCGI)s?req=recent&days=7">week</A>
+<LI><A HREF="%(FAQCGI)s?req=recent&days=28">4 weeks</A>
+<LI><A HREF="%(FAQCGI)s?req=recent&days=365250">millennium</A>
+</UL>
+The following %(count)d %(FAQNAME)s entries were changed
+in the last %(period)s, most recently changed shown first:
+"""
+
+TAIL_RECENT = """
+<HR>
+View entries changed in the last:
+<UL>
+<LI><A HREF="%(FAQCGI)s?req=recent&days=1">24 hours</A>
+<LI><A HREF="%(FAQCGI)s?req=recent&days=2">2 days</A>
+<LI><A HREF="%(FAQCGI)s?req=recent&days=3">3 days</A>
+<LI><A HREF="%(FAQCGI)s?req=recent&days=7">week</A>
+<LI><A HREF="%(FAQCGI)s?req=recent&days=28">4 weeks</A>
+<LI><A HREF="%(FAQCGI)s?req=recent&days=365250">millennium</A>
+</UL>
+"""
+
+# Last changed banner on "all" (strftime format)
+LAST_CHANGED = "Last changed on %c %Z"
+
+# "Compat" command prologue (no <BODY> tag)
+COMPAT = """
+<H1>The whole %(FAQNAME)s</H1>
+"""
+
+# Editing
+
+EDITHEAD = """
+<A HREF="%(FAQCGI)s?req=help">Click for Help</A>
+"""
+
+REVIEWHEAD = EDITHEAD
+
+
+EDITFORM1 = """
+<FORM ACTION="%(FAQCGI)s" METHOD=POST>
+<INPUT TYPE=hidden NAME=req VALUE=review>
+<INPUT TYPE=hidden NAME=file VALUE=%(file)s>
+<INPUT TYPE=hidden NAME=editversion VALUE=%(editversion)s>
+<HR>
+"""
+
+EDITFORM2 = """
+Title: <INPUT TYPE=text SIZE=70 NAME=title VALUE="%(title)s"><BR>
+<TEXTAREA COLS=72 ROWS=20 NAME=body>%(body)s
+</TEXTAREA><BR>
+Log message (reason for the change):<BR>
+<TEXTAREA COLS=72 ROWS=5 NAME=log>%(log)s
+</TEXTAREA><BR>
+Please provide the following information for logging purposes:
+<TABLE FRAME=none COLS=2>
+    <TR>
+       <TD>Name:
+       <TD><INPUT TYPE=text SIZE=40 NAME=author VALUE="%(author)s">
+    <TR>
+       <TD>Email:
+       <TD><INPUT TYPE=text SIZE=40 NAME=email VALUE="%(email)s">
+    <TR>
+       <TD>Password:
+       <TD><INPUT TYPE=password SIZE=20 NAME=password VALUE="%(password)s">
+</TABLE>
+
+<INPUT TYPE=submit NAME=review VALUE="Preview Edit">
+Click this button to preview your changes.
+"""
+
+EDITFORM3 = """
+</FORM>
+"""
+
+COMMIT = """
+<INPUT TYPE=submit NAME=commit VALUE="Commit">
+Click this button to commit your changes.
+<HR>
+"""
+
+NOCOMMIT = """
+You can't commit your changes unless you enter a log message, your
+name, email addres, and the correct password in the form below.
+<HR>
+"""
+
+CANTCOMMIT_HEAD = """
+Some required information is missing:
+<UL>
+"""
+NEED_PASSWD = "<LI>You must provide the correct passwd.\n"
+NEED_AUTHOR = "<LI>You must enter your name.\n"
+NEED_EMAIL = "<LI>You must enter your email address.\n"
+NEED_LOG = "<LI>You must enter a log message.\n"
+CANTCOMMIT_TAIL = """
+</UL>
+Please use your browser's Back command to correct the form and commit
+again.
+"""
+
+VERSIONCONFLICT = """
+<P>
+You edited version %(editversion)s but the current version is %(version)s.
+<P>
+The two most common causes of this problem are:
+<UL>
+<LI>After committing a change, you went back in your browser,
+    edited the entry some more, and clicked Commit again.
+<LI>Someone else started editing the same entry and committed
+    before you did.
+</UL>
+<P>
+<A HREF="%(FAQCGI)s?req=show&file=%(file)s">Click here to reload the entry
+and try again.</A>
+<P>
+"""
+
+CANTWRITE = """
+Can't write file %(file)s (%(why)s).
+"""
+
+FILEHEADER = """\
+Title: %(title)s
+Last-Changed-Date: %(date)s
+Last-Changed-Author: %(author)s
+Last-Changed-Email: %(email)s
+Last-Changed-Remote-Host: %(REMOTE_HOST)s
+Last-Changed-Remote-Address: %(REMOTE_ADDR)s
+"""
+
+LOGHEADER = """\
+Last-Changed-Date: %(date)s
+Last-Changed-Author: %(author)s
+Last-Changed-Email: %(email)s
+Last-Changed-Remote-Host: %(REMOTE_HOST)s
+Last-Changed-Remote-Address: %(REMOTE_ADDR)s
+
+%(log)s
+"""
+
+COMMITTED = """
+Your changes have been committed.
+"""
+
+COMMITFAILED = """
+Exit status %(sts)04x.
+"""
+
+HELP = """
+Using the %(FAQNAME)s Edit Wizard speaks mostly for itself.  Here are
+some answers to questions you are likely to ask:
+
+<P><HR>
+
+<H2>I can review an entry but I can't commit it.</H2>
+
+The commit button only appears if the following conditions are met:
+
+<UL>
+
+<LI>The Name field is not empty.
+
+<LI>The Email field contains at least an @ character.
+
+<LI>The Log message box is not empty.
+
+<LI>The Password field contains the proper password.
+
+</UL>
+
+<P><HR>
+
+<H2>What is the password?</H2>
+
+At the moment, only PSA members will be told the password.  This is a
+good time to join the PSA!  See <A
+HREF="http://www.python.org/psa/">the PSA home page</A>.
+
+<P><HR>
+
+<H2>Can I use HTML in the FAQ entry?</H2>
+
+No, but if you include a URL or an email address in the text it will
+automatigally become an anchor of the right type.  Also, *word*
+is made italic (but only for single alphabetic words).
+
+<P><HR>
+
+<H2>How do I delineate paragraphs?</H2>
+
+Use blank lines to separate paragraphs.
+
+<P><HR>
+
+<H2>How do I enter example text?</H2>
+
+Any line that begins with a space or tab is assumed to be part of
+literal text.  Blocks of literal text delineated by blank lines are
+placed inside &lt;PRE&gt;...&lt;/PRE&gt;.
+"""
diff --git a/Tools/faqwiz/faqwiz.py b/Tools/faqwiz/faqwiz.py
new file mode 100644 (file)
index 0000000..47aa3b7
--- /dev/null
@@ -0,0 +1,666 @@
+import sys, string, time, os, stat, regex, cgi, faqconf
+
+from cgi import escape
+
+class FileError:
+    def __init__(self, file):
+       self.file = file
+
+class InvalidFile(FileError):
+    pass
+
+class NoSuchFile(FileError):
+    def __init__(self, file, why=None):
+       FileError.__init__(self, file)
+       self.why = why
+
+def escapeq(s):
+    s = escape(s)
+    import regsub
+    s = regsub.gsub('"', '&quot;', s)
+    return s
+
+def interpolate(format, entry={}, kwdict={}, **kw):
+    s = format % MDict(kw, entry, kwdict, faqconf.__dict__)
+    return s
+
+def emit(format, entry={}, kwdict={}, file=sys.stdout, **kw):
+    s = format % MDict(kw, entry, kwdict, faqconf.__dict__)
+    file.write(s)
+
+translate_prog = None
+
+def translate(text):
+    global translate_prog
+    if not translate_prog:
+       import regex
+       url = '\(http\|ftp\)://[^ \t\r\n]*'
+       email = '\<[-a-zA-Z0-9._]+@[-a-zA-Z0-9._]+'
+       translate_prog = prog = regex.compile(url + "\|" + email)
+    else:
+       prog = translate_prog
+    i = 0
+    list = []
+    while 1:
+       j = prog.search(text, i)
+       if j < 0:
+           break
+       list.append(cgi.escape(text[i:j]))
+       i = j
+       url = prog.group(0)
+       while url[-1] in ");:,.?'\"":
+           url = url[:-1]
+       url = escape(url)
+       if ':' in url:
+           repl = '<A HREF="%s">%s</A>' % (url, url)
+       else:
+           repl = '<A HREF="mailto:%s">&lt;%s&gt;</A>' % (url, url)
+       list.append(repl)
+       i = i + len(url)
+    j = len(text)
+    list.append(cgi.escape(text[i:j]))
+    return string.join(list, '')
+
+emphasize_prog = None
+
+def emphasize(line):
+    global emphasize_prog
+    import regsub
+    if not emphasize_prog:
+       import regex
+       pat = "\*\([a-zA-Z]+\)\*"
+       emphasize_prog = prog = regex.compile(pat)
+    else:
+       prog = emphasize_prog
+    return regsub.gsub(prog, "<I>\\1</I>", line)
+
+def load_cookies():
+    if not os.environ.has_key('HTTP_COOKIE'):
+       return {}
+    raw = os.environ['HTTP_COOKIE']
+    words = map(string.strip, 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
+    return cookies
+
+def load_my_cookie():
+    cookies = load_cookies()
+    try:
+       value = cookies[faqconf.COOKIE_NAME]
+    except KeyError:
+       return {}
+    import urllib
+    value = urllib.unquote(value)
+    words = string.split(value, '/')
+    while len(words) < 3:
+       words.append('')
+    author = string.join(words[:-2], '/')
+    email = words[-2]
+    password = words[-1]
+    return {'author': author,
+           'email': email,
+           'password': password}
+
+class MDict:
+
+    def __init__(self, *d):
+       self.__d = d
+
+    def __getitem__(self, key):
+       for d in self.__d:
+           try:
+               value = d[key]
+               if value:
+                   return value
+           except KeyError:
+               pass
+       return ""
+
+class UserInput:
+
+    def __init__(self):
+       self.__form = cgi.FieldStorage()
+
+    def __getattr__(self, name):
+       if name[0] == '_':
+           raise AttributeError
+       try:
+           value = self.__form[name].value
+       except (TypeError, KeyError):
+           value = ''
+       else:
+           value = string.strip(value)
+       setattr(self, name, value)
+       return value
+
+    def __getitem__(self, key):
+       return getattr(self, key)
+
+class FaqFormatter:
+
+    def __init__(self, entry):
+       self.entry = entry
+
+    def show(self, edit=1):
+       entry = self.entry
+       print "<HR>"
+       print "<H2>%s</H2>" % escape(entry.title)
+       pre = 0
+       for line in string.split(entry.body, '\n'):
+           if not string.strip(line):
+               if pre:
+                   print '</PRE>'
+                   pre = 0
+               else:
+                   print '<P>'
+           else:
+               if line[0] not in string.whitespace:
+                   if pre:
+                       print '</PRE>'
+                       pre = 0
+               else:
+                   if not pre:
+                       print '<PRE>'
+                       pre = 1
+               if '/' in line or '@' in line:
+                   line = translate(line)
+               elif '<' in line or '&' in line:
+                   line = escape(line)
+               if not pre and '*' in line:
+                   line = emphasize(line)
+               print line
+       if pre:
+           print '</PRE>'
+           pre = 0
+       if edit:
+           print '<P>'
+           emit(faqconf.ENTRY_FOOTER, self.entry)
+           if self.entry.last_changed_date:
+               emit(faqconf.ENTRY_LOGINFO, self.entry)
+       print '<P>'
+
+class FaqEntry:
+
+    formatterclass = FaqFormatter
+
+    def __init__(self, fp, file, sec_num):
+       import rfc822
+       self.file = file
+       self.sec, self.num = sec_num
+       self.__headers = rfc822.Message(fp)
+       self.body = string.strip(fp.read())
+
+    def __getattr__(self, name):
+       if name[0] == '_':
+           raise AttributeError
+       key = string.join(string.split(name, '_'), '-')
+       try:
+           value = self.__headers[key]
+       except KeyError:
+           value = ''
+       setattr(self, name, value)
+       return value
+
+    def __getitem__(self, key):
+       return getattr(self, key)
+
+    def show(self, edit=1):
+       self.formatterclass(self).show(edit=edit)
+
+    def load_version(self):
+       command = interpolate(faqconf.SH_RLOG_H, self)
+       p = os.popen(command)
+       version = ""
+       while 1:
+           line = p.readline()
+           if not line:
+               break
+           if line[:5] == 'head:':
+               version = string.strip(line[5:])
+       p.close()
+       self.version = version
+
+class FaqDir:
+
+    entryclass = FaqEntry
+
+    __okprog = regex.compile('^faq\([0-9][0-9]\)\.\([0-9][0-9][0-9]\)\.htp$')
+
+    def __init__(self, dir=os.curdir):
+       self.__dir = dir
+       self.__files = None
+
+    def __fill(self):
+       if self.__files is not None:
+           return
+       self.__files = files = []
+       okprog = self.__okprog
+       for file in os.listdir(self.__dir):
+           if okprog.match(file) >= 0:
+               files.append(file)
+       files.sort()
+
+    def good(self, file):
+       return self.__okprog.match(file) >= 0
+
+    def parse(self, file):
+       if not self.good(file):
+           return None
+       sec, num = self.__okprog.group(1, 2)
+       return string.atoi(sec), string.atoi(num)
+
+    def roulette(self):
+       self.__fill()
+       import whrandom
+       return whrandom.choice(self.__files)
+
+    def list(self):
+       # XXX Caller shouldn't modify result
+       self.__fill()
+       return self.__files
+
+    def open(self, file):
+       sec_num = self.parse(file)
+       if not sec_num:
+           raise InvalidFile(file)
+       try:
+           fp = open(file)
+       except IOError, msg:
+           raise NoSuchFile(file, msg)
+       try:
+           return self.entryclass(fp, file, sec_num)
+       finally:
+           fp.close()
+
+    def show(self, file, edit=1):
+       self.open(file).show(edit=edit)
+
+    def new(self, sec):
+       XXX
+
+class FaqWizard:
+
+    def __init__(self):
+       self.ui = UserInput()
+       self.dir = FaqDir()
+
+    def go(self):
+       print "Content-type: text/html"
+       req = self.ui.req or "home"
+       mname = 'do_%s' % req
+       try:
+           meth = getattr(self, mname)
+       except AttributeError:
+           self.error("Bad request %s" % `req`)
+       else:
+           try:
+               meth()
+           except InvalidFile, exc:
+               self.error("Invalid entry file name %s" % exc.file)
+           except NoSuchFile, exc:
+               self.error("No entry with file name %s" % exc.file)
+       self.epilogue()
+
+    def error(self, message, **kw):
+       self.prologue(faqconf.T_ERROR)
+       apply(emit, (message,), kw)
+
+    def prologue(self, title, entry=None, **kw):
+       emit(faqconf.PROLOGUE, entry, kwdict=kw, title=escape(title))
+
+    def epilogue(self):
+       emit(faqconf.EPILOGUE)
+
+    def do_home(self):
+       self.prologue(faqconf.T_HOME)
+       emit(faqconf.HOME)
+
+    def do_search(self):
+       query = self.ui.query
+       if not query:
+           self.error("No query string")
+           return
+       self.prologue(faqconf.T_SEARCH)
+       if self.ui.casefold == "no":
+           p = regex.compile(query)
+       else:
+           p = regex.compile(query, regex.casefold)
+       hits = []
+       for file in self.dir.list():
+           try:
+               entry = self.dir.open(file)
+           except FileError:
+               constants
+           if p.search(entry.title) >= 0 or p.search(entry.body) >= 0:
+               hits.append(file)
+       if not hits:
+           emit(faqconf.NO_HITS, count=0)
+       elif len(hits) <= faqconf.MAXHITS:
+           if len(hits) == 1:
+               emit(faqconf.ONE_HIT, count=1)
+           else:
+               emit(faqconf.FEW_HITS, count=len(hits))
+           self.format_all(hits)
+       else:
+           emit(faqconf.MANY_HITS, count=len(hits))
+           self.format_index(hits)
+
+    def do_all(self):
+       self.prologue(faqconf.T_ALL)
+       files = self.dir.list()
+       self.last_changed(files)
+       self.format_all(files)
+
+    def do_compat(self):
+       files = self.dir.list()
+       emit(faqconf.COMPAT)
+       self.last_changed(files)
+       self.format_all(files, edit=0)
+       sys.exit(0)
+
+    def last_changed(self, files):
+       latest = 0
+       for file in files:
+           try:
+               st = os.stat(file)
+           except os.error:
+               continue
+           mtime = st[stat.ST_MTIME]
+           if mtime > latest:
+               latest = mtime
+       print time.strftime(faqconf.LAST_CHANGED,
+                           time.localtime(time.time()))
+
+    def format_all(self, files, edit=1):
+       for file in files:
+           self.dir.show(file, edit=edit)
+
+    def do_index(self):
+       self.prologue(faqconf.T_INDEX)
+       self.format_index(self.dir.list())
+
+    def format_index(self, files):
+       sec = 0
+       for file in files:
+           try:
+               entry = self.dir.open(file)
+           except NoSuchFile:
+               continue
+           if entry.sec != sec:
+               if sec:
+                   emit(faqconf.INDEX_ENDSECTION, sec=sec)
+               sec = entry.sec
+               emit(faqconf.INDEX_SECTION,
+                           sec=sec,
+                           title=faqconf.SECTION_TITLES[sec])
+           emit(faqconf.INDEX_ENTRY, entry)
+       if sec:
+           emit(faqconf.INDEX_ENDSECTION, sec=sec)
+
+    def do_recent(self):
+       if not self.ui.days:
+           days = 1
+       else:
+           days = string.atof(self.ui.days)
+       now = time.time()
+       try:
+           cutoff = now - days * 24 * 3600
+       except OverflowError:
+           cutoff = 0
+       list = []
+       for file in self.dir.list():
+           try:
+               st = os.stat(file)
+           except os.error:
+               continue
+           mtime = st[stat.ST_MTIME]
+           if mtime >= cutoff:
+               list.append((mtime, file))
+       list.sort()
+       list.reverse()
+       self.prologue(faqconf.T_RECENT)
+       if days <= 1:
+           period = "%.2g hours" % (days*24)
+       else:
+           period = "%.6g days" % days
+       if not list:
+           emit(faqconf.NO_RECENT, period=period)
+       elif len(list) == 1:
+           emit(faqconf.ONE_RECENT, period=period)
+       else:
+           emit(faqconf.SOME_RECENT, period=period, count=len(list))
+       self.format_all(map(lambda (mtime, file): file, list))
+       emit(faqconf.TAIL_RECENT)
+
+    def do_roulette(self):
+       self.prologue(faqconf.T_ROULETTE)
+       file = self.dir.roulette()
+       self.dir.show(file)
+
+    def do_help(self):
+       self.prologue(faqconf.T_HELP)
+       emit(faqconf.HELP)
+
+    def do_show(self):
+       entry = self.dir.open(self.ui.file)
+       self.prologue("Python FAQ Entry")
+       entry.show()
+
+    def do_add(self):
+       self.prologue(T_ADD)
+       self.error("Not yet implemented")
+
+    def do_delete(self):
+       self.prologue(T_DELETE)
+       self.error("Not yet implemented")
+
+    def do_log(self):
+       entry = self.dir.open(self.ui.file)
+       self.prologue(faqconf.T_LOG, entry)
+       emit(faqconf.LOG, entry)
+       self.rlog(interpolate(faqconf.SH_RLOG, entry), entry)
+
+    def rlog(self, command, entry=None):
+       output = os.popen(command).read()
+       sys.stdout.write("<PRE>")
+       athead = 0
+       lines = string.split(output, "\n")
+       while lines and not lines[-1]:
+           del lines[-1]
+       if lines:
+           line = lines[-1]
+           if line[:1] == '=' and len(line) >= 40 and \
+              line == line[0]*len(line):
+               del lines[-1]
+       for line in lines:
+           if entry and athead and line[:9] == 'revision ':
+               rev = string.strip(line[9:])
+               if rev != "1.1":
+                   emit(faqconf.DIFFLINK, entry, rev=rev, line=line)
+               else:
+                   print line
+               athead = 0
+           else:
+               athead = 0
+               if line[:1] == '-' and len(line) >= 20 and \
+                  line == len(line) * line[0]:
+                   athead = 1
+                   sys.stdout.write("<HR>")
+               else:
+                   print line
+       print "</PRE>"
+
+    def do_diff(self):
+       entry = self.dir.open(self.ui.file)
+       rev = self.ui.rev
+       r = regex.compile(
+           "^\([1-9][0-9]?[0-9]?\)\.\([1-9][0-9]?[0-9]?[0-9]?\)$")
+       if r.match(rev) < 0:
+           self.error("Invalid revision number: %s" % `rev`)
+       [major, minor] = map(string.atoi, r.group(1, 2))
+       if minor == 1:
+           self.error("No previous revision")
+           return
+       prev = "%d.%d" % (major, minor-1)
+       self.prologue(faqconf.T_DIFF, entry)
+       self.shell(interpolate(faqconf.SH_RDIFF, entry, rev=rev, prev=prev))
+
+    def shell(self, command):
+       output = os.popen(command).read()
+       sys.stdout.write("<PRE>")
+       print escape(output)
+       print "</PRE>"
+
+    def do_new(self):
+       editor = FaqEditor(self.ui, self.dir.new(self.file))
+       self.prologue(faqconf.T_NEW)
+       self.error("Not yet implemented")
+
+    def do_edit(self):
+       entry = self.dir.open(self.ui.file)
+       entry.load_version()
+       self.prologue(faqconf.T_EDIT)
+       emit(faqconf.EDITHEAD)
+       emit(faqconf.EDITFORM1, entry, editversion=entry.version)
+       emit(faqconf.EDITFORM2, entry, load_my_cookie(), log=self.ui.log)
+       emit(faqconf.EDITFORM3)
+       entry.show(edit=0)
+
+    def do_review(self):
+       entry = self.dir.open(self.ui.file)
+       entry.load_version()
+       # Check that the FAQ entry number didn't change
+       if string.split(self.ui.title)[:1] != string.split(entry.title)[:1]:
+           self.error("Don't change the FAQ entry number please.")
+           return
+       # Check that the edited version is the current version
+       if entry.version != self.ui.editversion:
+           self.error("Version conflict.")
+           emit(faqconf.VERSIONCONFLICT, entry, self.ui)
+           return
+       commit_ok = ((not faqconf.PASSWORD
+                     or self.ui.password == faqconf.PASSWORD) 
+                    and self.ui.author
+                    and '@' in self.ui.email
+                    and self.ui.log)
+       if self.ui.commit:
+           if not commit_ok:
+               self.cantcommit()
+           else:
+               self.commit()
+           return
+       self.prologue(faqconf.T_REVIEW)
+       emit(faqconf.REVIEWHEAD)
+       entry.body = self.ui.body
+       entry.title = self.ui.title
+       entry.show(edit=0)
+       emit(faqconf.EDITFORM1, entry, self.ui)
+       if commit_ok:
+           emit(faqconf.COMMIT)
+       else:
+           emit(faqconf.NOCOMMIT)
+       emit(faqconf.EDITFORM2, entry, load_my_cookie(), log=self.ui.log)
+       emit(faqconf.EDITFORM3)
+
+    def cantcommit(self):
+       self.prologue(faqconf.T_CANTCOMMIT)
+       print faqconf.CANTCOMMIT_HEAD
+       if not self.ui.passwd:
+           emit(faqconf.NEED_PASSWD)
+       if not self.ui.log:
+           emit(faqconf.NEED_LOG)
+       if not self.ui.author:
+           emit(faqconf.NEED_AUTHOR)
+       if not self.ui.email:
+           emit(faqconf.NEED_EMAIL)
+       print faqconf.CANTCOMMIT_TAIL
+
+    def commit(self):
+       file = self.ui.file
+       entry = self.dir.open(file)
+       # Chech that there were any changes
+       if self.ui.body == entry.body and self.ui.title == entry.title:
+           self.error("No changes.")
+           return
+       # XXX Should lock here
+       try:
+           os.unlink(file)
+       except os.error:
+           pass
+       try:
+           f = open(file, "w")
+       except IOError, why:
+           self.error(faqconf.CANTWRITE, file=file, why=why)
+           return
+       date = time.ctime(time.time())
+       emit(faqconf.FILEHEADER, self.ui, os.environ, date=date, file=f)
+       f.write("\n")
+       f.write(self.ui.body)
+       f.write("\n")
+       f.close()
+
+       import tempfile
+       tfn = tempfile.mktemp()
+       f = open(tfn, "w")
+       emit(faqconf.LOGHEADER, self.ui, os.environ, date=date, file=f)
+       f.close()
+
+       command = interpolate(
+           faqconf.SH_LOCK + "\n" + faqconf.SH_CHECKIN,
+           file=file, tfn=tfn)
+
+       p = os.popen(command)
+       output = p.read()
+       sts = p.close()
+       # XXX Should unlock here
+       if not sts:
+           self.prologue(faqconf.T_COMMITTED)
+           emit(faqconf.COMMITTED)
+       else:
+           self.error(faqconf.T_COMMITFAILED)
+           emit(faqconf.COMMITFAILED, sts=sts)
+       print "<PRE>%s</PRE>" % cgi.escape(output)
+
+       try:
+           os.unlink(tfn)
+       except os.error:
+           pass
+
+       entry = self.dir.open(file)
+       entry.show()
+
+wiz = FaqWizard()
+wiz.go()
+
+BOOTSTRAP = """\
+#! /usr/local/bin/python
+FAQDIR = "/usr/people/guido/python/FAQ"
+
+# This bootstrap script should be placed in your cgi-bin directory.
+# You only need to edit the first two lines (above): Change
+# /usr/local/bin/python to where your Python interpreter lives (you
+# can't use /usr/bin/env here!); change FAQDIR to where your FAQ
+# lives.  The faqwiz.py and faqconf.py files should live there, too.
+
+import posix
+t1 = posix.times()
+import os, sys, time, operator
+os.chdir(FAQDIR)
+sys.path.insert(0, FAQDIR)
+try:
+    import faqwiz
+except SystemExit, n:
+    sys.exit(n)
+except:
+    t, v, tb = sys.exc_type, sys.exc_value, sys.exc_traceback
+    print
+    import cgi
+    cgi.print_exception(t, v, tb)
+t2 = posix.times()
+fmt = "<BR>(times: user %.3g, sys %.3g, ch-user %.3g, ch-sys %.3g, real %.3g)"
+print fmt % tuple(map(operator.sub, t2, t1))
+"""