| """Generic FAQ Wizard. |
| |
| This is a CGI program that maintains a user-editable FAQ. It uses RCS |
| to keep track of changes to individual FAQ entries. It is fully |
| configurable; everything you might want to change when using this |
| program to maintain some other FAQ than the Python FAQ is contained in |
| the configuration module, faqconf.py. |
| |
| Note that this is not an executable script; it's an importable module. |
| The actual script to place in cgi-bin is faqw.py. |
| |
| """ |
| |
| import sys, time, os, stat, re, cgi, faqconf |
| from faqconf import * # This imports all uppercase names |
| now = time.time() |
| |
| class FileError: |
| def __init__(self, file): |
| self.file = file |
| |
| class InvalidFile(FileError): |
| pass |
| |
| class NoSuchSection(FileError): |
| def __init__(self, section): |
| FileError.__init__(self, NEWFILENAME %(section, 1)) |
| self.section = section |
| |
| class NoSuchFile(FileError): |
| def __init__(self, file, why=None): |
| FileError.__init__(self, file) |
| self.why = why |
| |
| def escape(s): |
| s = s.replace('&', '&') |
| s = s.replace('<', '<') |
| s = s.replace('>', '>') |
| return s |
| |
| def escapeq(s): |
| s = escape(s) |
| s = s.replace('"', '"') |
| return s |
| |
| def _interpolate(format, args, kw): |
| try: |
| quote = kw['_quote'] |
| except KeyError: |
| quote = 1 |
| d = (kw,) + args + (faqconf.__dict__,) |
| m = MagicDict(d, quote) |
| return format % m |
| |
| def interpolate(format, *args, **kw): |
| return _interpolate(format, args, kw) |
| |
| def emit(format, *args, **kw): |
| try: |
| f = kw['_file'] |
| except KeyError: |
| f = sys.stdout |
| f.write(_interpolate(format, args, kw)) |
| |
| translate_prog = None |
| |
| def translate(text, pre=0): |
| global translate_prog |
| if not translate_prog: |
| translate_prog = prog = re.compile( |
| r'\b(http|ftp|https)://\S+(\b|/)|\b[-.\w]+@[-.\w]+') |
| else: |
| prog = translate_prog |
| i = 0 |
| list = [] |
| while 1: |
| m = prog.search(text, i) |
| if not m: |
| break |
| j = m.start() |
| list.append(escape(text[i:j])) |
| i = j |
| url = m.group(0) |
| while url[-1] in '();:,.?\'"<>': |
| url = url[:-1] |
| i = i + len(url) |
| url = escape(url) |
| if not pre or (pre and PROCESS_PREFORMAT): |
| if ':' in url: |
| repl = '<A HREF="%s">%s</A>' % (url, url) |
| else: |
| repl = '<A HREF="mailto:%s">%s</A>' % (url, url) |
| else: |
| repl = url |
| list.append(repl) |
| j = len(text) |
| list.append(escape(text[i:j])) |
| return ''.join(list) |
| |
| def emphasize(line): |
| return re.sub(r'\*([a-zA-Z]+)\*', r'<I>\1</I>', line) |
| |
| revparse_prog = None |
| |
| def revparse(rev): |
| global revparse_prog |
| if not revparse_prog: |
| revparse_prog = re.compile(r'^(\d{1,3})\.(\d{1,4})$') |
| m = revparse_prog.match(rev) |
| if not m: |
| return None |
| [major, minor] = map(int, m.group(1, 2)) |
| return major, minor |
| |
| logon = 0 |
| def log(text): |
| if logon: |
| logfile = open("logfile", "a") |
| logfile.write(text + "\n") |
| logfile.close() |
| |
| def load_cookies(): |
| if not os.environ.has_key('HTTP_COOKIE'): |
| return {} |
| raw = os.environ['HTTP_COOKIE'] |
| words = [s.strip() for s in raw.split(';')] |
| cookies = {} |
| for word in words: |
| i = word.find('=') |
| 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[COOKIE_NAME] |
| except KeyError: |
| return {} |
| import urllib |
| value = urllib.unquote(value) |
| words = value.split('/') |
| while len(words) < 3: |
| words.append('') |
| author = '/'.join(words[:-2]) |
| email = words[-2] |
| password = words[-1] |
| return {'author': author, |
| 'email': email, |
| 'password': password} |
| |
| def send_my_cookie(ui): |
| name = COOKIE_NAME |
| value = "%s/%s/%s" % (ui.author, ui.email, ui.password) |
| import urllib |
| value = urllib.quote(value) |
| then = now + COOKIE_LIFETIME |
| gmt = time.gmtime(then) |
| path = os.environ.get('SCRIPT_NAME', '/cgi-bin/') |
| print "Set-Cookie: %s=%s; path=%s;" % (name, value, path), |
| print time.strftime("expires=%a, %d-%b-%y %X GMT", gmt) |
| |
| class MagicDict: |
| |
| def __init__(self, d, quote): |
| self.__d = d |
| self.__quote = quote |
| |
| def __getitem__(self, key): |
| for d in self.__d: |
| try: |
| value = d[key] |
| if value: |
| value = str(value) |
| if self.__quote: |
| value = escapeq(value) |
| return value |
| except KeyError: |
| pass |
| return '' |
| |
| class UserInput: |
| |
| def __init__(self): |
| self.__form = cgi.FieldStorage() |
| #log("\n\nbody: " + self.body) |
| |
| def __getattr__(self, name): |
| if name[0] == '_': |
| raise AttributeError |
| try: |
| value = self.__form[name].value |
| except (TypeError, KeyError): |
| value = '' |
| else: |
| value = value.strip() |
| setattr(self, name, value) |
| return value |
| |
| def __getitem__(self, key): |
| return getattr(self, key) |
| |
| class FaqEntry: |
| |
| def __init__(self, fp, file, sec_num): |
| self.file = file |
| self.sec, self.num = sec_num |
| if fp: |
| import rfc822 |
| self.__headers = rfc822.Message(fp) |
| self.body = fp.read().strip() |
| else: |
| self.__headers = {'title': "%d.%d. " % sec_num} |
| self.body = '' |
| |
| def __getattr__(self, name): |
| if name[0] == '_': |
| raise AttributeError |
| key = '-'.join(name.split('_')) |
| try: |
| value = self.__headers[key] |
| except KeyError: |
| value = '' |
| setattr(self, name, value) |
| return value |
| |
| def __getitem__(self, key): |
| return getattr(self, key) |
| |
| def load_version(self): |
| command = interpolate(SH_RLOG_H, self) |
| p = os.popen(command) |
| version = '' |
| while 1: |
| line = p.readline() |
| if not line: |
| break |
| if line[:5] == 'head:': |
| version = line[5:].strip() |
| p.close() |
| self.version = version |
| |
| def getmtime(self): |
| if not self.last_changed_date: |
| return 0 |
| try: |
| return os.stat(self.file)[stat.ST_MTIME] |
| except os.error: |
| return 0 |
| |
| def emit_marks(self): |
| mtime = self.getmtime() |
| if mtime >= now - DT_VERY_RECENT: |
| emit(MARK_VERY_RECENT, self) |
| elif mtime >= now - DT_RECENT: |
| emit(MARK_RECENT, self) |
| |
| def show(self, edit=1): |
| emit(ENTRY_HEADER1, self) |
| self.emit_marks() |
| emit(ENTRY_HEADER2, self) |
| pre = 0 |
| raw = 0 |
| for line in self.body.split('\n'): |
| # Allow the user to insert raw html into a FAQ answer |
| # (Skip Montanaro, with changes by Guido) |
| tag = line.rstrip().lower() |
| if tag == '<html>': |
| raw = 1 |
| continue |
| if tag == '</html>': |
| raw = 0 |
| continue |
| if raw: |
| print line |
| continue |
| if not line.strip(): |
| if pre: |
| print '</PRE>' |
| pre = 0 |
| else: |
| print '<P>' |
| else: |
| if not line[0].isspace(): |
| if pre: |
| print '</PRE>' |
| pre = 0 |
| else: |
| if not pre: |
| print '<PRE>' |
| pre = 1 |
| if '/' in line or '@' in line: |
| line = translate(line, pre) |
| 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(ENTRY_FOOTER, self) |
| if self.last_changed_date: |
| emit(ENTRY_LOGINFO, self) |
| print '<P>' |
| |
| class FaqDir: |
| |
| entryclass = FaqEntry |
| |
| __okprog = re.compile(OKFILENAME) |
| |
| 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 self.__okprog.match(file): |
| files.append(file) |
| files.sort() |
| |
| def good(self, file): |
| return self.__okprog.match(file) |
| |
| def parse(self, file): |
| m = self.good(file) |
| if not m: |
| return None |
| sec, num = m.group(1, 2) |
| return int(sec), int(num) |
| |
| 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, section): |
| if not SECTION_TITLES.has_key(section): |
| raise NoSuchSection(section) |
| maxnum = 0 |
| for file in self.list(): |
| sec, num = self.parse(file) |
| if sec == section: |
| maxnum = max(maxnum, num) |
| sec_num = (section, maxnum+1) |
| file = NEWFILENAME % sec_num |
| return self.entryclass(None, file, sec_num) |
| |
| 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 type %r." % (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) |
| except NoSuchSection, exc: |
| self.error("No section number %s" % exc.section) |
| self.epilogue() |
| |
| def error(self, message, **kw): |
| self.prologue(T_ERROR) |
| emit(message, kw) |
| |
| def prologue(self, title, entry=None, **kw): |
| emit(PROLOGUE, entry, kwdict=kw, title=escape(title)) |
| |
| def epilogue(self): |
| emit(EPILOGUE) |
| |
| def do_home(self): |
| self.prologue(T_HOME) |
| emit(HOME) |
| |
| def do_debug(self): |
| self.prologue("FAQ Wizard Debugging") |
| form = cgi.FieldStorage() |
| cgi.print_form(form) |
| cgi.print_environ(os.environ) |
| cgi.print_directory() |
| cgi.print_arguments() |
| |
| def do_search(self): |
| query = self.ui.query |
| if not query: |
| self.error("Empty query string!") |
| return |
| if self.ui.querytype == 'simple': |
| query = re.escape(query) |
| queries = [query] |
| elif self.ui.querytype in ('anykeywords', 'allkeywords'): |
| words = filter(None, re.split('\W+', query)) |
| if not words: |
| self.error("No keywords specified!") |
| return |
| words = map(lambda w: r'\b%s\b' % w, words) |
| if self.ui.querytype[:3] == 'any': |
| queries = ['|'.join(words)] |
| else: |
| # Each of the individual queries must match |
| queries = words |
| else: |
| # Default to regular expression |
| queries = [query] |
| self.prologue(T_SEARCH) |
| progs = [] |
| for query in queries: |
| if self.ui.casefold == 'no': |
| p = re.compile(query) |
| else: |
| p = re.compile(query, re.IGNORECASE) |
| progs.append(p) |
| hits = [] |
| for file in self.dir.list(): |
| try: |
| entry = self.dir.open(file) |
| except FileError: |
| constants |
| for p in progs: |
| if not p.search(entry.title) and not p.search(entry.body): |
| break |
| else: |
| hits.append(file) |
| if not hits: |
| emit(NO_HITS, self.ui, count=0) |
| elif len(hits) <= MAXHITS: |
| if len(hits) == 1: |
| emit(ONE_HIT, count=1) |
| else: |
| emit(FEW_HITS, count=len(hits)) |
| self.format_all(hits, headers=0) |
| else: |
| emit(MANY_HITS, count=len(hits)) |
| self.format_index(hits) |
| |
| def do_all(self): |
| self.prologue(T_ALL) |
| files = self.dir.list() |
| self.last_changed(files) |
| self.format_index(files, localrefs=1) |
| self.format_all(files) |
| |
| def do_compat(self): |
| files = self.dir.list() |
| emit(COMPAT) |
| self.last_changed(files) |
| self.format_index(files, localrefs=1) |
| self.format_all(files, edit=0) |
| sys.exit(0) # XXX Hack to suppress epilogue |
| |
| def last_changed(self, files): |
| latest = 0 |
| for file in files: |
| entry = self.dir.open(file) |
| if entry: |
| mtime = mtime = entry.getmtime() |
| if mtime > latest: |
| latest = mtime |
| print time.strftime(LAST_CHANGED, time.localtime(latest)) |
| emit(EXPLAIN_MARKS) |
| |
| def format_all(self, files, edit=1, headers=1): |
| sec = 0 |
| for file in files: |
| try: |
| entry = self.dir.open(file) |
| except NoSuchFile: |
| continue |
| if headers and entry.sec != sec: |
| sec = entry.sec |
| try: |
| title = SECTION_TITLES[sec] |
| except KeyError: |
| title = "Untitled" |
| emit("\n<HR>\n<H1>%(sec)s. %(title)s</H1>\n", |
| sec=sec, title=title) |
| entry.show(edit=edit) |
| |
| def do_index(self): |
| self.prologue(T_INDEX) |
| files = self.dir.list() |
| self.last_changed(files) |
| self.format_index(files, add=1) |
| |
| def format_index(self, files, add=0, localrefs=0): |
| sec = 0 |
| for file in files: |
| try: |
| entry = self.dir.open(file) |
| except NoSuchFile: |
| continue |
| if entry.sec != sec: |
| if sec: |
| if add: |
| emit(INDEX_ADDSECTION, sec=sec) |
| emit(INDEX_ENDSECTION, sec=sec) |
| sec = entry.sec |
| try: |
| title = SECTION_TITLES[sec] |
| except KeyError: |
| title = "Untitled" |
| emit(INDEX_SECTION, sec=sec, title=title) |
| if localrefs: |
| emit(LOCAL_ENTRY, entry) |
| else: |
| emit(INDEX_ENTRY, entry) |
| entry.emit_marks() |
| if sec: |
| if add: |
| emit(INDEX_ADDSECTION, sec=sec) |
| emit(INDEX_ENDSECTION, sec=sec) |
| |
| def do_recent(self): |
| if not self.ui.days: |
| days = 1 |
| else: |
| days = float(self.ui.days) |
| try: |
| cutoff = now - days * 24 * 3600 |
| except OverflowError: |
| cutoff = 0 |
| list = [] |
| for file in self.dir.list(): |
| entry = self.dir.open(file) |
| if not entry: |
| continue |
| mtime = entry.getmtime() |
| if mtime >= cutoff: |
| list.append((mtime, file)) |
| list.sort() |
| list.reverse() |
| self.prologue(T_RECENT) |
| if days <= 1: |
| period = "%.2g hours" % (days*24) |
| else: |
| period = "%.6g days" % days |
| if not list: |
| emit(NO_RECENT, period=period) |
| elif len(list) == 1: |
| emit(ONE_RECENT, period=period) |
| else: |
| emit(SOME_RECENT, period=period, count=len(list)) |
| self.format_all(map(lambda (mtime, file): file, list), headers=0) |
| emit(TAIL_RECENT) |
| |
| def do_roulette(self): |
| import random |
| files = self.dir.list() |
| if not files: |
| self.error("No entries.") |
| return |
| file = random.choice(files) |
| self.prologue(T_ROULETTE) |
| emit(ROULETTE) |
| self.dir.show(file) |
| |
| def do_help(self): |
| self.prologue(T_HELP) |
| emit(HELP) |
| |
| def do_show(self): |
| entry = self.dir.open(self.ui.file) |
| self.prologue(T_SHOW) |
| entry.show() |
| |
| def do_add(self): |
| self.prologue(T_ADD) |
| emit(ADD_HEAD) |
| sections = SECTION_TITLES.items() |
| sections.sort() |
| for section, title in sections: |
| emit(ADD_SECTION, section=section, title=title) |
| emit(ADD_TAIL) |
| |
| def do_delete(self): |
| self.prologue(T_DELETE) |
| emit(DELETE) |
| |
| def do_log(self): |
| entry = self.dir.open(self.ui.file) |
| self.prologue(T_LOG, entry) |
| emit(LOG, entry) |
| self.rlog(interpolate(SH_RLOG, entry), entry) |
| |
| def rlog(self, command, entry=None): |
| output = os.popen(command).read() |
| sys.stdout.write('<PRE>') |
| athead = 0 |
| lines = output.split('\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] |
| headrev = None |
| for line in lines: |
| if entry and athead and line[:9] == 'revision ': |
| rev = line[9:].split() |
| mami = revparse(rev) |
| if not mami: |
| print line |
| else: |
| emit(REVISIONLINK, entry, rev=rev, line=line) |
| if mami[1] > 1: |
| prev = "%d.%d" % (mami[0], mami[1]-1) |
| emit(DIFFLINK, entry, prev=prev, rev=rev) |
| if headrev: |
| emit(DIFFLINK, entry, prev=rev, rev=headrev) |
| else: |
| headrev = rev |
| print |
| 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_revision(self): |
| entry = self.dir.open(self.ui.file) |
| rev = self.ui.rev |
| mami = revparse(rev) |
| if not mami: |
| self.error("Invalid revision number: %r." % (rev,)) |
| self.prologue(T_REVISION, entry) |
| self.shell(interpolate(SH_REVISION, entry, rev=rev)) |
| |
| def do_diff(self): |
| entry = self.dir.open(self.ui.file) |
| prev = self.ui.prev |
| rev = self.ui.rev |
| mami = revparse(rev) |
| if not mami: |
| self.error("Invalid revision number: %r." % (rev,)) |
| if prev: |
| if not revparse(prev): |
| self.error("Invalid previous revision number: %r." % (prev,)) |
| else: |
| prev = '%d.%d' % (mami[0], mami[1]) |
| self.prologue(T_DIFF, entry) |
| self.shell(interpolate(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): |
| entry = self.dir.new(section=int(self.ui.section)) |
| entry.version = '*new*' |
| self.prologue(T_EDIT) |
| emit(EDITHEAD) |
| emit(EDITFORM1, entry, editversion=entry.version) |
| emit(EDITFORM2, entry, load_my_cookie()) |
| emit(EDITFORM3) |
| entry.show(edit=0) |
| |
| def do_edit(self): |
| entry = self.dir.open(self.ui.file) |
| entry.load_version() |
| self.prologue(T_EDIT) |
| emit(EDITHEAD) |
| emit(EDITFORM1, entry, editversion=entry.version) |
| emit(EDITFORM2, entry, load_my_cookie()) |
| emit(EDITFORM3) |
| entry.show(edit=0) |
| |
| def do_review(self): |
| send_my_cookie(self.ui) |
| if self.ui.editversion == '*new*': |
| sec, num = self.dir.parse(self.ui.file) |
| entry = self.dir.new(section=sec) |
| entry.version = "*new*" |
| if entry.file != self.ui.file: |
| self.error("Commit version conflict!") |
| emit(NEWCONFLICT, self.ui, sec=sec, num=num) |
| return |
| else: |
| entry = self.dir.open(self.ui.file) |
| entry.load_version() |
| # Check that the FAQ entry number didn't change |
| if self.ui.title.split()[:1] != entry.title.split()[:1]: |
| self.error("Don't change the entry number please!") |
| return |
| # Check that the edited version is the current version |
| if entry.version != self.ui.editversion: |
| self.error("Commit version conflict!") |
| emit(VERSIONCONFLICT, entry, self.ui) |
| return |
| commit_ok = ((not PASSWORD |
| or self.ui.password == 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(entry) |
| return |
| self.prologue(T_REVIEW) |
| emit(REVIEWHEAD) |
| entry.body = self.ui.body |
| entry.title = self.ui.title |
| entry.show(edit=0) |
| emit(EDITFORM1, self.ui, entry) |
| if commit_ok: |
| emit(COMMIT) |
| else: |
| emit(NOCOMMIT_HEAD) |
| self.errordetail() |
| emit(NOCOMMIT_TAIL) |
| emit(EDITFORM2, self.ui, entry, load_my_cookie()) |
| emit(EDITFORM3) |
| |
| def cantcommit(self): |
| self.prologue(T_CANTCOMMIT) |
| print CANTCOMMIT_HEAD |
| self.errordetail() |
| print CANTCOMMIT_TAIL |
| |
| def errordetail(self): |
| if PASSWORD and self.ui.password != PASSWORD: |
| emit(NEED_PASSWD) |
| if not self.ui.log: |
| emit(NEED_LOG) |
| if not self.ui.author: |
| emit(NEED_AUTHOR) |
| if not self.ui.email: |
| emit(NEED_EMAIL) |
| |
| def commit(self, entry): |
| file = entry.file |
| # Normalize line endings in body |
| if '\r' in self.ui.body: |
| self.ui.body = re.sub('\r\n?', '\n', self.ui.body) |
| # Normalize whitespace in title |
| self.ui.title = ' '.join(self.ui.title.split()) |
| # Check that there were any changes |
| if self.ui.body == entry.body and self.ui.title == entry.title: |
| self.error("You didn't make any changes!") |
| return |
| |
| # need to lock here because otherwise the file exists and is not writable (on NT) |
| command = interpolate(SH_LOCK, file=file) |
| p = os.popen(command) |
| output = p.read() |
| |
| try: |
| os.unlink(file) |
| except os.error: |
| pass |
| try: |
| f = open(file, 'w') |
| except IOError, why: |
| self.error(CANTWRITE, file=file, why=why) |
| return |
| date = time.ctime(now) |
| emit(FILEHEADER, self.ui, os.environ, date=date, _file=f, _quote=0) |
| f.write('\n') |
| f.write(self.ui.body) |
| f.write('\n') |
| f.close() |
| |
| import tempfile |
| tf = tempfile.NamedTemporaryFile() |
| emit(LOGHEADER, self.ui, os.environ, date=date, _file=tf) |
| tf.flush() |
| tf.seek(0) |
| |
| command = interpolate(SH_CHECKIN, file=file, tfn=tf.name) |
| log("\n\n" + command) |
| p = os.popen(command) |
| output = p.read() |
| sts = p.close() |
| log("output: " + output) |
| log("done: " + str(sts)) |
| log("TempFile:\n" + tf.read() + "end") |
| |
| if not sts: |
| self.prologue(T_COMMITTED) |
| emit(COMMITTED) |
| else: |
| self.error(T_COMMITFAILED) |
| emit(COMMITFAILED, sts=sts) |
| print '<PRE>%s</PRE>' % escape(output) |
| |
| try: |
| os.unlink(tf.name) |
| except os.error: |
| pass |
| |
| entry = self.dir.open(file) |
| entry.show() |
| |
| wiz = FaqWizard() |
| wiz.go() |