| 1 | """Generic FAQ Wizard. | 
|---|
| 2 |  | 
|---|
| 3 | This is a CGI program that maintains a user-editable FAQ.  It uses RCS | 
|---|
| 4 | to keep track of changes to individual FAQ entries.  It is fully | 
|---|
| 5 | configurable; everything you might want to change when using this | 
|---|
| 6 | program to maintain some other FAQ than the Python FAQ is contained in | 
|---|
| 7 | the configuration module, faqconf.py. | 
|---|
| 8 |  | 
|---|
| 9 | Note that this is not an executable script; it's an importable module. | 
|---|
| 10 | The actual script to place in cgi-bin is faqw.py. | 
|---|
| 11 |  | 
|---|
| 12 | """ | 
|---|
| 13 |  | 
|---|
| 14 | import sys, time, os, stat, re, cgi, faqconf | 
|---|
| 15 | from faqconf import *                   # This imports all uppercase names | 
|---|
| 16 | now = time.time() | 
|---|
| 17 |  | 
|---|
| 18 | class FileError: | 
|---|
| 19 | def __init__(self, file): | 
|---|
| 20 | self.file = file | 
|---|
| 21 |  | 
|---|
| 22 | class InvalidFile(FileError): | 
|---|
| 23 | pass | 
|---|
| 24 |  | 
|---|
| 25 | class NoSuchSection(FileError): | 
|---|
| 26 | def __init__(self, section): | 
|---|
| 27 | FileError.__init__(self, NEWFILENAME %(section, 1)) | 
|---|
| 28 | self.section = section | 
|---|
| 29 |  | 
|---|
| 30 | class NoSuchFile(FileError): | 
|---|
| 31 | def __init__(self, file, why=None): | 
|---|
| 32 | FileError.__init__(self, file) | 
|---|
| 33 | self.why = why | 
|---|
| 34 |  | 
|---|
| 35 | def escape(s): | 
|---|
| 36 | s = s.replace('&', '&') | 
|---|
| 37 | s = s.replace('<', '<') | 
|---|
| 38 | s = s.replace('>', '>') | 
|---|
| 39 | return s | 
|---|
| 40 |  | 
|---|
| 41 | def escapeq(s): | 
|---|
| 42 | s = escape(s) | 
|---|
| 43 | s = s.replace('"', '"') | 
|---|
| 44 | return s | 
|---|
| 45 |  | 
|---|
| 46 | def _interpolate(format, args, kw): | 
|---|
| 47 | try: | 
|---|
| 48 | quote = kw['_quote'] | 
|---|
| 49 | except KeyError: | 
|---|
| 50 | quote = 1 | 
|---|
| 51 | d = (kw,) + args + (faqconf.__dict__,) | 
|---|
| 52 | m = MagicDict(d, quote) | 
|---|
| 53 | return format % m | 
|---|
| 54 |  | 
|---|
| 55 | def interpolate(format, *args, **kw): | 
|---|
| 56 | return _interpolate(format, args, kw) | 
|---|
| 57 |  | 
|---|
| 58 | def emit(format, *args, **kw): | 
|---|
| 59 | try: | 
|---|
| 60 | f = kw['_file'] | 
|---|
| 61 | except KeyError: | 
|---|
| 62 | f = sys.stdout | 
|---|
| 63 | f.write(_interpolate(format, args, kw)) | 
|---|
| 64 |  | 
|---|
| 65 | translate_prog = None | 
|---|
| 66 |  | 
|---|
| 67 | def translate(text, pre=0): | 
|---|
| 68 | global translate_prog | 
|---|
| 69 | if not translate_prog: | 
|---|
| 70 | translate_prog = prog = re.compile( | 
|---|
| 71 | r'\b(http|ftp|https)://\S+(\b|/)|\b[-.\w]+@[-.\w]+') | 
|---|
| 72 | else: | 
|---|
| 73 | prog = translate_prog | 
|---|
| 74 | i = 0 | 
|---|
| 75 | list = [] | 
|---|
| 76 | while 1: | 
|---|
| 77 | m = prog.search(text, i) | 
|---|
| 78 | if not m: | 
|---|
| 79 | break | 
|---|
| 80 | j = m.start() | 
|---|
| 81 | list.append(escape(text[i:j])) | 
|---|
| 82 | i = j | 
|---|
| 83 | url = m.group(0) | 
|---|
| 84 | while url[-1] in '();:,.?\'"<>': | 
|---|
| 85 | url = url[:-1] | 
|---|
| 86 | i = i + len(url) | 
|---|
| 87 | url = escape(url) | 
|---|
| 88 | if not pre or (pre and PROCESS_PREFORMAT): | 
|---|
| 89 | if ':' in url: | 
|---|
| 90 | repl = '<A HREF="%s">%s</A>' % (url, url) | 
|---|
| 91 | else: | 
|---|
| 92 | repl = '<A HREF="mailto:%s">%s</A>' % (url, url) | 
|---|
| 93 | else: | 
|---|
| 94 | repl = url | 
|---|
| 95 | list.append(repl) | 
|---|
| 96 | j = len(text) | 
|---|
| 97 | list.append(escape(text[i:j])) | 
|---|
| 98 | return ''.join(list) | 
|---|
| 99 |  | 
|---|
| 100 | def emphasize(line): | 
|---|
| 101 | return re.sub(r'\*([a-zA-Z]+)\*', r'<I>\1</I>', line) | 
|---|
| 102 |  | 
|---|
| 103 | revparse_prog = None | 
|---|
| 104 |  | 
|---|
| 105 | def revparse(rev): | 
|---|
| 106 | global revparse_prog | 
|---|
| 107 | if not revparse_prog: | 
|---|
| 108 | revparse_prog = re.compile(r'^(\d{1,3})\.(\d{1,4})$') | 
|---|
| 109 | m = revparse_prog.match(rev) | 
|---|
| 110 | if not m: | 
|---|
| 111 | return None | 
|---|
| 112 | [major, minor] = map(int, m.group(1, 2)) | 
|---|
| 113 | return major, minor | 
|---|
| 114 |  | 
|---|
| 115 | logon = 0 | 
|---|
| 116 | def log(text): | 
|---|
| 117 | if logon: | 
|---|
| 118 | logfile = open("logfile", "a") | 
|---|
| 119 | logfile.write(text + "\n") | 
|---|
| 120 | logfile.close() | 
|---|
| 121 |  | 
|---|
| 122 | def load_cookies(): | 
|---|
| 123 | if not os.environ.has_key('HTTP_COOKIE'): | 
|---|
| 124 | return {} | 
|---|
| 125 | raw = os.environ['HTTP_COOKIE'] | 
|---|
| 126 | words = [s.strip() for s in raw.split(';')] | 
|---|
| 127 | cookies = {} | 
|---|
| 128 | for word in words: | 
|---|
| 129 | i = word.find('=') | 
|---|
| 130 | if i >= 0: | 
|---|
| 131 | key, value = word[:i], word[i+1:] | 
|---|
| 132 | cookies[key] = value | 
|---|
| 133 | return cookies | 
|---|
| 134 |  | 
|---|
| 135 | def load_my_cookie(): | 
|---|
| 136 | cookies = load_cookies() | 
|---|
| 137 | try: | 
|---|
| 138 | value = cookies[COOKIE_NAME] | 
|---|
| 139 | except KeyError: | 
|---|
| 140 | return {} | 
|---|
| 141 | import urllib | 
|---|
| 142 | value = urllib.unquote(value) | 
|---|
| 143 | words = value.split('/') | 
|---|
| 144 | while len(words) < 3: | 
|---|
| 145 | words.append('') | 
|---|
| 146 | author = '/'.join(words[:-2]) | 
|---|
| 147 | email = words[-2] | 
|---|
| 148 | password = words[-1] | 
|---|
| 149 | return {'author': author, | 
|---|
| 150 | 'email': email, | 
|---|
| 151 | 'password': password} | 
|---|
| 152 |  | 
|---|
| 153 | def send_my_cookie(ui): | 
|---|
| 154 | name = COOKIE_NAME | 
|---|
| 155 | value = "%s/%s/%s" % (ui.author, ui.email, ui.password) | 
|---|
| 156 | import urllib | 
|---|
| 157 | value = urllib.quote(value) | 
|---|
| 158 | then = now + COOKIE_LIFETIME | 
|---|
| 159 | gmt = time.gmtime(then) | 
|---|
| 160 | path = os.environ.get('SCRIPT_NAME', '/cgi-bin/') | 
|---|
| 161 | print "Set-Cookie: %s=%s; path=%s;" % (name, value, path), | 
|---|
| 162 | print time.strftime("expires=%a, %d-%b-%y %X GMT", gmt) | 
|---|
| 163 |  | 
|---|
| 164 | class MagicDict: | 
|---|
| 165 |  | 
|---|
| 166 | def __init__(self, d, quote): | 
|---|
| 167 | self.__d = d | 
|---|
| 168 | self.__quote = quote | 
|---|
| 169 |  | 
|---|
| 170 | def __getitem__(self, key): | 
|---|
| 171 | for d in self.__d: | 
|---|
| 172 | try: | 
|---|
| 173 | value = d[key] | 
|---|
| 174 | if value: | 
|---|
| 175 | value = str(value) | 
|---|
| 176 | if self.__quote: | 
|---|
| 177 | value = escapeq(value) | 
|---|
| 178 | return value | 
|---|
| 179 | except KeyError: | 
|---|
| 180 | pass | 
|---|
| 181 | return '' | 
|---|
| 182 |  | 
|---|
| 183 | class UserInput: | 
|---|
| 184 |  | 
|---|
| 185 | def __init__(self): | 
|---|
| 186 | self.__form = cgi.FieldStorage() | 
|---|
| 187 | #log("\n\nbody: " + self.body) | 
|---|
| 188 |  | 
|---|
| 189 | def __getattr__(self, name): | 
|---|
| 190 | if name[0] == '_': | 
|---|
| 191 | raise AttributeError | 
|---|
| 192 | try: | 
|---|
| 193 | value = self.__form[name].value | 
|---|
| 194 | except (TypeError, KeyError): | 
|---|
| 195 | value = '' | 
|---|
| 196 | else: | 
|---|
| 197 | value = value.strip() | 
|---|
| 198 | setattr(self, name, value) | 
|---|
| 199 | return value | 
|---|
| 200 |  | 
|---|
| 201 | def __getitem__(self, key): | 
|---|
| 202 | return getattr(self, key) | 
|---|
| 203 |  | 
|---|
| 204 | class FaqEntry: | 
|---|
| 205 |  | 
|---|
| 206 | def __init__(self, fp, file, sec_num): | 
|---|
| 207 | self.file = file | 
|---|
| 208 | self.sec, self.num = sec_num | 
|---|
| 209 | if fp: | 
|---|
| 210 | import rfc822 | 
|---|
| 211 | self.__headers = rfc822.Message(fp) | 
|---|
| 212 | self.body = fp.read().strip() | 
|---|
| 213 | else: | 
|---|
| 214 | self.__headers = {'title': "%d.%d. " % sec_num} | 
|---|
| 215 | self.body = '' | 
|---|
| 216 |  | 
|---|
| 217 | def __getattr__(self, name): | 
|---|
| 218 | if name[0] == '_': | 
|---|
| 219 | raise AttributeError | 
|---|
| 220 | key = '-'.join(name.split('_')) | 
|---|
| 221 | try: | 
|---|
| 222 | value = self.__headers[key] | 
|---|
| 223 | except KeyError: | 
|---|
| 224 | value = '' | 
|---|
| 225 | setattr(self, name, value) | 
|---|
| 226 | return value | 
|---|
| 227 |  | 
|---|
| 228 | def __getitem__(self, key): | 
|---|
| 229 | return getattr(self, key) | 
|---|
| 230 |  | 
|---|
| 231 | def load_version(self): | 
|---|
| 232 | command = interpolate(SH_RLOG_H, self) | 
|---|
| 233 | p = os.popen(command) | 
|---|
| 234 | version = '' | 
|---|
| 235 | while 1: | 
|---|
| 236 | line = p.readline() | 
|---|
| 237 | if not line: | 
|---|
| 238 | break | 
|---|
| 239 | if line[:5] == 'head:': | 
|---|
| 240 | version = line[5:].strip() | 
|---|
| 241 | p.close() | 
|---|
| 242 | self.version = version | 
|---|
| 243 |  | 
|---|
| 244 | def getmtime(self): | 
|---|
| 245 | if not self.last_changed_date: | 
|---|
| 246 | return 0 | 
|---|
| 247 | try: | 
|---|
| 248 | return os.stat(self.file)[stat.ST_MTIME] | 
|---|
| 249 | except os.error: | 
|---|
| 250 | return 0 | 
|---|
| 251 |  | 
|---|
| 252 | def emit_marks(self): | 
|---|
| 253 | mtime = self.getmtime() | 
|---|
| 254 | if mtime >= now - DT_VERY_RECENT: | 
|---|
| 255 | emit(MARK_VERY_RECENT, self) | 
|---|
| 256 | elif mtime >= now - DT_RECENT: | 
|---|
| 257 | emit(MARK_RECENT, self) | 
|---|
| 258 |  | 
|---|
| 259 | def show(self, edit=1): | 
|---|
| 260 | emit(ENTRY_HEADER1, self) | 
|---|
| 261 | self.emit_marks() | 
|---|
| 262 | emit(ENTRY_HEADER2, self) | 
|---|
| 263 | pre = 0 | 
|---|
| 264 | raw = 0 | 
|---|
| 265 | for line in self.body.split('\n'): | 
|---|
| 266 | # Allow the user to insert raw html into a FAQ answer | 
|---|
| 267 | # (Skip Montanaro, with changes by Guido) | 
|---|
| 268 | tag = line.rstrip().lower() | 
|---|
| 269 | if tag == '<html>': | 
|---|
| 270 | raw = 1 | 
|---|
| 271 | continue | 
|---|
| 272 | if tag == '</html>': | 
|---|
| 273 | raw = 0 | 
|---|
| 274 | continue | 
|---|
| 275 | if raw: | 
|---|
| 276 | print line | 
|---|
| 277 | continue | 
|---|
| 278 | if not line.strip(): | 
|---|
| 279 | if pre: | 
|---|
| 280 | print '</PRE>' | 
|---|
| 281 | pre = 0 | 
|---|
| 282 | else: | 
|---|
| 283 | print '<P>' | 
|---|
| 284 | else: | 
|---|
| 285 | if not line[0].isspace(): | 
|---|
| 286 | if pre: | 
|---|
| 287 | print '</PRE>' | 
|---|
| 288 | pre = 0 | 
|---|
| 289 | else: | 
|---|
| 290 | if not pre: | 
|---|
| 291 | print '<PRE>' | 
|---|
| 292 | pre = 1 | 
|---|
| 293 | if '/' in line or '@' in line: | 
|---|
| 294 | line = translate(line, pre) | 
|---|
| 295 | elif '<' in line or '&' in line: | 
|---|
| 296 | line = escape(line) | 
|---|
| 297 | if not pre and '*' in line: | 
|---|
| 298 | line = emphasize(line) | 
|---|
| 299 | print line | 
|---|
| 300 | if pre: | 
|---|
| 301 | print '</PRE>' | 
|---|
| 302 | pre = 0 | 
|---|
| 303 | if edit: | 
|---|
| 304 | print '<P>' | 
|---|
| 305 | emit(ENTRY_FOOTER, self) | 
|---|
| 306 | if self.last_changed_date: | 
|---|
| 307 | emit(ENTRY_LOGINFO, self) | 
|---|
| 308 | print '<P>' | 
|---|
| 309 |  | 
|---|
| 310 | class FaqDir: | 
|---|
| 311 |  | 
|---|
| 312 | entryclass = FaqEntry | 
|---|
| 313 |  | 
|---|
| 314 | __okprog = re.compile(OKFILENAME) | 
|---|
| 315 |  | 
|---|
| 316 | def __init__(self, dir=os.curdir): | 
|---|
| 317 | self.__dir = dir | 
|---|
| 318 | self.__files = None | 
|---|
| 319 |  | 
|---|
| 320 | def __fill(self): | 
|---|
| 321 | if self.__files is not None: | 
|---|
| 322 | return | 
|---|
| 323 | self.__files = files = [] | 
|---|
| 324 | okprog = self.__okprog | 
|---|
| 325 | for file in os.listdir(self.__dir): | 
|---|
| 326 | if self.__okprog.match(file): | 
|---|
| 327 | files.append(file) | 
|---|
| 328 | files.sort() | 
|---|
| 329 |  | 
|---|
| 330 | def good(self, file): | 
|---|
| 331 | return self.__okprog.match(file) | 
|---|
| 332 |  | 
|---|
| 333 | def parse(self, file): | 
|---|
| 334 | m = self.good(file) | 
|---|
| 335 | if not m: | 
|---|
| 336 | return None | 
|---|
| 337 | sec, num = m.group(1, 2) | 
|---|
| 338 | return int(sec), int(num) | 
|---|
| 339 |  | 
|---|
| 340 | def list(self): | 
|---|
| 341 | # XXX Caller shouldn't modify result | 
|---|
| 342 | self.__fill() | 
|---|
| 343 | return self.__files | 
|---|
| 344 |  | 
|---|
| 345 | def open(self, file): | 
|---|
| 346 | sec_num = self.parse(file) | 
|---|
| 347 | if not sec_num: | 
|---|
| 348 | raise InvalidFile(file) | 
|---|
| 349 | try: | 
|---|
| 350 | fp = open(file) | 
|---|
| 351 | except IOError, msg: | 
|---|
| 352 | raise NoSuchFile(file, msg) | 
|---|
| 353 | try: | 
|---|
| 354 | return self.entryclass(fp, file, sec_num) | 
|---|
| 355 | finally: | 
|---|
| 356 | fp.close() | 
|---|
| 357 |  | 
|---|
| 358 | def show(self, file, edit=1): | 
|---|
| 359 | self.open(file).show(edit=edit) | 
|---|
| 360 |  | 
|---|
| 361 | def new(self, section): | 
|---|
| 362 | if not SECTION_TITLES.has_key(section): | 
|---|
| 363 | raise NoSuchSection(section) | 
|---|
| 364 | maxnum = 0 | 
|---|
| 365 | for file in self.list(): | 
|---|
| 366 | sec, num = self.parse(file) | 
|---|
| 367 | if sec == section: | 
|---|
| 368 | maxnum = max(maxnum, num) | 
|---|
| 369 | sec_num = (section, maxnum+1) | 
|---|
| 370 | file = NEWFILENAME % sec_num | 
|---|
| 371 | return self.entryclass(None, file, sec_num) | 
|---|
| 372 |  | 
|---|
| 373 | class FaqWizard: | 
|---|
| 374 |  | 
|---|
| 375 | def __init__(self): | 
|---|
| 376 | self.ui = UserInput() | 
|---|
| 377 | self.dir = FaqDir() | 
|---|
| 378 |  | 
|---|
| 379 | def go(self): | 
|---|
| 380 | print 'Content-type: text/html' | 
|---|
| 381 | req = self.ui.req or 'home' | 
|---|
| 382 | mname = 'do_%s' % req | 
|---|
| 383 | try: | 
|---|
| 384 | meth = getattr(self, mname) | 
|---|
| 385 | except AttributeError: | 
|---|
| 386 | self.error("Bad request type %r." % (req,)) | 
|---|
| 387 | else: | 
|---|
| 388 | try: | 
|---|
| 389 | meth() | 
|---|
| 390 | except InvalidFile, exc: | 
|---|
| 391 | self.error("Invalid entry file name %s" % exc.file) | 
|---|
| 392 | except NoSuchFile, exc: | 
|---|
| 393 | self.error("No entry with file name %s" % exc.file) | 
|---|
| 394 | except NoSuchSection, exc: | 
|---|
| 395 | self.error("No section number %s" % exc.section) | 
|---|
| 396 | self.epilogue() | 
|---|
| 397 |  | 
|---|
| 398 | def error(self, message, **kw): | 
|---|
| 399 | self.prologue(T_ERROR) | 
|---|
| 400 | emit(message, kw) | 
|---|
| 401 |  | 
|---|
| 402 | def prologue(self, title, entry=None, **kw): | 
|---|
| 403 | emit(PROLOGUE, entry, kwdict=kw, title=escape(title)) | 
|---|
| 404 |  | 
|---|
| 405 | def epilogue(self): | 
|---|
| 406 | emit(EPILOGUE) | 
|---|
| 407 |  | 
|---|
| 408 | def do_home(self): | 
|---|
| 409 | self.prologue(T_HOME) | 
|---|
| 410 | emit(HOME) | 
|---|
| 411 |  | 
|---|
| 412 | def do_debug(self): | 
|---|
| 413 | self.prologue("FAQ Wizard Debugging") | 
|---|
| 414 | form = cgi.FieldStorage() | 
|---|
| 415 | cgi.print_form(form) | 
|---|
| 416 | cgi.print_environ(os.environ) | 
|---|
| 417 | cgi.print_directory() | 
|---|
| 418 | cgi.print_arguments() | 
|---|
| 419 |  | 
|---|
| 420 | def do_search(self): | 
|---|
| 421 | query = self.ui.query | 
|---|
| 422 | if not query: | 
|---|
| 423 | self.error("Empty query string!") | 
|---|
| 424 | return | 
|---|
| 425 | if self.ui.querytype == 'simple': | 
|---|
| 426 | query = re.escape(query) | 
|---|
| 427 | queries = [query] | 
|---|
| 428 | elif self.ui.querytype in ('anykeywords', 'allkeywords'): | 
|---|
| 429 | words = filter(None, re.split('\W+', query)) | 
|---|
| 430 | if not words: | 
|---|
| 431 | self.error("No keywords specified!") | 
|---|
| 432 | return | 
|---|
| 433 | words = map(lambda w: r'\b%s\b' % w, words) | 
|---|
| 434 | if self.ui.querytype[:3] == 'any': | 
|---|
| 435 | queries = ['|'.join(words)] | 
|---|
| 436 | else: | 
|---|
| 437 | # Each of the individual queries must match | 
|---|
| 438 | queries = words | 
|---|
| 439 | else: | 
|---|
| 440 | # Default to regular expression | 
|---|
| 441 | queries = [query] | 
|---|
| 442 | self.prologue(T_SEARCH) | 
|---|
| 443 | progs = [] | 
|---|
| 444 | for query in queries: | 
|---|
| 445 | if self.ui.casefold == 'no': | 
|---|
| 446 | p = re.compile(query) | 
|---|
| 447 | else: | 
|---|
| 448 | p = re.compile(query, re.IGNORECASE) | 
|---|
| 449 | progs.append(p) | 
|---|
| 450 | hits = [] | 
|---|
| 451 | for file in self.dir.list(): | 
|---|
| 452 | try: | 
|---|
| 453 | entry = self.dir.open(file) | 
|---|
| 454 | except FileError: | 
|---|
| 455 | constants | 
|---|
| 456 | for p in progs: | 
|---|
| 457 | if not p.search(entry.title) and not p.search(entry.body): | 
|---|
| 458 | break | 
|---|
| 459 | else: | 
|---|
| 460 | hits.append(file) | 
|---|
| 461 | if not hits: | 
|---|
| 462 | emit(NO_HITS, self.ui, count=0) | 
|---|
| 463 | elif len(hits) <= MAXHITS: | 
|---|
| 464 | if len(hits) == 1: | 
|---|
| 465 | emit(ONE_HIT, count=1) | 
|---|
| 466 | else: | 
|---|
| 467 | emit(FEW_HITS, count=len(hits)) | 
|---|
| 468 | self.format_all(hits, headers=0) | 
|---|
| 469 | else: | 
|---|
| 470 | emit(MANY_HITS, count=len(hits)) | 
|---|
| 471 | self.format_index(hits) | 
|---|
| 472 |  | 
|---|
| 473 | def do_all(self): | 
|---|
| 474 | self.prologue(T_ALL) | 
|---|
| 475 | files = self.dir.list() | 
|---|
| 476 | self.last_changed(files) | 
|---|
| 477 | self.format_index(files, localrefs=1) | 
|---|
| 478 | self.format_all(files) | 
|---|
| 479 |  | 
|---|
| 480 | def do_compat(self): | 
|---|
| 481 | files = self.dir.list() | 
|---|
| 482 | emit(COMPAT) | 
|---|
| 483 | self.last_changed(files) | 
|---|
| 484 | self.format_index(files, localrefs=1) | 
|---|
| 485 | self.format_all(files, edit=0) | 
|---|
| 486 | sys.exit(0)                     # XXX Hack to suppress epilogue | 
|---|
| 487 |  | 
|---|
| 488 | def last_changed(self, files): | 
|---|
| 489 | latest = 0 | 
|---|
| 490 | for file in files: | 
|---|
| 491 | entry = self.dir.open(file) | 
|---|
| 492 | if entry: | 
|---|
| 493 | mtime = mtime = entry.getmtime() | 
|---|
| 494 | if mtime > latest: | 
|---|
| 495 | latest = mtime | 
|---|
| 496 | print time.strftime(LAST_CHANGED, time.localtime(latest)) | 
|---|
| 497 | emit(EXPLAIN_MARKS) | 
|---|
| 498 |  | 
|---|
| 499 | def format_all(self, files, edit=1, headers=1): | 
|---|
| 500 | sec = 0 | 
|---|
| 501 | for file in files: | 
|---|
| 502 | try: | 
|---|
| 503 | entry = self.dir.open(file) | 
|---|
| 504 | except NoSuchFile: | 
|---|
| 505 | continue | 
|---|
| 506 | if headers and entry.sec != sec: | 
|---|
| 507 | sec = entry.sec | 
|---|
| 508 | try: | 
|---|
| 509 | title = SECTION_TITLES[sec] | 
|---|
| 510 | except KeyError: | 
|---|
| 511 | title = "Untitled" | 
|---|
| 512 | emit("\n<HR>\n<H1>%(sec)s. %(title)s</H1>\n", | 
|---|
| 513 | sec=sec, title=title) | 
|---|
| 514 | entry.show(edit=edit) | 
|---|
| 515 |  | 
|---|
| 516 | def do_index(self): | 
|---|
| 517 | self.prologue(T_INDEX) | 
|---|
| 518 | files = self.dir.list() | 
|---|
| 519 | self.last_changed(files) | 
|---|
| 520 | self.format_index(files, add=1) | 
|---|
| 521 |  | 
|---|
| 522 | def format_index(self, files, add=0, localrefs=0): | 
|---|
| 523 | sec = 0 | 
|---|
| 524 | for file in files: | 
|---|
| 525 | try: | 
|---|
| 526 | entry = self.dir.open(file) | 
|---|
| 527 | except NoSuchFile: | 
|---|
| 528 | continue | 
|---|
| 529 | if entry.sec != sec: | 
|---|
| 530 | if sec: | 
|---|
| 531 | if add: | 
|---|
| 532 | emit(INDEX_ADDSECTION, sec=sec) | 
|---|
| 533 | emit(INDEX_ENDSECTION, sec=sec) | 
|---|
| 534 | sec = entry.sec | 
|---|
| 535 | try: | 
|---|
| 536 | title = SECTION_TITLES[sec] | 
|---|
| 537 | except KeyError: | 
|---|
| 538 | title = "Untitled" | 
|---|
| 539 | emit(INDEX_SECTION, sec=sec, title=title) | 
|---|
| 540 | if localrefs: | 
|---|
| 541 | emit(LOCAL_ENTRY, entry) | 
|---|
| 542 | else: | 
|---|
| 543 | emit(INDEX_ENTRY, entry) | 
|---|
| 544 | entry.emit_marks() | 
|---|
| 545 | if sec: | 
|---|
| 546 | if add: | 
|---|
| 547 | emit(INDEX_ADDSECTION, sec=sec) | 
|---|
| 548 | emit(INDEX_ENDSECTION, sec=sec) | 
|---|
| 549 |  | 
|---|
| 550 | def do_recent(self): | 
|---|
| 551 | if not self.ui.days: | 
|---|
| 552 | days = 1 | 
|---|
| 553 | else: | 
|---|
| 554 | days = float(self.ui.days) | 
|---|
| 555 | try: | 
|---|
| 556 | cutoff = now - days * 24 * 3600 | 
|---|
| 557 | except OverflowError: | 
|---|
| 558 | cutoff = 0 | 
|---|
| 559 | list = [] | 
|---|
| 560 | for file in self.dir.list(): | 
|---|
| 561 | entry = self.dir.open(file) | 
|---|
| 562 | if not entry: | 
|---|
| 563 | continue | 
|---|
| 564 | mtime = entry.getmtime() | 
|---|
| 565 | if mtime >= cutoff: | 
|---|
| 566 | list.append((mtime, file)) | 
|---|
| 567 | list.sort() | 
|---|
| 568 | list.reverse() | 
|---|
| 569 | self.prologue(T_RECENT) | 
|---|
| 570 | if days <= 1: | 
|---|
| 571 | period = "%.2g hours" % (days*24) | 
|---|
| 572 | else: | 
|---|
| 573 | period = "%.6g days" % days | 
|---|
| 574 | if not list: | 
|---|
| 575 | emit(NO_RECENT, period=period) | 
|---|
| 576 | elif len(list) == 1: | 
|---|
| 577 | emit(ONE_RECENT, period=period) | 
|---|
| 578 | else: | 
|---|
| 579 | emit(SOME_RECENT, period=period, count=len(list)) | 
|---|
| 580 | self.format_all(map(lambda (mtime, file): file, list), headers=0) | 
|---|
| 581 | emit(TAIL_RECENT) | 
|---|
| 582 |  | 
|---|
| 583 | def do_roulette(self): | 
|---|
| 584 | import random | 
|---|
| 585 | files = self.dir.list() | 
|---|
| 586 | if not files: | 
|---|
| 587 | self.error("No entries.") | 
|---|
| 588 | return | 
|---|
| 589 | file = random.choice(files) | 
|---|
| 590 | self.prologue(T_ROULETTE) | 
|---|
| 591 | emit(ROULETTE) | 
|---|
| 592 | self.dir.show(file) | 
|---|
| 593 |  | 
|---|
| 594 | def do_help(self): | 
|---|
| 595 | self.prologue(T_HELP) | 
|---|
| 596 | emit(HELP) | 
|---|
| 597 |  | 
|---|
| 598 | def do_show(self): | 
|---|
| 599 | entry = self.dir.open(self.ui.file) | 
|---|
| 600 | self.prologue(T_SHOW) | 
|---|
| 601 | entry.show() | 
|---|
| 602 |  | 
|---|
| 603 | def do_add(self): | 
|---|
| 604 | self.prologue(T_ADD) | 
|---|
| 605 | emit(ADD_HEAD) | 
|---|
| 606 | sections = SECTION_TITLES.items() | 
|---|
| 607 | sections.sort() | 
|---|
| 608 | for section, title in sections: | 
|---|
| 609 | emit(ADD_SECTION, section=section, title=title) | 
|---|
| 610 | emit(ADD_TAIL) | 
|---|
| 611 |  | 
|---|
| 612 | def do_delete(self): | 
|---|
| 613 | self.prologue(T_DELETE) | 
|---|
| 614 | emit(DELETE) | 
|---|
| 615 |  | 
|---|
| 616 | def do_log(self): | 
|---|
| 617 | entry = self.dir.open(self.ui.file) | 
|---|
| 618 | self.prologue(T_LOG, entry) | 
|---|
| 619 | emit(LOG, entry) | 
|---|
| 620 | self.rlog(interpolate(SH_RLOG, entry), entry) | 
|---|
| 621 |  | 
|---|
| 622 | def rlog(self, command, entry=None): | 
|---|
| 623 | output = os.popen(command).read() | 
|---|
| 624 | sys.stdout.write('<PRE>') | 
|---|
| 625 | athead = 0 | 
|---|
| 626 | lines = output.split('\n') | 
|---|
| 627 | while lines and not lines[-1]: | 
|---|
| 628 | del lines[-1] | 
|---|
| 629 | if lines: | 
|---|
| 630 | line = lines[-1] | 
|---|
| 631 | if line[:1] == '=' and len(line) >= 40 and \ | 
|---|
| 632 | line == line[0]*len(line): | 
|---|
| 633 | del lines[-1] | 
|---|
| 634 | headrev = None | 
|---|
| 635 | for line in lines: | 
|---|
| 636 | if entry and athead and line[:9] == 'revision ': | 
|---|
| 637 | rev = line[9:].split() | 
|---|
| 638 | mami = revparse(rev) | 
|---|
| 639 | if not mami: | 
|---|
| 640 | print line | 
|---|
| 641 | else: | 
|---|
| 642 | emit(REVISIONLINK, entry, rev=rev, line=line) | 
|---|
| 643 | if mami[1] > 1: | 
|---|
| 644 | prev = "%d.%d" % (mami[0], mami[1]-1) | 
|---|
| 645 | emit(DIFFLINK, entry, prev=prev, rev=rev) | 
|---|
| 646 | if headrev: | 
|---|
| 647 | emit(DIFFLINK, entry, prev=rev, rev=headrev) | 
|---|
| 648 | else: | 
|---|
| 649 | headrev = rev | 
|---|
| 650 | print | 
|---|
| 651 | athead = 0 | 
|---|
| 652 | else: | 
|---|
| 653 | athead = 0 | 
|---|
| 654 | if line[:1] == '-' and len(line) >= 20 and \ | 
|---|
| 655 | line == len(line) * line[0]: | 
|---|
| 656 | athead = 1 | 
|---|
| 657 | sys.stdout.write('<HR>') | 
|---|
| 658 | else: | 
|---|
| 659 | print line | 
|---|
| 660 | print '</PRE>' | 
|---|
| 661 |  | 
|---|
| 662 | def do_revision(self): | 
|---|
| 663 | entry = self.dir.open(self.ui.file) | 
|---|
| 664 | rev = self.ui.rev | 
|---|
| 665 | mami = revparse(rev) | 
|---|
| 666 | if not mami: | 
|---|
| 667 | self.error("Invalid revision number: %r." % (rev,)) | 
|---|
| 668 | self.prologue(T_REVISION, entry) | 
|---|
| 669 | self.shell(interpolate(SH_REVISION, entry, rev=rev)) | 
|---|
| 670 |  | 
|---|
| 671 | def do_diff(self): | 
|---|
| 672 | entry = self.dir.open(self.ui.file) | 
|---|
| 673 | prev = self.ui.prev | 
|---|
| 674 | rev = self.ui.rev | 
|---|
| 675 | mami = revparse(rev) | 
|---|
| 676 | if not mami: | 
|---|
| 677 | self.error("Invalid revision number: %r." % (rev,)) | 
|---|
| 678 | if prev: | 
|---|
| 679 | if not revparse(prev): | 
|---|
| 680 | self.error("Invalid previous revision number: %r." % (prev,)) | 
|---|
| 681 | else: | 
|---|
| 682 | prev = '%d.%d' % (mami[0], mami[1]) | 
|---|
| 683 | self.prologue(T_DIFF, entry) | 
|---|
| 684 | self.shell(interpolate(SH_RDIFF, entry, rev=rev, prev=prev)) | 
|---|
| 685 |  | 
|---|
| 686 | def shell(self, command): | 
|---|
| 687 | output = os.popen(command).read() | 
|---|
| 688 | sys.stdout.write('<PRE>') | 
|---|
| 689 | print escape(output) | 
|---|
| 690 | print '</PRE>' | 
|---|
| 691 |  | 
|---|
| 692 | def do_new(self): | 
|---|
| 693 | entry = self.dir.new(section=int(self.ui.section)) | 
|---|
| 694 | entry.version = '*new*' | 
|---|
| 695 | self.prologue(T_EDIT) | 
|---|
| 696 | emit(EDITHEAD) | 
|---|
| 697 | emit(EDITFORM1, entry, editversion=entry.version) | 
|---|
| 698 | emit(EDITFORM2, entry, load_my_cookie()) | 
|---|
| 699 | emit(EDITFORM3) | 
|---|
| 700 | entry.show(edit=0) | 
|---|
| 701 |  | 
|---|
| 702 | def do_edit(self): | 
|---|
| 703 | entry = self.dir.open(self.ui.file) | 
|---|
| 704 | entry.load_version() | 
|---|
| 705 | self.prologue(T_EDIT) | 
|---|
| 706 | emit(EDITHEAD) | 
|---|
| 707 | emit(EDITFORM1, entry, editversion=entry.version) | 
|---|
| 708 | emit(EDITFORM2, entry, load_my_cookie()) | 
|---|
| 709 | emit(EDITFORM3) | 
|---|
| 710 | entry.show(edit=0) | 
|---|
| 711 |  | 
|---|
| 712 | def do_review(self): | 
|---|
| 713 | send_my_cookie(self.ui) | 
|---|
| 714 | if self.ui.editversion == '*new*': | 
|---|
| 715 | sec, num = self.dir.parse(self.ui.file) | 
|---|
| 716 | entry = self.dir.new(section=sec) | 
|---|
| 717 | entry.version = "*new*" | 
|---|
| 718 | if entry.file != self.ui.file: | 
|---|
| 719 | self.error("Commit version conflict!") | 
|---|
| 720 | emit(NEWCONFLICT, self.ui, sec=sec, num=num) | 
|---|
| 721 | return | 
|---|
| 722 | else: | 
|---|
| 723 | entry = self.dir.open(self.ui.file) | 
|---|
| 724 | entry.load_version() | 
|---|
| 725 | # Check that the FAQ entry number didn't change | 
|---|
| 726 | if self.ui.title.split()[:1] != entry.title.split()[:1]: | 
|---|
| 727 | self.error("Don't change the entry number please!") | 
|---|
| 728 | return | 
|---|
| 729 | # Check that the edited version is the current version | 
|---|
| 730 | if entry.version != self.ui.editversion: | 
|---|
| 731 | self.error("Commit version conflict!") | 
|---|
| 732 | emit(VERSIONCONFLICT, entry, self.ui) | 
|---|
| 733 | return | 
|---|
| 734 | commit_ok = ((not PASSWORD | 
|---|
| 735 | or self.ui.password == PASSWORD) | 
|---|
| 736 | and self.ui.author | 
|---|
| 737 | and '@' in self.ui.email | 
|---|
| 738 | and self.ui.log) | 
|---|
| 739 | if self.ui.commit: | 
|---|
| 740 | if not commit_ok: | 
|---|
| 741 | self.cantcommit() | 
|---|
| 742 | else: | 
|---|
| 743 | self.commit(entry) | 
|---|
| 744 | return | 
|---|
| 745 | self.prologue(T_REVIEW) | 
|---|
| 746 | emit(REVIEWHEAD) | 
|---|
| 747 | entry.body = self.ui.body | 
|---|
| 748 | entry.title = self.ui.title | 
|---|
| 749 | entry.show(edit=0) | 
|---|
| 750 | emit(EDITFORM1, self.ui, entry) | 
|---|
| 751 | if commit_ok: | 
|---|
| 752 | emit(COMMIT) | 
|---|
| 753 | else: | 
|---|
| 754 | emit(NOCOMMIT_HEAD) | 
|---|
| 755 | self.errordetail() | 
|---|
| 756 | emit(NOCOMMIT_TAIL) | 
|---|
| 757 | emit(EDITFORM2, self.ui, entry, load_my_cookie()) | 
|---|
| 758 | emit(EDITFORM3) | 
|---|
| 759 |  | 
|---|
| 760 | def cantcommit(self): | 
|---|
| 761 | self.prologue(T_CANTCOMMIT) | 
|---|
| 762 | print CANTCOMMIT_HEAD | 
|---|
| 763 | self.errordetail() | 
|---|
| 764 | print CANTCOMMIT_TAIL | 
|---|
| 765 |  | 
|---|
| 766 | def errordetail(self): | 
|---|
| 767 | if PASSWORD and self.ui.password != PASSWORD: | 
|---|
| 768 | emit(NEED_PASSWD) | 
|---|
| 769 | if not self.ui.log: | 
|---|
| 770 | emit(NEED_LOG) | 
|---|
| 771 | if not self.ui.author: | 
|---|
| 772 | emit(NEED_AUTHOR) | 
|---|
| 773 | if not self.ui.email: | 
|---|
| 774 | emit(NEED_EMAIL) | 
|---|
| 775 |  | 
|---|
| 776 | def commit(self, entry): | 
|---|
| 777 | file = entry.file | 
|---|
| 778 | # Normalize line endings in body | 
|---|
| 779 | if '\r' in self.ui.body: | 
|---|
| 780 | self.ui.body = re.sub('\r\n?', '\n', self.ui.body) | 
|---|
| 781 | # Normalize whitespace in title | 
|---|
| 782 | self.ui.title = ' '.join(self.ui.title.split()) | 
|---|
| 783 | # Check that there were any changes | 
|---|
| 784 | if self.ui.body == entry.body and self.ui.title == entry.title: | 
|---|
| 785 | self.error("You didn't make any changes!") | 
|---|
| 786 | return | 
|---|
| 787 |  | 
|---|
| 788 | # need to lock here because otherwise the file exists and is not writable (on NT) | 
|---|
| 789 | command = interpolate(SH_LOCK, file=file) | 
|---|
| 790 | p = os.popen(command) | 
|---|
| 791 | output = p.read() | 
|---|
| 792 |  | 
|---|
| 793 | try: | 
|---|
| 794 | os.unlink(file) | 
|---|
| 795 | except os.error: | 
|---|
| 796 | pass | 
|---|
| 797 | try: | 
|---|
| 798 | f = open(file, 'w') | 
|---|
| 799 | except IOError, why: | 
|---|
| 800 | self.error(CANTWRITE, file=file, why=why) | 
|---|
| 801 | return | 
|---|
| 802 | date = time.ctime(now) | 
|---|
| 803 | emit(FILEHEADER, self.ui, os.environ, date=date, _file=f, _quote=0) | 
|---|
| 804 | f.write('\n') | 
|---|
| 805 | f.write(self.ui.body) | 
|---|
| 806 | f.write('\n') | 
|---|
| 807 | f.close() | 
|---|
| 808 |  | 
|---|
| 809 | import tempfile | 
|---|
| 810 | tf = tempfile.NamedTemporaryFile() | 
|---|
| 811 | emit(LOGHEADER, self.ui, os.environ, date=date, _file=tf) | 
|---|
| 812 | tf.flush() | 
|---|
| 813 | tf.seek(0) | 
|---|
| 814 |  | 
|---|
| 815 | command = interpolate(SH_CHECKIN, file=file, tfn=tf.name) | 
|---|
| 816 | log("\n\n" + command) | 
|---|
| 817 | p = os.popen(command) | 
|---|
| 818 | output = p.read() | 
|---|
| 819 | sts = p.close() | 
|---|
| 820 | log("output: " + output) | 
|---|
| 821 | log("done: " + str(sts)) | 
|---|
| 822 | log("TempFile:\n" + tf.read() + "end") | 
|---|
| 823 |  | 
|---|
| 824 | if not sts: | 
|---|
| 825 | self.prologue(T_COMMITTED) | 
|---|
| 826 | emit(COMMITTED) | 
|---|
| 827 | else: | 
|---|
| 828 | self.error(T_COMMITFAILED) | 
|---|
| 829 | emit(COMMITFAILED, sts=sts) | 
|---|
| 830 | print '<PRE>%s</PRE>' % escape(output) | 
|---|
| 831 |  | 
|---|
| 832 | try: | 
|---|
| 833 | os.unlink(tf.name) | 
|---|
| 834 | except os.error: | 
|---|
| 835 | pass | 
|---|
| 836 |  | 
|---|
| 837 | entry = self.dir.open(file) | 
|---|
| 838 | entry.show() | 
|---|
| 839 |  | 
|---|
| 840 | wiz = FaqWizard() | 
|---|
| 841 | wiz.go() | 
|---|