Added documentation for WatchedFileHandler (based on SF patch #1598415)
[python.git] / Lib / pydoc.py
blob94927d02706939996dcc2566e758fb627e54a1aa
1 #!/usr/bin/env python
2 # -*- coding: Latin-1 -*-
3 """Generate Python documentation in HTML or text for interactive use.
5 In the Python interpreter, do "from pydoc import help" to provide online
6 help. Calling help(thing) on a Python object documents the object.
8 Or, at the shell command line outside of Python:
10 Run "pydoc <name>" to show documentation on something. <name> may be
11 the name of a function, module, package, or a dotted reference to a
12 class or function within a module or module in a package. If the
13 argument contains a path segment delimiter (e.g. slash on Unix,
14 backslash on Windows) it is treated as the path to a Python source file.
16 Run "pydoc -k <keyword>" to search for a keyword in the synopsis lines
17 of all available modules.
19 Run "pydoc -p <port>" to start an HTTP server on a given port on the
20 local machine to generate documentation web pages.
22 For platforms without a command line, "pydoc -g" starts the HTTP server
23 and also pops up a little window for controlling it.
25 Run "pydoc -w <name>" to write out the HTML documentation for a module
26 to a file named "<name>.html".
28 Module docs for core modules are assumed to be in
30 http://www.python.org/doc/current/lib/
32 This can be overridden by setting the PYTHONDOCS environment variable
33 to a different URL or to a local directory containing the Library
34 Reference Manual pages.
35 """
37 __author__ = "Ka-Ping Yee <ping@lfw.org>"
38 __date__ = "26 February 2001"
40 __version__ = "$Revision$"
41 __credits__ = """Guido van Rossum, for an excellent programming language.
42 Tommy Burnette, the original creator of manpy.
43 Paul Prescod, for all his work on onlinehelp.
44 Richard Chamberlain, for the first implementation of textdoc.
45 """
47 # Known bugs that can't be fixed here:
48 # - imp.load_module() cannot be prevented from clobbering existing
49 # loaded modules, so calling synopsis() on a binary module file
50 # changes the contents of any existing module with the same name.
51 # - If the __file__ attribute on a module is a relative path and
52 # the current directory is changed with os.chdir(), an incorrect
53 # path will be displayed.
55 import sys, imp, os, re, types, inspect, __builtin__, pkgutil
56 from repr import Repr
57 from string import expandtabs, find, join, lower, split, strip, rfind, rstrip
58 try:
59 from collections import deque
60 except ImportError:
61 # Python 2.3 compatibility
62 class deque(list):
63 def popleft(self):
64 return self.pop(0)
66 # --------------------------------------------------------- common routines
68 def pathdirs():
69 """Convert sys.path into a list of absolute, existing, unique paths."""
70 dirs = []
71 normdirs = []
72 for dir in sys.path:
73 dir = os.path.abspath(dir or '.')
74 normdir = os.path.normcase(dir)
75 if normdir not in normdirs and os.path.isdir(dir):
76 dirs.append(dir)
77 normdirs.append(normdir)
78 return dirs
80 def getdoc(object):
81 """Get the doc string or comments for an object."""
82 result = inspect.getdoc(object) or inspect.getcomments(object)
83 return result and re.sub('^ *\n', '', rstrip(result)) or ''
85 def splitdoc(doc):
86 """Split a doc string into a synopsis line (if any) and the rest."""
87 lines = split(strip(doc), '\n')
88 if len(lines) == 1:
89 return lines[0], ''
90 elif len(lines) >= 2 and not rstrip(lines[1]):
91 return lines[0], join(lines[2:], '\n')
92 return '', join(lines, '\n')
94 def classname(object, modname):
95 """Get a class name and qualify it with a module name if necessary."""
96 name = object.__name__
97 if object.__module__ != modname:
98 name = object.__module__ + '.' + name
99 return name
101 def isdata(object):
102 """Check if an object is of a type that probably means it's data."""
103 return not (inspect.ismodule(object) or inspect.isclass(object) or
104 inspect.isroutine(object) or inspect.isframe(object) or
105 inspect.istraceback(object) or inspect.iscode(object))
107 def replace(text, *pairs):
108 """Do a series of global replacements on a string."""
109 while pairs:
110 text = join(split(text, pairs[0]), pairs[1])
111 pairs = pairs[2:]
112 return text
114 def cram(text, maxlen):
115 """Omit part of a string if needed to make it fit in a maximum length."""
116 if len(text) > maxlen:
117 pre = max(0, (maxlen-3)//2)
118 post = max(0, maxlen-3-pre)
119 return text[:pre] + '...' + text[len(text)-post:]
120 return text
122 _re_stripid = re.compile(r' at 0x[0-9a-f]{6,16}(>+)$', re.IGNORECASE)
123 def stripid(text):
124 """Remove the hexadecimal id from a Python object representation."""
125 # The behaviour of %p is implementation-dependent in terms of case.
126 if _re_stripid.search(repr(Exception)):
127 return _re_stripid.sub(r'\1', text)
128 return text
130 def _is_some_method(obj):
131 return inspect.ismethod(obj) or inspect.ismethoddescriptor(obj)
133 def allmethods(cl):
134 methods = {}
135 for key, value in inspect.getmembers(cl, _is_some_method):
136 methods[key] = 1
137 for base in cl.__bases__:
138 methods.update(allmethods(base)) # all your base are belong to us
139 for key in methods.keys():
140 methods[key] = getattr(cl, key)
141 return methods
143 def _split_list(s, predicate):
144 """Split sequence s via predicate, and return pair ([true], [false]).
146 The return value is a 2-tuple of lists,
147 ([x for x in s if predicate(x)],
148 [x for x in s if not predicate(x)])
151 yes = []
152 no = []
153 for x in s:
154 if predicate(x):
155 yes.append(x)
156 else:
157 no.append(x)
158 return yes, no
160 def visiblename(name, all=None):
161 """Decide whether to show documentation on a variable."""
162 # Certain special names are redundant.
163 if name in ('__builtins__', '__doc__', '__file__', '__path__',
164 '__module__', '__name__', '__slots__'): return 0
165 # Private names are hidden, but special names are displayed.
166 if name.startswith('__') and name.endswith('__'): return 1
167 if all is not None:
168 # only document that which the programmer exported in __all__
169 return name in all
170 else:
171 return not name.startswith('_')
173 def classify_class_attrs(object):
174 """Wrap inspect.classify_class_attrs, with fixup for data descriptors."""
175 def fixup((name, kind, cls, value)):
176 if inspect.isdatadescriptor(value):
177 kind = 'data descriptor'
178 return name, kind, cls, value
179 return map(fixup, inspect.classify_class_attrs(object))
181 # ----------------------------------------------------- module manipulation
183 def ispackage(path):
184 """Guess whether a path refers to a package directory."""
185 if os.path.isdir(path):
186 for ext in ('.py', '.pyc', '.pyo'):
187 if os.path.isfile(os.path.join(path, '__init__' + ext)):
188 return True
189 return False
191 def source_synopsis(file):
192 line = file.readline()
193 while line[:1] == '#' or not strip(line):
194 line = file.readline()
195 if not line: break
196 line = strip(line)
197 if line[:4] == 'r"""': line = line[1:]
198 if line[:3] == '"""':
199 line = line[3:]
200 if line[-1:] == '\\': line = line[:-1]
201 while not strip(line):
202 line = file.readline()
203 if not line: break
204 result = strip(split(line, '"""')[0])
205 else: result = None
206 return result
208 def synopsis(filename, cache={}):
209 """Get the one-line summary out of a module file."""
210 mtime = os.stat(filename).st_mtime
211 lastupdate, result = cache.get(filename, (0, None))
212 if lastupdate < mtime:
213 info = inspect.getmoduleinfo(filename)
214 try:
215 file = open(filename)
216 except IOError:
217 # module can't be opened, so skip it
218 return None
219 if info and 'b' in info[2]: # binary modules have to be imported
220 try: module = imp.load_module('__temp__', file, filename, info[1:])
221 except: return None
222 result = (module.__doc__ or '').splitlines()[0]
223 del sys.modules['__temp__']
224 else: # text modules can be directly examined
225 result = source_synopsis(file)
226 file.close()
227 cache[filename] = (mtime, result)
228 return result
230 class ErrorDuringImport(Exception):
231 """Errors that occurred while trying to import something to document it."""
232 def __init__(self, filename, (exc, value, tb)):
233 self.filename = filename
234 self.exc = exc
235 self.value = value
236 self.tb = tb
238 def __str__(self):
239 exc = self.exc
240 if type(exc) is types.ClassType:
241 exc = exc.__name__
242 return 'problem in %s - %s: %s' % (self.filename, exc, self.value)
244 def importfile(path):
245 """Import a Python source file or compiled file given its path."""
246 magic = imp.get_magic()
247 file = open(path, 'r')
248 if file.read(len(magic)) == magic:
249 kind = imp.PY_COMPILED
250 else:
251 kind = imp.PY_SOURCE
252 file.close()
253 filename = os.path.basename(path)
254 name, ext = os.path.splitext(filename)
255 file = open(path, 'r')
256 try:
257 module = imp.load_module(name, file, path, (ext, 'r', kind))
258 except:
259 raise ErrorDuringImport(path, sys.exc_info())
260 file.close()
261 return module
263 def safeimport(path, forceload=0, cache={}):
264 """Import a module; handle errors; return None if the module isn't found.
266 If the module *is* found but an exception occurs, it's wrapped in an
267 ErrorDuringImport exception and reraised. Unlike __import__, if a
268 package path is specified, the module at the end of the path is returned,
269 not the package at the beginning. If the optional 'forceload' argument
270 is 1, we reload the module from disk (unless it's a dynamic extension)."""
271 try:
272 # If forceload is 1 and the module has been previously loaded from
273 # disk, we always have to reload the module. Checking the file's
274 # mtime isn't good enough (e.g. the module could contain a class
275 # that inherits from another module that has changed).
276 if forceload and path in sys.modules:
277 if path not in sys.builtin_module_names:
278 # Avoid simply calling reload() because it leaves names in
279 # the currently loaded module lying around if they're not
280 # defined in the new source file. Instead, remove the
281 # module from sys.modules and re-import. Also remove any
282 # submodules because they won't appear in the newly loaded
283 # module's namespace if they're already in sys.modules.
284 subs = [m for m in sys.modules if m.startswith(path + '.')]
285 for key in [path] + subs:
286 # Prevent garbage collection.
287 cache[key] = sys.modules[key]
288 del sys.modules[key]
289 module = __import__(path)
290 except:
291 # Did the error occur before or after the module was found?
292 (exc, value, tb) = info = sys.exc_info()
293 if path in sys.modules:
294 # An error occurred while executing the imported module.
295 raise ErrorDuringImport(sys.modules[path].__file__, info)
296 elif exc is SyntaxError:
297 # A SyntaxError occurred before we could execute the module.
298 raise ErrorDuringImport(value.filename, info)
299 elif exc is ImportError and \
300 split(lower(str(value)))[:2] == ['no', 'module']:
301 # The module was not found.
302 return None
303 else:
304 # Some other error occurred during the importing process.
305 raise ErrorDuringImport(path, sys.exc_info())
306 for part in split(path, '.')[1:]:
307 try: module = getattr(module, part)
308 except AttributeError: return None
309 return module
311 # ---------------------------------------------------- formatter base class
313 class Doc:
314 def document(self, object, name=None, *args):
315 """Generate documentation for an object."""
316 args = (object, name) + args
317 # 'try' clause is to attempt to handle the possibility that inspect
318 # identifies something in a way that pydoc itself has issues handling;
319 # think 'super' and how it is a descriptor (which raises the exception
320 # by lacking a __name__ attribute) and an instance.
321 if inspect.isgetsetdescriptor(object): return self.docdata(*args)
322 if inspect.ismemberdescriptor(object): return self.docdata(*args)
323 try:
324 if inspect.ismodule(object): return self.docmodule(*args)
325 if inspect.isclass(object): return self.docclass(*args)
326 if inspect.isroutine(object): return self.docroutine(*args)
327 except AttributeError:
328 pass
329 if isinstance(object, property): return self.docproperty(*args)
330 return self.docother(*args)
332 def fail(self, object, name=None, *args):
333 """Raise an exception for unimplemented types."""
334 message = "don't know how to document object%s of type %s" % (
335 name and ' ' + repr(name), type(object).__name__)
336 raise TypeError, message
338 docmodule = docclass = docroutine = docother = docproperty = docdata = fail
340 def getdocloc(self, object):
341 """Return the location of module docs or None"""
343 try:
344 file = inspect.getabsfile(object)
345 except TypeError:
346 file = '(built-in)'
348 docloc = os.environ.get("PYTHONDOCS",
349 "http://www.python.org/doc/current/lib")
350 basedir = os.path.join(sys.exec_prefix, "lib",
351 "python"+sys.version[0:3])
352 if (isinstance(object, type(os)) and
353 (object.__name__ in ('errno', 'exceptions', 'gc', 'imp',
354 'marshal', 'posix', 'signal', 'sys',
355 'thread', 'zipimport') or
356 (file.startswith(basedir) and
357 not file.startswith(os.path.join(basedir, 'site-packages'))))):
358 htmlfile = "module-%s.html" % object.__name__
359 if docloc.startswith("http://"):
360 docloc = "%s/%s" % (docloc.rstrip("/"), htmlfile)
361 else:
362 docloc = os.path.join(docloc, htmlfile)
363 else:
364 docloc = None
365 return docloc
367 # -------------------------------------------- HTML documentation generator
369 class HTMLRepr(Repr):
370 """Class for safely making an HTML representation of a Python object."""
371 def __init__(self):
372 Repr.__init__(self)
373 self.maxlist = self.maxtuple = 20
374 self.maxdict = 10
375 self.maxstring = self.maxother = 100
377 def escape(self, text):
378 return replace(text, '&', '&amp;', '<', '&lt;', '>', '&gt;')
380 def repr(self, object):
381 return Repr.repr(self, object)
383 def repr1(self, x, level):
384 if hasattr(type(x), '__name__'):
385 methodname = 'repr_' + join(split(type(x).__name__), '_')
386 if hasattr(self, methodname):
387 return getattr(self, methodname)(x, level)
388 return self.escape(cram(stripid(repr(x)), self.maxother))
390 def repr_string(self, x, level):
391 test = cram(x, self.maxstring)
392 testrepr = repr(test)
393 if '\\' in test and '\\' not in replace(testrepr, r'\\', ''):
394 # Backslashes are only literal in the string and are never
395 # needed to make any special characters, so show a raw string.
396 return 'r' + testrepr[0] + self.escape(test) + testrepr[0]
397 return re.sub(r'((\\[\\abfnrtv\'"]|\\[0-9]..|\\x..|\\u....)+)',
398 r'<font color="#c040c0">\1</font>',
399 self.escape(testrepr))
401 repr_str = repr_string
403 def repr_instance(self, x, level):
404 try:
405 return self.escape(cram(stripid(repr(x)), self.maxstring))
406 except:
407 return self.escape('<%s instance>' % x.__class__.__name__)
409 repr_unicode = repr_string
411 class HTMLDoc(Doc):
412 """Formatter class for HTML documentation."""
414 # ------------------------------------------- HTML formatting utilities
416 _repr_instance = HTMLRepr()
417 repr = _repr_instance.repr
418 escape = _repr_instance.escape
420 def page(self, title, contents):
421 """Format an HTML page."""
422 return '''
423 <!doctype html PUBLIC "-//W3C//DTD HTML 4.0 Transitional//EN">
424 <html><head><title>Python: %s</title>
425 </head><body bgcolor="#f0f0f8">
427 </body></html>''' % (title, contents)
429 def heading(self, title, fgcol, bgcol, extras=''):
430 """Format a page heading."""
431 return '''
432 <table width="100%%" cellspacing=0 cellpadding=2 border=0 summary="heading">
433 <tr bgcolor="%s">
434 <td valign=bottom>&nbsp;<br>
435 <font color="%s" face="helvetica, arial">&nbsp;<br>%s</font></td
436 ><td align=right valign=bottom
437 ><font color="%s" face="helvetica, arial">%s</font></td></tr></table>
438 ''' % (bgcol, fgcol, title, fgcol, extras or '&nbsp;')
440 def section(self, title, fgcol, bgcol, contents, width=6,
441 prelude='', marginalia=None, gap='&nbsp;'):
442 """Format a section with a heading."""
443 if marginalia is None:
444 marginalia = '<tt>' + '&nbsp;' * width + '</tt>'
445 result = '''<p>
446 <table width="100%%" cellspacing=0 cellpadding=2 border=0 summary="section">
447 <tr bgcolor="%s">
448 <td colspan=3 valign=bottom>&nbsp;<br>
449 <font color="%s" face="helvetica, arial">%s</font></td></tr>
450 ''' % (bgcol, fgcol, title)
451 if prelude:
452 result = result + '''
453 <tr bgcolor="%s"><td rowspan=2>%s</td>
454 <td colspan=2>%s</td></tr>
455 <tr><td>%s</td>''' % (bgcol, marginalia, prelude, gap)
456 else:
457 result = result + '''
458 <tr><td bgcolor="%s">%s</td><td>%s</td>''' % (bgcol, marginalia, gap)
460 return result + '\n<td width="100%%">%s</td></tr></table>' % contents
462 def bigsection(self, title, *args):
463 """Format a section with a big heading."""
464 title = '<big><strong>%s</strong></big>' % title
465 return self.section(title, *args)
467 def preformat(self, text):
468 """Format literal preformatted text."""
469 text = self.escape(expandtabs(text))
470 return replace(text, '\n\n', '\n \n', '\n\n', '\n \n',
471 ' ', '&nbsp;', '\n', '<br>\n')
473 def multicolumn(self, list, format, cols=4):
474 """Format a list of items into a multi-column list."""
475 result = ''
476 rows = (len(list)+cols-1)/cols
477 for col in range(cols):
478 result = result + '<td width="%d%%" valign=top>' % (100/cols)
479 for i in range(rows*col, rows*col+rows):
480 if i < len(list):
481 result = result + format(list[i]) + '<br>\n'
482 result = result + '</td>'
483 return '<table width="100%%" summary="list"><tr>%s</tr></table>' % result
485 def grey(self, text): return '<font color="#909090">%s</font>' % text
487 def namelink(self, name, *dicts):
488 """Make a link for an identifier, given name-to-URL mappings."""
489 for dict in dicts:
490 if name in dict:
491 return '<a href="%s">%s</a>' % (dict[name], name)
492 return name
494 def classlink(self, object, modname):
495 """Make a link for a class."""
496 name, module = object.__name__, sys.modules.get(object.__module__)
497 if hasattr(module, name) and getattr(module, name) is object:
498 return '<a href="%s.html#%s">%s</a>' % (
499 module.__name__, name, classname(object, modname))
500 return classname(object, modname)
502 def modulelink(self, object):
503 """Make a link for a module."""
504 return '<a href="%s.html">%s</a>' % (object.__name__, object.__name__)
506 def modpkglink(self, (name, path, ispackage, shadowed)):
507 """Make a link for a module or package to display in an index."""
508 if shadowed:
509 return self.grey(name)
510 if path:
511 url = '%s.%s.html' % (path, name)
512 else:
513 url = '%s.html' % name
514 if ispackage:
515 text = '<strong>%s</strong>&nbsp;(package)' % name
516 else:
517 text = name
518 return '<a href="%s">%s</a>' % (url, text)
520 def markup(self, text, escape=None, funcs={}, classes={}, methods={}):
521 """Mark up some plain text, given a context of symbols to look for.
522 Each context dictionary maps object names to anchor names."""
523 escape = escape or self.escape
524 results = []
525 here = 0
526 pattern = re.compile(r'\b((http|ftp)://\S+[\w/]|'
527 r'RFC[- ]?(\d+)|'
528 r'PEP[- ]?(\d+)|'
529 r'(self\.)?(\w+))')
530 while True:
531 match = pattern.search(text, here)
532 if not match: break
533 start, end = match.span()
534 results.append(escape(text[here:start]))
536 all, scheme, rfc, pep, selfdot, name = match.groups()
537 if scheme:
538 url = escape(all).replace('"', '&quot;')
539 results.append('<a href="%s">%s</a>' % (url, url))
540 elif rfc:
541 url = 'http://www.rfc-editor.org/rfc/rfc%d.txt' % int(rfc)
542 results.append('<a href="%s">%s</a>' % (url, escape(all)))
543 elif pep:
544 url = 'http://www.python.org/peps/pep-%04d.html' % int(pep)
545 results.append('<a href="%s">%s</a>' % (url, escape(all)))
546 elif text[end:end+1] == '(':
547 results.append(self.namelink(name, methods, funcs, classes))
548 elif selfdot:
549 results.append('self.<strong>%s</strong>' % name)
550 else:
551 results.append(self.namelink(name, classes))
552 here = end
553 results.append(escape(text[here:]))
554 return join(results, '')
556 # ---------------------------------------------- type-specific routines
558 def formattree(self, tree, modname, parent=None):
559 """Produce HTML for a class tree as given by inspect.getclasstree()."""
560 result = ''
561 for entry in tree:
562 if type(entry) is type(()):
563 c, bases = entry
564 result = result + '<dt><font face="helvetica, arial">'
565 result = result + self.classlink(c, modname)
566 if bases and bases != (parent,):
567 parents = []
568 for base in bases:
569 parents.append(self.classlink(base, modname))
570 result = result + '(' + join(parents, ', ') + ')'
571 result = result + '\n</font></dt>'
572 elif type(entry) is type([]):
573 result = result + '<dd>\n%s</dd>\n' % self.formattree(
574 entry, modname, c)
575 return '<dl>\n%s</dl>\n' % result
577 def docmodule(self, object, name=None, mod=None, *ignored):
578 """Produce HTML documentation for a module object."""
579 name = object.__name__ # ignore the passed-in name
580 try:
581 all = object.__all__
582 except AttributeError:
583 all = None
584 parts = split(name, '.')
585 links = []
586 for i in range(len(parts)-1):
587 links.append(
588 '<a href="%s.html"><font color="#ffffff">%s</font></a>' %
589 (join(parts[:i+1], '.'), parts[i]))
590 linkedname = join(links + parts[-1:], '.')
591 head = '<big><big><strong>%s</strong></big></big>' % linkedname
592 try:
593 path = inspect.getabsfile(object)
594 url = path
595 if sys.platform == 'win32':
596 import nturl2path
597 url = nturl2path.pathname2url(path)
598 filelink = '<a href="file:%s">%s</a>' % (url, path)
599 except TypeError:
600 filelink = '(built-in)'
601 info = []
602 if hasattr(object, '__version__'):
603 version = str(object.__version__)
604 if version[:11] == '$' + 'Revision: ' and version[-1:] == '$':
605 version = strip(version[11:-1])
606 info.append('version %s' % self.escape(version))
607 if hasattr(object, '__date__'):
608 info.append(self.escape(str(object.__date__)))
609 if info:
610 head = head + ' (%s)' % join(info, ', ')
611 docloc = self.getdocloc(object)
612 if docloc is not None:
613 docloc = '<br><a href="%(docloc)s">Module Docs</a>' % locals()
614 else:
615 docloc = ''
616 result = self.heading(
617 head, '#ffffff', '#7799ee',
618 '<a href=".">index</a><br>' + filelink + docloc)
620 modules = inspect.getmembers(object, inspect.ismodule)
622 classes, cdict = [], {}
623 for key, value in inspect.getmembers(object, inspect.isclass):
624 # if __all__ exists, believe it. Otherwise use old heuristic.
625 if (all is not None or
626 (inspect.getmodule(value) or object) is object):
627 if visiblename(key, all):
628 classes.append((key, value))
629 cdict[key] = cdict[value] = '#' + key
630 for key, value in classes:
631 for base in value.__bases__:
632 key, modname = base.__name__, base.__module__
633 module = sys.modules.get(modname)
634 if modname != name and module and hasattr(module, key):
635 if getattr(module, key) is base:
636 if not key in cdict:
637 cdict[key] = cdict[base] = modname + '.html#' + key
638 funcs, fdict = [], {}
639 for key, value in inspect.getmembers(object, inspect.isroutine):
640 # if __all__ exists, believe it. Otherwise use old heuristic.
641 if (all is not None or
642 inspect.isbuiltin(value) or inspect.getmodule(value) is object):
643 if visiblename(key, all):
644 funcs.append((key, value))
645 fdict[key] = '#-' + key
646 if inspect.isfunction(value): fdict[value] = fdict[key]
647 data = []
648 for key, value in inspect.getmembers(object, isdata):
649 if visiblename(key, all):
650 data.append((key, value))
652 doc = self.markup(getdoc(object), self.preformat, fdict, cdict)
653 doc = doc and '<tt>%s</tt>' % doc
654 result = result + '<p>%s</p>\n' % doc
656 if hasattr(object, '__path__'):
657 modpkgs = []
658 for importer, modname, ispkg in pkgutil.iter_modules(object.__path__):
659 modpkgs.append((modname, name, ispkg, 0))
660 modpkgs.sort()
661 contents = self.multicolumn(modpkgs, self.modpkglink)
662 result = result + self.bigsection(
663 'Package Contents', '#ffffff', '#aa55cc', contents)
664 elif modules:
665 contents = self.multicolumn(
666 modules, lambda (key, value), s=self: s.modulelink(value))
667 result = result + self.bigsection(
668 'Modules', '#fffff', '#aa55cc', contents)
670 if classes:
671 classlist = map(lambda (key, value): value, classes)
672 contents = [
673 self.formattree(inspect.getclasstree(classlist, 1), name)]
674 for key, value in classes:
675 contents.append(self.document(value, key, name, fdict, cdict))
676 result = result + self.bigsection(
677 'Classes', '#ffffff', '#ee77aa', join(contents))
678 if funcs:
679 contents = []
680 for key, value in funcs:
681 contents.append(self.document(value, key, name, fdict, cdict))
682 result = result + self.bigsection(
683 'Functions', '#ffffff', '#eeaa77', join(contents))
684 if data:
685 contents = []
686 for key, value in data:
687 contents.append(self.document(value, key))
688 result = result + self.bigsection(
689 'Data', '#ffffff', '#55aa55', join(contents, '<br>\n'))
690 if hasattr(object, '__author__'):
691 contents = self.markup(str(object.__author__), self.preformat)
692 result = result + self.bigsection(
693 'Author', '#ffffff', '#7799ee', contents)
694 if hasattr(object, '__credits__'):
695 contents = self.markup(str(object.__credits__), self.preformat)
696 result = result + self.bigsection(
697 'Credits', '#ffffff', '#7799ee', contents)
699 return result
701 def docclass(self, object, name=None, mod=None, funcs={}, classes={},
702 *ignored):
703 """Produce HTML documentation for a class object."""
704 realname = object.__name__
705 name = name or realname
706 bases = object.__bases__
708 contents = []
709 push = contents.append
711 # Cute little class to pump out a horizontal rule between sections.
712 class HorizontalRule:
713 def __init__(self):
714 self.needone = 0
715 def maybe(self):
716 if self.needone:
717 push('<hr>\n')
718 self.needone = 1
719 hr = HorizontalRule()
721 # List the mro, if non-trivial.
722 mro = deque(inspect.getmro(object))
723 if len(mro) > 2:
724 hr.maybe()
725 push('<dl><dt>Method resolution order:</dt>\n')
726 for base in mro:
727 push('<dd>%s</dd>\n' % self.classlink(base,
728 object.__module__))
729 push('</dl>\n')
731 def spill(msg, attrs, predicate):
732 ok, attrs = _split_list(attrs, predicate)
733 if ok:
734 hr.maybe()
735 push(msg)
736 for name, kind, homecls, value in ok:
737 push(self.document(getattr(object, name), name, mod,
738 funcs, classes, mdict, object))
739 push('\n')
740 return attrs
742 def spilldescriptors(msg, attrs, predicate):
743 ok, attrs = _split_list(attrs, predicate)
744 if ok:
745 hr.maybe()
746 push(msg)
747 for name, kind, homecls, value in ok:
748 push(self._docdescriptor(name, value, mod))
749 return attrs
751 def spilldata(msg, attrs, predicate):
752 ok, attrs = _split_list(attrs, predicate)
753 if ok:
754 hr.maybe()
755 push(msg)
756 for name, kind, homecls, value in ok:
757 base = self.docother(getattr(object, name), name, mod)
758 if callable(value) or inspect.isdatadescriptor(value):
759 doc = getattr(value, "__doc__", None)
760 else:
761 doc = None
762 if doc is None:
763 push('<dl><dt>%s</dl>\n' % base)
764 else:
765 doc = self.markup(getdoc(value), self.preformat,
766 funcs, classes, mdict)
767 doc = '<dd><tt>%s</tt>' % doc
768 push('<dl><dt>%s%s</dl>\n' % (base, doc))
769 push('\n')
770 return attrs
772 attrs = filter(lambda (name, kind, cls, value): visiblename(name),
773 classify_class_attrs(object))
774 mdict = {}
775 for key, kind, homecls, value in attrs:
776 mdict[key] = anchor = '#' + name + '-' + key
777 value = getattr(object, key)
778 try:
779 # The value may not be hashable (e.g., a data attr with
780 # a dict or list value).
781 mdict[value] = anchor
782 except TypeError:
783 pass
785 while attrs:
786 if mro:
787 thisclass = mro.popleft()
788 else:
789 thisclass = attrs[0][2]
790 attrs, inherited = _split_list(attrs, lambda t: t[2] is thisclass)
792 if thisclass is __builtin__.object:
793 attrs = inherited
794 continue
795 elif thisclass is object:
796 tag = 'defined here'
797 else:
798 tag = 'inherited from %s' % self.classlink(thisclass,
799 object.__module__)
800 tag += ':<br>\n'
802 # Sort attrs by name.
803 try:
804 attrs.sort(key=lambda t: t[0])
805 except TypeError:
806 attrs.sort(lambda t1, t2: cmp(t1[0], t2[0])) # 2.3 compat
808 # Pump out the attrs, segregated by kind.
809 attrs = spill('Methods %s' % tag, attrs,
810 lambda t: t[1] == 'method')
811 attrs = spill('Class methods %s' % tag, attrs,
812 lambda t: t[1] == 'class method')
813 attrs = spill('Static methods %s' % tag, attrs,
814 lambda t: t[1] == 'static method')
815 attrs = spilldescriptors('Data descriptors %s' % tag, attrs,
816 lambda t: t[1] == 'data descriptor')
817 attrs = spilldata('Data and other attributes %s' % tag, attrs,
818 lambda t: t[1] == 'data')
819 assert attrs == []
820 attrs = inherited
822 contents = ''.join(contents)
824 if name == realname:
825 title = '<a name="%s">class <strong>%s</strong></a>' % (
826 name, realname)
827 else:
828 title = '<strong>%s</strong> = <a name="%s">class %s</a>' % (
829 name, name, realname)
830 if bases:
831 parents = []
832 for base in bases:
833 parents.append(self.classlink(base, object.__module__))
834 title = title + '(%s)' % join(parents, ', ')
835 doc = self.markup(getdoc(object), self.preformat, funcs, classes, mdict)
836 doc = doc and '<tt>%s<br>&nbsp;</tt>' % doc
838 return self.section(title, '#000000', '#ffc8d8', contents, 3, doc)
840 def formatvalue(self, object):
841 """Format an argument default value as text."""
842 return self.grey('=' + self.repr(object))
844 def docroutine(self, object, name=None, mod=None,
845 funcs={}, classes={}, methods={}, cl=None):
846 """Produce HTML documentation for a function or method object."""
847 realname = object.__name__
848 name = name or realname
849 anchor = (cl and cl.__name__ or '') + '-' + name
850 note = ''
851 skipdocs = 0
852 if inspect.ismethod(object):
853 imclass = object.im_class
854 if cl:
855 if imclass is not cl:
856 note = ' from ' + self.classlink(imclass, mod)
857 else:
858 if object.im_self:
859 note = ' method of %s instance' % self.classlink(
860 object.im_self.__class__, mod)
861 else:
862 note = ' unbound %s method' % self.classlink(imclass,mod)
863 object = object.im_func
865 if name == realname:
866 title = '<a name="%s"><strong>%s</strong></a>' % (anchor, realname)
867 else:
868 if (cl and realname in cl.__dict__ and
869 cl.__dict__[realname] is object):
870 reallink = '<a href="#%s">%s</a>' % (
871 cl.__name__ + '-' + realname, realname)
872 skipdocs = 1
873 else:
874 reallink = realname
875 title = '<a name="%s"><strong>%s</strong></a> = %s' % (
876 anchor, name, reallink)
877 if inspect.isfunction(object):
878 args, varargs, varkw, defaults = inspect.getargspec(object)
879 argspec = inspect.formatargspec(
880 args, varargs, varkw, defaults, formatvalue=self.formatvalue)
881 if realname == '<lambda>':
882 title = '<strong>%s</strong> <em>lambda</em> ' % name
883 argspec = argspec[1:-1] # remove parentheses
884 else:
885 argspec = '(...)'
887 decl = title + argspec + (note and self.grey(
888 '<font face="helvetica, arial">%s</font>' % note))
890 if skipdocs:
891 return '<dl><dt>%s</dt></dl>\n' % decl
892 else:
893 doc = self.markup(
894 getdoc(object), self.preformat, funcs, classes, methods)
895 doc = doc and '<dd><tt>%s</tt></dd>' % doc
896 return '<dl><dt>%s</dt>%s</dl>\n' % (decl, doc)
898 def _docdescriptor(self, name, value, mod):
899 results = []
900 push = results.append
902 if name:
903 push('<dl><dt><strong>%s</strong></dt>\n' % name)
904 if value.__doc__ is not None:
905 doc = self.markup(getdoc(value), self.preformat)
906 push('<dd><tt>%s</tt></dd>\n' % doc)
907 push('</dl>\n')
909 return ''.join(results)
911 def docproperty(self, object, name=None, mod=None, cl=None):
912 """Produce html documentation for a property."""
913 return self._docdescriptor(name, object, mod)
915 def docother(self, object, name=None, mod=None, *ignored):
916 """Produce HTML documentation for a data object."""
917 lhs = name and '<strong>%s</strong> = ' % name or ''
918 return lhs + self.repr(object)
920 def docdata(self, object, name=None, mod=None, cl=None):
921 """Produce html documentation for a data descriptor."""
922 return self._docdescriptor(name, object, mod)
924 def index(self, dir, shadowed=None):
925 """Generate an HTML index for a directory of modules."""
926 modpkgs = []
927 if shadowed is None: shadowed = {}
928 for importer, name, ispkg in pkgutil.iter_modules([dir]):
929 modpkgs.append((name, '', ispkg, name in shadowed))
930 shadowed[name] = 1
932 modpkgs.sort()
933 contents = self.multicolumn(modpkgs, self.modpkglink)
934 return self.bigsection(dir, '#ffffff', '#ee77aa', contents)
936 # -------------------------------------------- text documentation generator
938 class TextRepr(Repr):
939 """Class for safely making a text representation of a Python object."""
940 def __init__(self):
941 Repr.__init__(self)
942 self.maxlist = self.maxtuple = 20
943 self.maxdict = 10
944 self.maxstring = self.maxother = 100
946 def repr1(self, x, level):
947 if hasattr(type(x), '__name__'):
948 methodname = 'repr_' + join(split(type(x).__name__), '_')
949 if hasattr(self, methodname):
950 return getattr(self, methodname)(x, level)
951 return cram(stripid(repr(x)), self.maxother)
953 def repr_string(self, x, level):
954 test = cram(x, self.maxstring)
955 testrepr = repr(test)
956 if '\\' in test and '\\' not in replace(testrepr, r'\\', ''):
957 # Backslashes are only literal in the string and are never
958 # needed to make any special characters, so show a raw string.
959 return 'r' + testrepr[0] + test + testrepr[0]
960 return testrepr
962 repr_str = repr_string
964 def repr_instance(self, x, level):
965 try:
966 return cram(stripid(repr(x)), self.maxstring)
967 except:
968 return '<%s instance>' % x.__class__.__name__
970 class TextDoc(Doc):
971 """Formatter class for text documentation."""
973 # ------------------------------------------- text formatting utilities
975 _repr_instance = TextRepr()
976 repr = _repr_instance.repr
978 def bold(self, text):
979 """Format a string in bold by overstriking."""
980 return join(map(lambda ch: ch + '\b' + ch, text), '')
982 def indent(self, text, prefix=' '):
983 """Indent text by prepending a given prefix to each line."""
984 if not text: return ''
985 lines = split(text, '\n')
986 lines = map(lambda line, prefix=prefix: prefix + line, lines)
987 if lines: lines[-1] = rstrip(lines[-1])
988 return join(lines, '\n')
990 def section(self, title, contents):
991 """Format a section with a given heading."""
992 return self.bold(title) + '\n' + rstrip(self.indent(contents)) + '\n\n'
994 # ---------------------------------------------- type-specific routines
996 def formattree(self, tree, modname, parent=None, prefix=''):
997 """Render in text a class tree as returned by inspect.getclasstree()."""
998 result = ''
999 for entry in tree:
1000 if type(entry) is type(()):
1001 c, bases = entry
1002 result = result + prefix + classname(c, modname)
1003 if bases and bases != (parent,):
1004 parents = map(lambda c, m=modname: classname(c, m), bases)
1005 result = result + '(%s)' % join(parents, ', ')
1006 result = result + '\n'
1007 elif type(entry) is type([]):
1008 result = result + self.formattree(
1009 entry, modname, c, prefix + ' ')
1010 return result
1012 def docmodule(self, object, name=None, mod=None):
1013 """Produce text documentation for a given module object."""
1014 name = object.__name__ # ignore the passed-in name
1015 synop, desc = splitdoc(getdoc(object))
1016 result = self.section('NAME', name + (synop and ' - ' + synop))
1018 try:
1019 all = object.__all__
1020 except AttributeError:
1021 all = None
1023 try:
1024 file = inspect.getabsfile(object)
1025 except TypeError:
1026 file = '(built-in)'
1027 result = result + self.section('FILE', file)
1029 docloc = self.getdocloc(object)
1030 if docloc is not None:
1031 result = result + self.section('MODULE DOCS', docloc)
1033 if desc:
1034 result = result + self.section('DESCRIPTION', desc)
1036 classes = []
1037 for key, value in inspect.getmembers(object, inspect.isclass):
1038 # if __all__ exists, believe it. Otherwise use old heuristic.
1039 if (all is not None
1040 or (inspect.getmodule(value) or object) is object):
1041 if visiblename(key, all):
1042 classes.append((key, value))
1043 funcs = []
1044 for key, value in inspect.getmembers(object, inspect.isroutine):
1045 # if __all__ exists, believe it. Otherwise use old heuristic.
1046 if (all is not None or
1047 inspect.isbuiltin(value) or inspect.getmodule(value) is object):
1048 if visiblename(key, all):
1049 funcs.append((key, value))
1050 data = []
1051 for key, value in inspect.getmembers(object, isdata):
1052 if visiblename(key, all):
1053 data.append((key, value))
1055 if hasattr(object, '__path__'):
1056 modpkgs = []
1057 for importer, modname, ispkg in pkgutil.iter_modules(object.__path__):
1058 if ispkg:
1059 modpkgs.append(modname + ' (package)')
1060 else:
1061 modpkgs.append(modname)
1063 modpkgs.sort()
1064 result = result + self.section(
1065 'PACKAGE CONTENTS', join(modpkgs, '\n'))
1067 if classes:
1068 classlist = map(lambda (key, value): value, classes)
1069 contents = [self.formattree(
1070 inspect.getclasstree(classlist, 1), name)]
1071 for key, value in classes:
1072 contents.append(self.document(value, key, name))
1073 result = result + self.section('CLASSES', join(contents, '\n'))
1075 if funcs:
1076 contents = []
1077 for key, value in funcs:
1078 contents.append(self.document(value, key, name))
1079 result = result + self.section('FUNCTIONS', join(contents, '\n'))
1081 if data:
1082 contents = []
1083 for key, value in data:
1084 contents.append(self.docother(value, key, name, maxlen=70))
1085 result = result + self.section('DATA', join(contents, '\n'))
1087 if hasattr(object, '__version__'):
1088 version = str(object.__version__)
1089 if version[:11] == '$' + 'Revision: ' and version[-1:] == '$':
1090 version = strip(version[11:-1])
1091 result = result + self.section('VERSION', version)
1092 if hasattr(object, '__date__'):
1093 result = result + self.section('DATE', str(object.__date__))
1094 if hasattr(object, '__author__'):
1095 result = result + self.section('AUTHOR', str(object.__author__))
1096 if hasattr(object, '__credits__'):
1097 result = result + self.section('CREDITS', str(object.__credits__))
1098 return result
1100 def docclass(self, object, name=None, mod=None):
1101 """Produce text documentation for a given class object."""
1102 realname = object.__name__
1103 name = name or realname
1104 bases = object.__bases__
1106 def makename(c, m=object.__module__):
1107 return classname(c, m)
1109 if name == realname:
1110 title = 'class ' + self.bold(realname)
1111 else:
1112 title = self.bold(name) + ' = class ' + realname
1113 if bases:
1114 parents = map(makename, bases)
1115 title = title + '(%s)' % join(parents, ', ')
1117 doc = getdoc(object)
1118 contents = doc and [doc + '\n'] or []
1119 push = contents.append
1121 # List the mro, if non-trivial.
1122 mro = deque(inspect.getmro(object))
1123 if len(mro) > 2:
1124 push("Method resolution order:")
1125 for base in mro:
1126 push(' ' + makename(base))
1127 push('')
1129 # Cute little class to pump out a horizontal rule between sections.
1130 class HorizontalRule:
1131 def __init__(self):
1132 self.needone = 0
1133 def maybe(self):
1134 if self.needone:
1135 push('-' * 70)
1136 self.needone = 1
1137 hr = HorizontalRule()
1139 def spill(msg, attrs, predicate):
1140 ok, attrs = _split_list(attrs, predicate)
1141 if ok:
1142 hr.maybe()
1143 push(msg)
1144 for name, kind, homecls, value in ok:
1145 push(self.document(getattr(object, name),
1146 name, mod, object))
1147 return attrs
1149 def spilldescriptors(msg, attrs, predicate):
1150 ok, attrs = _split_list(attrs, predicate)
1151 if ok:
1152 hr.maybe()
1153 push(msg)
1154 for name, kind, homecls, value in ok:
1155 push(self._docdescriptor(name, value, mod))
1156 return attrs
1158 def spilldata(msg, attrs, predicate):
1159 ok, attrs = _split_list(attrs, predicate)
1160 if ok:
1161 hr.maybe()
1162 push(msg)
1163 for name, kind, homecls, value in ok:
1164 if callable(value) or inspect.isdatadescriptor(value):
1165 doc = getdoc(value)
1166 else:
1167 doc = None
1168 push(self.docother(getattr(object, name),
1169 name, mod, maxlen=70, doc=doc) + '\n')
1170 return attrs
1172 attrs = filter(lambda (name, kind, cls, value): visiblename(name),
1173 classify_class_attrs(object))
1174 while attrs:
1175 if mro:
1176 thisclass = mro.popleft()
1177 else:
1178 thisclass = attrs[0][2]
1179 attrs, inherited = _split_list(attrs, lambda t: t[2] is thisclass)
1181 if thisclass is __builtin__.object:
1182 attrs = inherited
1183 continue
1184 elif thisclass is object:
1185 tag = "defined here"
1186 else:
1187 tag = "inherited from %s" % classname(thisclass,
1188 object.__module__)
1189 filter(lambda t: not t[0].startswith('_'), attrs)
1191 # Sort attrs by name.
1192 attrs.sort()
1194 # Pump out the attrs, segregated by kind.
1195 attrs = spill("Methods %s:\n" % tag, attrs,
1196 lambda t: t[1] == 'method')
1197 attrs = spill("Class methods %s:\n" % tag, attrs,
1198 lambda t: t[1] == 'class method')
1199 attrs = spill("Static methods %s:\n" % tag, attrs,
1200 lambda t: t[1] == 'static method')
1201 attrs = spilldescriptors("Data descriptors %s:\n" % tag, attrs,
1202 lambda t: t[1] == 'data descriptor')
1203 attrs = spilldata("Data and other attributes %s:\n" % tag, attrs,
1204 lambda t: t[1] == 'data')
1205 assert attrs == []
1206 attrs = inherited
1208 contents = '\n'.join(contents)
1209 if not contents:
1210 return title + '\n'
1211 return title + '\n' + self.indent(rstrip(contents), ' | ') + '\n'
1213 def formatvalue(self, object):
1214 """Format an argument default value as text."""
1215 return '=' + self.repr(object)
1217 def docroutine(self, object, name=None, mod=None, cl=None):
1218 """Produce text documentation for a function or method object."""
1219 realname = object.__name__
1220 name = name or realname
1221 note = ''
1222 skipdocs = 0
1223 if inspect.ismethod(object):
1224 imclass = object.im_class
1225 if cl:
1226 if imclass is not cl:
1227 note = ' from ' + classname(imclass, mod)
1228 else:
1229 if object.im_self:
1230 note = ' method of %s instance' % classname(
1231 object.im_self.__class__, mod)
1232 else:
1233 note = ' unbound %s method' % classname(imclass,mod)
1234 object = object.im_func
1236 if name == realname:
1237 title = self.bold(realname)
1238 else:
1239 if (cl and realname in cl.__dict__ and
1240 cl.__dict__[realname] is object):
1241 skipdocs = 1
1242 title = self.bold(name) + ' = ' + realname
1243 if inspect.isfunction(object):
1244 args, varargs, varkw, defaults = inspect.getargspec(object)
1245 argspec = inspect.formatargspec(
1246 args, varargs, varkw, defaults, formatvalue=self.formatvalue)
1247 if realname == '<lambda>':
1248 title = self.bold(name) + ' lambda '
1249 argspec = argspec[1:-1] # remove parentheses
1250 else:
1251 argspec = '(...)'
1252 decl = title + argspec + note
1254 if skipdocs:
1255 return decl + '\n'
1256 else:
1257 doc = getdoc(object) or ''
1258 return decl + '\n' + (doc and rstrip(self.indent(doc)) + '\n')
1260 def _docdescriptor(self, name, value, mod):
1261 results = []
1262 push = results.append
1264 if name:
1265 push(self.bold(name))
1266 push('\n')
1267 doc = getdoc(value) or ''
1268 if doc:
1269 push(self.indent(doc))
1270 push('\n')
1271 return ''.join(results)
1273 def docproperty(self, object, name=None, mod=None, cl=None):
1274 """Produce text documentation for a property."""
1275 return self._docdescriptor(name, object, mod)
1277 def docdata(self, object, name=None, mod=None, cl=None):
1278 """Produce text documentation for a data descriptor."""
1279 return self._docdescriptor(name, object, mod)
1281 def docother(self, object, name=None, mod=None, parent=None, maxlen=None, doc=None):
1282 """Produce text documentation for a data object."""
1283 repr = self.repr(object)
1284 if maxlen:
1285 line = (name and name + ' = ' or '') + repr
1286 chop = maxlen - len(line)
1287 if chop < 0: repr = repr[:chop] + '...'
1288 line = (name and self.bold(name) + ' = ' or '') + repr
1289 if doc is not None:
1290 line += '\n' + self.indent(str(doc))
1291 return line
1293 # --------------------------------------------------------- user interfaces
1295 def pager(text):
1296 """The first time this is called, determine what kind of pager to use."""
1297 global pager
1298 pager = getpager()
1299 pager(text)
1301 def getpager():
1302 """Decide what method to use for paging through text."""
1303 if type(sys.stdout) is not types.FileType:
1304 return plainpager
1305 if not sys.stdin.isatty() or not sys.stdout.isatty():
1306 return plainpager
1307 if 'PAGER' in os.environ:
1308 if sys.platform == 'win32': # pipes completely broken in Windows
1309 return lambda text: tempfilepager(plain(text), os.environ['PAGER'])
1310 elif os.environ.get('TERM') in ('dumb', 'emacs'):
1311 return lambda text: pipepager(plain(text), os.environ['PAGER'])
1312 else:
1313 return lambda text: pipepager(text, os.environ['PAGER'])
1314 if os.environ.get('TERM') in ('dumb', 'emacs'):
1315 return plainpager
1316 if sys.platform == 'win32' or sys.platform.startswith('os2'):
1317 return lambda text: tempfilepager(plain(text), 'more <')
1318 if hasattr(os, 'system') and os.system('(less) 2>/dev/null') == 0:
1319 return lambda text: pipepager(text, 'less')
1321 import tempfile
1322 (fd, filename) = tempfile.mkstemp()
1323 os.close(fd)
1324 try:
1325 if hasattr(os, 'system') and os.system('more %s' % filename) == 0:
1326 return lambda text: pipepager(text, 'more')
1327 else:
1328 return ttypager
1329 finally:
1330 os.unlink(filename)
1332 def plain(text):
1333 """Remove boldface formatting from text."""
1334 return re.sub('.\b', '', text)
1336 def pipepager(text, cmd):
1337 """Page through text by feeding it to another program."""
1338 pipe = os.popen(cmd, 'w')
1339 try:
1340 pipe.write(text)
1341 pipe.close()
1342 except IOError:
1343 pass # Ignore broken pipes caused by quitting the pager program.
1345 def tempfilepager(text, cmd):
1346 """Page through text by invoking a program on a temporary file."""
1347 import tempfile
1348 filename = tempfile.mktemp()
1349 file = open(filename, 'w')
1350 file.write(text)
1351 file.close()
1352 try:
1353 os.system(cmd + ' ' + filename)
1354 finally:
1355 os.unlink(filename)
1357 def ttypager(text):
1358 """Page through text on a text terminal."""
1359 lines = split(plain(text), '\n')
1360 try:
1361 import tty
1362 fd = sys.stdin.fileno()
1363 old = tty.tcgetattr(fd)
1364 tty.setcbreak(fd)
1365 getchar = lambda: sys.stdin.read(1)
1366 except (ImportError, AttributeError):
1367 tty = None
1368 getchar = lambda: sys.stdin.readline()[:-1][:1]
1370 try:
1371 r = inc = os.environ.get('LINES', 25) - 1
1372 sys.stdout.write(join(lines[:inc], '\n') + '\n')
1373 while lines[r:]:
1374 sys.stdout.write('-- more --')
1375 sys.stdout.flush()
1376 c = getchar()
1378 if c in ('q', 'Q'):
1379 sys.stdout.write('\r \r')
1380 break
1381 elif c in ('\r', '\n'):
1382 sys.stdout.write('\r \r' + lines[r] + '\n')
1383 r = r + 1
1384 continue
1385 if c in ('b', 'B', '\x1b'):
1386 r = r - inc - inc
1387 if r < 0: r = 0
1388 sys.stdout.write('\n' + join(lines[r:r+inc], '\n') + '\n')
1389 r = r + inc
1391 finally:
1392 if tty:
1393 tty.tcsetattr(fd, tty.TCSAFLUSH, old)
1395 def plainpager(text):
1396 """Simply print unformatted text. This is the ultimate fallback."""
1397 sys.stdout.write(plain(text))
1399 def describe(thing):
1400 """Produce a short description of the given thing."""
1401 if inspect.ismodule(thing):
1402 if thing.__name__ in sys.builtin_module_names:
1403 return 'built-in module ' + thing.__name__
1404 if hasattr(thing, '__path__'):
1405 return 'package ' + thing.__name__
1406 else:
1407 return 'module ' + thing.__name__
1408 if inspect.isbuiltin(thing):
1409 return 'built-in function ' + thing.__name__
1410 if inspect.isgetsetdescriptor(thing):
1411 return 'getset descriptor %s.%s.%s' % (
1412 thing.__objclass__.__module__, thing.__objclass__.__name__,
1413 thing.__name__)
1414 if inspect.ismemberdescriptor(thing):
1415 return 'member descriptor %s.%s.%s' % (
1416 thing.__objclass__.__module__, thing.__objclass__.__name__,
1417 thing.__name__)
1418 if inspect.isclass(thing):
1419 return 'class ' + thing.__name__
1420 if inspect.isfunction(thing):
1421 return 'function ' + thing.__name__
1422 if inspect.ismethod(thing):
1423 return 'method ' + thing.__name__
1424 if type(thing) is types.InstanceType:
1425 return 'instance of ' + thing.__class__.__name__
1426 return type(thing).__name__
1428 def locate(path, forceload=0):
1429 """Locate an object by name or dotted path, importing as necessary."""
1430 parts = [part for part in split(path, '.') if part]
1431 module, n = None, 0
1432 while n < len(parts):
1433 nextmodule = safeimport(join(parts[:n+1], '.'), forceload)
1434 if nextmodule: module, n = nextmodule, n + 1
1435 else: break
1436 if module:
1437 object = module
1438 for part in parts[n:]:
1439 try: object = getattr(object, part)
1440 except AttributeError: return None
1441 return object
1442 else:
1443 if hasattr(__builtin__, path):
1444 return getattr(__builtin__, path)
1446 # --------------------------------------- interactive interpreter interface
1448 text = TextDoc()
1449 html = HTMLDoc()
1451 class _OldStyleClass: pass
1452 _OLD_INSTANCE_TYPE = type(_OldStyleClass())
1454 def resolve(thing, forceload=0):
1455 """Given an object or a path to an object, get the object and its name."""
1456 if isinstance(thing, str):
1457 object = locate(thing, forceload)
1458 if not object:
1459 raise ImportError, 'no Python documentation found for %r' % thing
1460 return object, thing
1461 else:
1462 return thing, getattr(thing, '__name__', None)
1464 def doc(thing, title='Python Library Documentation: %s', forceload=0):
1465 """Display text documentation, given an object or a path to an object."""
1466 try:
1467 object, name = resolve(thing, forceload)
1468 desc = describe(object)
1469 module = inspect.getmodule(object)
1470 if name and '.' in name:
1471 desc += ' in ' + name[:name.rfind('.')]
1472 elif module and module is not object:
1473 desc += ' in module ' + module.__name__
1474 if type(object) is _OLD_INSTANCE_TYPE:
1475 # If the passed object is an instance of an old-style class,
1476 # document its available methods instead of its value.
1477 object = object.__class__
1478 elif not (inspect.ismodule(object) or
1479 inspect.isclass(object) or
1480 inspect.isroutine(object) or
1481 inspect.isgetsetdescriptor(object) or
1482 inspect.ismemberdescriptor(object) or
1483 isinstance(object, property)):
1484 # If the passed object is a piece of data or an instance,
1485 # document its available methods instead of its value.
1486 object = type(object)
1487 desc += ' object'
1488 pager(title % desc + '\n\n' + text.document(object, name))
1489 except (ImportError, ErrorDuringImport), value:
1490 print value
1492 def writedoc(thing, forceload=0):
1493 """Write HTML documentation to a file in the current directory."""
1494 try:
1495 object, name = resolve(thing, forceload)
1496 page = html.page(describe(object), html.document(object, name))
1497 file = open(name + '.html', 'w')
1498 file.write(page)
1499 file.close()
1500 print 'wrote', name + '.html'
1501 except (ImportError, ErrorDuringImport), value:
1502 print value
1504 def writedocs(dir, pkgpath='', done=None):
1505 """Write out HTML documentation for all modules in a directory tree."""
1506 if done is None: done = {}
1507 for importer, modname, ispkg in pkgutil.walk_packages([dir], pkgpath):
1508 writedoc(modname)
1509 return
1511 class Helper:
1512 keywords = {
1513 'and': 'BOOLEAN',
1514 'assert': ('ref/assert', ''),
1515 'break': ('ref/break', 'while for'),
1516 'class': ('ref/class', 'CLASSES SPECIALMETHODS'),
1517 'continue': ('ref/continue', 'while for'),
1518 'def': ('ref/function', ''),
1519 'del': ('ref/del', 'BASICMETHODS'),
1520 'elif': 'if',
1521 'else': ('ref/if', 'while for'),
1522 'except': 'try',
1523 'exec': ('ref/exec', ''),
1524 'finally': 'try',
1525 'for': ('ref/for', 'break continue while'),
1526 'from': 'import',
1527 'global': ('ref/global', 'NAMESPACES'),
1528 'if': ('ref/if', 'TRUTHVALUE'),
1529 'import': ('ref/import', 'MODULES'),
1530 'in': ('ref/comparisons', 'SEQUENCEMETHODS2'),
1531 'is': 'COMPARISON',
1532 'lambda': ('ref/lambdas', 'FUNCTIONS'),
1533 'not': 'BOOLEAN',
1534 'or': 'BOOLEAN',
1535 'pass': ('ref/pass', ''),
1536 'print': ('ref/print', ''),
1537 'raise': ('ref/raise', 'EXCEPTIONS'),
1538 'return': ('ref/return', 'FUNCTIONS'),
1539 'try': ('ref/try', 'EXCEPTIONS'),
1540 'while': ('ref/while', 'break continue if TRUTHVALUE'),
1541 'yield': ('ref/yield', ''),
1544 topics = {
1545 'TYPES': ('ref/types', 'STRINGS UNICODE NUMBERS SEQUENCES MAPPINGS FUNCTIONS CLASSES MODULES FILES inspect'),
1546 'STRINGS': ('ref/strings', 'str UNICODE SEQUENCES STRINGMETHODS FORMATTING TYPES'),
1547 'STRINGMETHODS': ('lib/string-methods', 'STRINGS FORMATTING'),
1548 'FORMATTING': ('lib/typesseq-strings', 'OPERATORS'),
1549 'UNICODE': ('ref/strings', 'encodings unicode SEQUENCES STRINGMETHODS FORMATTING TYPES'),
1550 'NUMBERS': ('ref/numbers', 'INTEGER FLOAT COMPLEX TYPES'),
1551 'INTEGER': ('ref/integers', 'int range'),
1552 'FLOAT': ('ref/floating', 'float math'),
1553 'COMPLEX': ('ref/imaginary', 'complex cmath'),
1554 'SEQUENCES': ('lib/typesseq', 'STRINGMETHODS FORMATTING xrange LISTS'),
1555 'MAPPINGS': 'DICTIONARIES',
1556 'FUNCTIONS': ('lib/typesfunctions', 'def TYPES'),
1557 'METHODS': ('lib/typesmethods', 'class def CLASSES TYPES'),
1558 'CODEOBJECTS': ('lib/bltin-code-objects', 'compile FUNCTIONS TYPES'),
1559 'TYPEOBJECTS': ('lib/bltin-type-objects', 'types TYPES'),
1560 'FRAMEOBJECTS': 'TYPES',
1561 'TRACEBACKS': 'TYPES',
1562 'NONE': ('lib/bltin-null-object', ''),
1563 'ELLIPSIS': ('lib/bltin-ellipsis-object', 'SLICINGS'),
1564 'FILES': ('lib/bltin-file-objects', ''),
1565 'SPECIALATTRIBUTES': ('lib/specialattrs', ''),
1566 'CLASSES': ('ref/types', 'class SPECIALMETHODS PRIVATENAMES'),
1567 'MODULES': ('lib/typesmodules', 'import'),
1568 'PACKAGES': 'import',
1569 'EXPRESSIONS': ('ref/summary', 'lambda or and not in is BOOLEAN COMPARISON BITWISE SHIFTING BINARY FORMATTING POWER UNARY ATTRIBUTES SUBSCRIPTS SLICINGS CALLS TUPLES LISTS DICTIONARIES BACKQUOTES'),
1570 'OPERATORS': 'EXPRESSIONS',
1571 'PRECEDENCE': 'EXPRESSIONS',
1572 'OBJECTS': ('ref/objects', 'TYPES'),
1573 'SPECIALMETHODS': ('ref/specialnames', 'BASICMETHODS ATTRIBUTEMETHODS CALLABLEMETHODS SEQUENCEMETHODS1 MAPPINGMETHODS SEQUENCEMETHODS2 NUMBERMETHODS CLASSES'),
1574 'BASICMETHODS': ('ref/customization', 'cmp hash repr str SPECIALMETHODS'),
1575 'ATTRIBUTEMETHODS': ('ref/attribute-access', 'ATTRIBUTES SPECIALMETHODS'),
1576 'CALLABLEMETHODS': ('ref/callable-types', 'CALLS SPECIALMETHODS'),
1577 'SEQUENCEMETHODS1': ('ref/sequence-types', 'SEQUENCES SEQUENCEMETHODS2 SPECIALMETHODS'),
1578 'SEQUENCEMETHODS2': ('ref/sequence-methods', 'SEQUENCES SEQUENCEMETHODS1 SPECIALMETHODS'),
1579 'MAPPINGMETHODS': ('ref/sequence-types', 'MAPPINGS SPECIALMETHODS'),
1580 'NUMBERMETHODS': ('ref/numeric-types', 'NUMBERS AUGMENTEDASSIGNMENT SPECIALMETHODS'),
1581 'EXECUTION': ('ref/execmodel', 'NAMESPACES DYNAMICFEATURES EXCEPTIONS'),
1582 'NAMESPACES': ('ref/naming', 'global ASSIGNMENT DELETION DYNAMICFEATURES'),
1583 'DYNAMICFEATURES': ('ref/dynamic-features', ''),
1584 'SCOPING': 'NAMESPACES',
1585 'FRAMES': 'NAMESPACES',
1586 'EXCEPTIONS': ('ref/exceptions', 'try except finally raise'),
1587 'COERCIONS': ('ref/coercion-rules','CONVERSIONS'),
1588 'CONVERSIONS': ('ref/conversions', 'COERCIONS'),
1589 'IDENTIFIERS': ('ref/identifiers', 'keywords SPECIALIDENTIFIERS'),
1590 'SPECIALIDENTIFIERS': ('ref/id-classes', ''),
1591 'PRIVATENAMES': ('ref/atom-identifiers', ''),
1592 'LITERALS': ('ref/atom-literals', 'STRINGS BACKQUOTES NUMBERS TUPLELITERALS LISTLITERALS DICTIONARYLITERALS'),
1593 'TUPLES': 'SEQUENCES',
1594 'TUPLELITERALS': ('ref/exprlists', 'TUPLES LITERALS'),
1595 'LISTS': ('lib/typesseq-mutable', 'LISTLITERALS'),
1596 'LISTLITERALS': ('ref/lists', 'LISTS LITERALS'),
1597 'DICTIONARIES': ('lib/typesmapping', 'DICTIONARYLITERALS'),
1598 'DICTIONARYLITERALS': ('ref/dict', 'DICTIONARIES LITERALS'),
1599 'BACKQUOTES': ('ref/string-conversions', 'repr str STRINGS LITERALS'),
1600 'ATTRIBUTES': ('ref/attribute-references', 'getattr hasattr setattr ATTRIBUTEMETHODS'),
1601 'SUBSCRIPTS': ('ref/subscriptions', 'SEQUENCEMETHODS1'),
1602 'SLICINGS': ('ref/slicings', 'SEQUENCEMETHODS2'),
1603 'CALLS': ('ref/calls', 'EXPRESSIONS'),
1604 'POWER': ('ref/power', 'EXPRESSIONS'),
1605 'UNARY': ('ref/unary', 'EXPRESSIONS'),
1606 'BINARY': ('ref/binary', 'EXPRESSIONS'),
1607 'SHIFTING': ('ref/shifting', 'EXPRESSIONS'),
1608 'BITWISE': ('ref/bitwise', 'EXPRESSIONS'),
1609 'COMPARISON': ('ref/comparisons', 'EXPRESSIONS BASICMETHODS'),
1610 'BOOLEAN': ('ref/Booleans', 'EXPRESSIONS TRUTHVALUE'),
1611 'ASSERTION': 'assert',
1612 'ASSIGNMENT': ('ref/assignment', 'AUGMENTEDASSIGNMENT'),
1613 'AUGMENTEDASSIGNMENT': ('ref/augassign', 'NUMBERMETHODS'),
1614 'DELETION': 'del',
1615 'PRINTING': 'print',
1616 'RETURNING': 'return',
1617 'IMPORTING': 'import',
1618 'CONDITIONAL': 'if',
1619 'LOOPING': ('ref/compound', 'for while break continue'),
1620 'TRUTHVALUE': ('lib/truth', 'if while and or not BASICMETHODS'),
1621 'DEBUGGING': ('lib/module-pdb', 'pdb'),
1624 def __init__(self, input, output):
1625 self.input = input
1626 self.output = output
1627 self.docdir = None
1628 execdir = os.path.dirname(sys.executable)
1629 homedir = os.environ.get('PYTHONHOME')
1630 for dir in [os.environ.get('PYTHONDOCS'),
1631 homedir and os.path.join(homedir, 'doc'),
1632 os.path.join(execdir, 'doc'),
1633 '/usr/doc/python-docs-' + split(sys.version)[0],
1634 '/usr/doc/python-' + split(sys.version)[0],
1635 '/usr/doc/python-docs-' + sys.version[:3],
1636 '/usr/doc/python-' + sys.version[:3],
1637 os.path.join(sys.prefix, 'Resources/English.lproj/Documentation')]:
1638 if dir and os.path.isdir(os.path.join(dir, 'lib')):
1639 self.docdir = dir
1641 def __repr__(self):
1642 if inspect.stack()[1][3] == '?':
1643 self()
1644 return ''
1645 return '<pydoc.Helper instance>'
1647 def __call__(self, request=None):
1648 if request is not None:
1649 self.help(request)
1650 else:
1651 self.intro()
1652 self.interact()
1653 self.output.write('''
1654 You are now leaving help and returning to the Python interpreter.
1655 If you want to ask for help on a particular object directly from the
1656 interpreter, you can type "help(object)". Executing "help('string')"
1657 has the same effect as typing a particular string at the help> prompt.
1658 ''')
1660 def interact(self):
1661 self.output.write('\n')
1662 while True:
1663 try:
1664 request = self.getline('help> ')
1665 if not request: break
1666 except (KeyboardInterrupt, EOFError):
1667 break
1668 request = strip(replace(request, '"', '', "'", ''))
1669 if lower(request) in ('q', 'quit'): break
1670 self.help(request)
1672 def getline(self, prompt):
1673 """Read one line, using raw_input when available."""
1674 if self.input is sys.stdin:
1675 return raw_input(prompt)
1676 else:
1677 self.output.write(prompt)
1678 self.output.flush()
1679 return self.input.readline()
1681 def help(self, request):
1682 if type(request) is type(''):
1683 if request == 'help': self.intro()
1684 elif request == 'keywords': self.listkeywords()
1685 elif request == 'topics': self.listtopics()
1686 elif request == 'modules': self.listmodules()
1687 elif request[:8] == 'modules ':
1688 self.listmodules(split(request)[1])
1689 elif request in self.keywords: self.showtopic(request)
1690 elif request in self.topics: self.showtopic(request)
1691 elif request: doc(request, 'Help on %s:')
1692 elif isinstance(request, Helper): self()
1693 else: doc(request, 'Help on %s:')
1694 self.output.write('\n')
1696 def intro(self):
1697 self.output.write('''
1698 Welcome to Python %s! This is the online help utility.
1700 If this is your first time using Python, you should definitely check out
1701 the tutorial on the Internet at http://www.python.org/doc/tut/.
1703 Enter the name of any module, keyword, or topic to get help on writing
1704 Python programs and using Python modules. To quit this help utility and
1705 return to the interpreter, just type "quit".
1707 To get a list of available modules, keywords, or topics, type "modules",
1708 "keywords", or "topics". Each module also comes with a one-line summary
1709 of what it does; to list the modules whose summaries contain a given word
1710 such as "spam", type "modules spam".
1711 ''' % sys.version[:3])
1713 def list(self, items, columns=4, width=80):
1714 items = items[:]
1715 items.sort()
1716 colw = width / columns
1717 rows = (len(items) + columns - 1) / columns
1718 for row in range(rows):
1719 for col in range(columns):
1720 i = col * rows + row
1721 if i < len(items):
1722 self.output.write(items[i])
1723 if col < columns - 1:
1724 self.output.write(' ' + ' ' * (colw-1 - len(items[i])))
1725 self.output.write('\n')
1727 def listkeywords(self):
1728 self.output.write('''
1729 Here is a list of the Python keywords. Enter any keyword to get more help.
1731 ''')
1732 self.list(self.keywords.keys())
1734 def listtopics(self):
1735 self.output.write('''
1736 Here is a list of available topics. Enter any topic name to get more help.
1738 ''')
1739 self.list(self.topics.keys())
1741 def showtopic(self, topic):
1742 if not self.docdir:
1743 self.output.write('''
1744 Sorry, topic and keyword documentation is not available because the Python
1745 HTML documentation files could not be found. If you have installed them,
1746 please set the environment variable PYTHONDOCS to indicate their location.
1748 On the Microsoft Windows operating system, the files can be built by
1749 running "hh -decompile . PythonNN.chm" in the C:\PythonNN\Doc> directory.
1750 ''')
1751 return
1752 target = self.topics.get(topic, self.keywords.get(topic))
1753 if not target:
1754 self.output.write('no documentation found for %s\n' % repr(topic))
1755 return
1756 if type(target) is type(''):
1757 return self.showtopic(target)
1759 filename, xrefs = target
1760 filename = self.docdir + '/' + filename + '.html'
1761 try:
1762 file = open(filename)
1763 except:
1764 self.output.write('could not read docs from %s\n' % filename)
1765 return
1767 divpat = re.compile('<div[^>]*navigat.*?</div.*?>', re.I | re.S)
1768 addrpat = re.compile('<address.*?>.*?</address.*?>', re.I | re.S)
1769 document = re.sub(addrpat, '', re.sub(divpat, '', file.read()))
1770 file.close()
1772 import htmllib, formatter, StringIO
1773 buffer = StringIO.StringIO()
1774 parser = htmllib.HTMLParser(
1775 formatter.AbstractFormatter(formatter.DumbWriter(buffer)))
1776 parser.start_table = parser.do_p
1777 parser.end_table = lambda parser=parser: parser.do_p({})
1778 parser.start_tr = parser.do_br
1779 parser.start_td = parser.start_th = lambda a, b=buffer: b.write('\t')
1780 parser.feed(document)
1781 buffer = replace(buffer.getvalue(), '\xa0', ' ', '\n', '\n ')
1782 pager(' ' + strip(buffer) + '\n')
1783 if xrefs:
1784 buffer = StringIO.StringIO()
1785 formatter.DumbWriter(buffer).send_flowing_data(
1786 'Related help topics: ' + join(split(xrefs), ', ') + '\n')
1787 self.output.write('\n%s\n' % buffer.getvalue())
1789 def listmodules(self, key=''):
1790 if key:
1791 self.output.write('''
1792 Here is a list of matching modules. Enter any module name to get more help.
1794 ''')
1795 apropos(key)
1796 else:
1797 self.output.write('''
1798 Please wait a moment while I gather a list of all available modules...
1800 ''')
1801 modules = {}
1802 def callback(path, modname, desc, modules=modules):
1803 if modname and modname[-9:] == '.__init__':
1804 modname = modname[:-9] + ' (package)'
1805 if find(modname, '.') < 0:
1806 modules[modname] = 1
1807 ModuleScanner().run(callback)
1808 self.list(modules.keys())
1809 self.output.write('''
1810 Enter any module name to get more help. Or, type "modules spam" to search
1811 for modules whose descriptions contain the word "spam".
1812 ''')
1814 help = Helper(sys.stdin, sys.stdout)
1816 class Scanner:
1817 """A generic tree iterator."""
1818 def __init__(self, roots, children, descendp):
1819 self.roots = roots[:]
1820 self.state = []
1821 self.children = children
1822 self.descendp = descendp
1824 def next(self):
1825 if not self.state:
1826 if not self.roots:
1827 return None
1828 root = self.roots.pop(0)
1829 self.state = [(root, self.children(root))]
1830 node, children = self.state[-1]
1831 if not children:
1832 self.state.pop()
1833 return self.next()
1834 child = children.pop(0)
1835 if self.descendp(child):
1836 self.state.append((child, self.children(child)))
1837 return child
1840 class ModuleScanner:
1841 """An interruptible scanner that searches module synopses."""
1843 def run(self, callback, key=None, completer=None):
1844 if key: key = lower(key)
1845 self.quit = False
1846 seen = {}
1848 for modname in sys.builtin_module_names:
1849 if modname != '__main__':
1850 seen[modname] = 1
1851 if key is None:
1852 callback(None, modname, '')
1853 else:
1854 desc = split(__import__(modname).__doc__ or '', '\n')[0]
1855 if find(lower(modname + ' - ' + desc), key) >= 0:
1856 callback(None, modname, desc)
1858 for importer, modname, ispkg in pkgutil.walk_packages():
1859 if self.quit:
1860 break
1861 if key is None:
1862 callback(None, modname, '')
1863 else:
1864 loader = importer.find_module(modname)
1865 if hasattr(loader,'get_source'):
1866 import StringIO
1867 desc = source_synopsis(
1868 StringIO.StringIO(loader.get_source(modname))
1869 ) or ''
1870 if hasattr(loader,'get_filename'):
1871 path = loader.get_filename(modname)
1872 else:
1873 path = None
1874 else:
1875 module = loader.load_module(modname)
1876 desc = (module.__doc__ or '').splitlines()[0]
1877 path = getattr(module,'__file__',None)
1878 if find(lower(modname + ' - ' + desc), key) >= 0:
1879 callback(path, modname, desc)
1881 if completer:
1882 completer()
1884 def apropos(key):
1885 """Print all the one-line module summaries that contain a substring."""
1886 def callback(path, modname, desc):
1887 if modname[-9:] == '.__init__':
1888 modname = modname[:-9] + ' (package)'
1889 print modname, desc and '- ' + desc
1890 try: import warnings
1891 except ImportError: pass
1892 else: warnings.filterwarnings('ignore') # ignore problems during import
1893 ModuleScanner().run(callback, key)
1895 # --------------------------------------------------- web browser interface
1897 def serve(port, callback=None, completer=None):
1898 import BaseHTTPServer, mimetools, select
1900 # Patch up mimetools.Message so it doesn't break if rfc822 is reloaded.
1901 class Message(mimetools.Message):
1902 def __init__(self, fp, seekable=1):
1903 Message = self.__class__
1904 Message.__bases__[0].__bases__[0].__init__(self, fp, seekable)
1905 self.encodingheader = self.getheader('content-transfer-encoding')
1906 self.typeheader = self.getheader('content-type')
1907 self.parsetype()
1908 self.parseplist()
1910 class DocHandler(BaseHTTPServer.BaseHTTPRequestHandler):
1911 def send_document(self, title, contents):
1912 try:
1913 self.send_response(200)
1914 self.send_header('Content-Type', 'text/html')
1915 self.end_headers()
1916 self.wfile.write(html.page(title, contents))
1917 except IOError: pass
1919 def do_GET(self):
1920 path = self.path
1921 if path[-5:] == '.html': path = path[:-5]
1922 if path[:1] == '/': path = path[1:]
1923 if path and path != '.':
1924 try:
1925 obj = locate(path, forceload=1)
1926 except ErrorDuringImport, value:
1927 self.send_document(path, html.escape(str(value)))
1928 return
1929 if obj:
1930 self.send_document(describe(obj), html.document(obj, path))
1931 else:
1932 self.send_document(path,
1933 'no Python documentation found for %s' % repr(path))
1934 else:
1935 heading = html.heading(
1936 '<big><big><strong>Python: Index of Modules</strong></big></big>',
1937 '#ffffff', '#7799ee')
1938 def bltinlink(name):
1939 return '<a href="%s.html">%s</a>' % (name, name)
1940 names = filter(lambda x: x != '__main__',
1941 sys.builtin_module_names)
1942 contents = html.multicolumn(names, bltinlink)
1943 indices = ['<p>' + html.bigsection(
1944 'Built-in Modules', '#ffffff', '#ee77aa', contents)]
1946 seen = {}
1947 for dir in sys.path:
1948 indices.append(html.index(dir, seen))
1949 contents = heading + join(indices) + '''<p align=right>
1950 <font color="#909090" face="helvetica, arial"><strong>
1951 pydoc</strong> by Ka-Ping Yee &lt;ping@lfw.org&gt;</font>'''
1952 self.send_document('Index of Modules', contents)
1954 def log_message(self, *args): pass
1956 class DocServer(BaseHTTPServer.HTTPServer):
1957 def __init__(self, port, callback):
1958 host = (sys.platform == 'mac') and '127.0.0.1' or 'localhost'
1959 self.address = ('', port)
1960 self.url = 'http://%s:%d/' % (host, port)
1961 self.callback = callback
1962 self.base.__init__(self, self.address, self.handler)
1964 def serve_until_quit(self):
1965 import select
1966 self.quit = False
1967 while not self.quit:
1968 rd, wr, ex = select.select([self.socket.fileno()], [], [], 1)
1969 if rd: self.handle_request()
1971 def server_activate(self):
1972 self.base.server_activate(self)
1973 if self.callback: self.callback(self)
1975 DocServer.base = BaseHTTPServer.HTTPServer
1976 DocServer.handler = DocHandler
1977 DocHandler.MessageClass = Message
1978 try:
1979 try:
1980 DocServer(port, callback).serve_until_quit()
1981 except (KeyboardInterrupt, select.error):
1982 pass
1983 finally:
1984 if completer: completer()
1986 # ----------------------------------------------------- graphical interface
1988 def gui():
1989 """Graphical interface (starts web server and pops up a control window)."""
1990 class GUI:
1991 def __init__(self, window, port=7464):
1992 self.window = window
1993 self.server = None
1994 self.scanner = None
1996 import Tkinter
1997 self.server_frm = Tkinter.Frame(window)
1998 self.title_lbl = Tkinter.Label(self.server_frm,
1999 text='Starting server...\n ')
2000 self.open_btn = Tkinter.Button(self.server_frm,
2001 text='open browser', command=self.open, state='disabled')
2002 self.quit_btn = Tkinter.Button(self.server_frm,
2003 text='quit serving', command=self.quit, state='disabled')
2005 self.search_frm = Tkinter.Frame(window)
2006 self.search_lbl = Tkinter.Label(self.search_frm, text='Search for')
2007 self.search_ent = Tkinter.Entry(self.search_frm)
2008 self.search_ent.bind('<Return>', self.search)
2009 self.stop_btn = Tkinter.Button(self.search_frm,
2010 text='stop', pady=0, command=self.stop, state='disabled')
2011 if sys.platform == 'win32':
2012 # Trying to hide and show this button crashes under Windows.
2013 self.stop_btn.pack(side='right')
2015 self.window.title('pydoc')
2016 self.window.protocol('WM_DELETE_WINDOW', self.quit)
2017 self.title_lbl.pack(side='top', fill='x')
2018 self.open_btn.pack(side='left', fill='x', expand=1)
2019 self.quit_btn.pack(side='right', fill='x', expand=1)
2020 self.server_frm.pack(side='top', fill='x')
2022 self.search_lbl.pack(side='left')
2023 self.search_ent.pack(side='right', fill='x', expand=1)
2024 self.search_frm.pack(side='top', fill='x')
2025 self.search_ent.focus_set()
2027 font = ('helvetica', sys.platform == 'win32' and 8 or 10)
2028 self.result_lst = Tkinter.Listbox(window, font=font, height=6)
2029 self.result_lst.bind('<Button-1>', self.select)
2030 self.result_lst.bind('<Double-Button-1>', self.goto)
2031 self.result_scr = Tkinter.Scrollbar(window,
2032 orient='vertical', command=self.result_lst.yview)
2033 self.result_lst.config(yscrollcommand=self.result_scr.set)
2035 self.result_frm = Tkinter.Frame(window)
2036 self.goto_btn = Tkinter.Button(self.result_frm,
2037 text='go to selected', command=self.goto)
2038 self.hide_btn = Tkinter.Button(self.result_frm,
2039 text='hide results', command=self.hide)
2040 self.goto_btn.pack(side='left', fill='x', expand=1)
2041 self.hide_btn.pack(side='right', fill='x', expand=1)
2043 self.window.update()
2044 self.minwidth = self.window.winfo_width()
2045 self.minheight = self.window.winfo_height()
2046 self.bigminheight = (self.server_frm.winfo_reqheight() +
2047 self.search_frm.winfo_reqheight() +
2048 self.result_lst.winfo_reqheight() +
2049 self.result_frm.winfo_reqheight())
2050 self.bigwidth, self.bigheight = self.minwidth, self.bigminheight
2051 self.expanded = 0
2052 self.window.wm_geometry('%dx%d' % (self.minwidth, self.minheight))
2053 self.window.wm_minsize(self.minwidth, self.minheight)
2054 self.window.tk.willdispatch()
2056 import threading
2057 threading.Thread(
2058 target=serve, args=(port, self.ready, self.quit)).start()
2060 def ready(self, server):
2061 self.server = server
2062 self.title_lbl.config(
2063 text='Python documentation server at\n' + server.url)
2064 self.open_btn.config(state='normal')
2065 self.quit_btn.config(state='normal')
2067 def open(self, event=None, url=None):
2068 url = url or self.server.url
2069 try:
2070 import webbrowser
2071 webbrowser.open(url)
2072 except ImportError: # pre-webbrowser.py compatibility
2073 if sys.platform == 'win32':
2074 os.system('start "%s"' % url)
2075 elif sys.platform == 'mac':
2076 try: import ic
2077 except ImportError: pass
2078 else: ic.launchurl(url)
2079 else:
2080 rc = os.system('netscape -remote "openURL(%s)" &' % url)
2081 if rc: os.system('netscape "%s" &' % url)
2083 def quit(self, event=None):
2084 if self.server:
2085 self.server.quit = 1
2086 self.window.quit()
2088 def search(self, event=None):
2089 key = self.search_ent.get()
2090 self.stop_btn.pack(side='right')
2091 self.stop_btn.config(state='normal')
2092 self.search_lbl.config(text='Searching for "%s"...' % key)
2093 self.search_ent.forget()
2094 self.search_lbl.pack(side='left')
2095 self.result_lst.delete(0, 'end')
2096 self.goto_btn.config(state='disabled')
2097 self.expand()
2099 import threading
2100 if self.scanner:
2101 self.scanner.quit = 1
2102 self.scanner = ModuleScanner()
2103 threading.Thread(target=self.scanner.run,
2104 args=(self.update, key, self.done)).start()
2106 def update(self, path, modname, desc):
2107 if modname[-9:] == '.__init__':
2108 modname = modname[:-9] + ' (package)'
2109 self.result_lst.insert('end',
2110 modname + ' - ' + (desc or '(no description)'))
2112 def stop(self, event=None):
2113 if self.scanner:
2114 self.scanner.quit = 1
2115 self.scanner = None
2117 def done(self):
2118 self.scanner = None
2119 self.search_lbl.config(text='Search for')
2120 self.search_lbl.pack(side='left')
2121 self.search_ent.pack(side='right', fill='x', expand=1)
2122 if sys.platform != 'win32': self.stop_btn.forget()
2123 self.stop_btn.config(state='disabled')
2125 def select(self, event=None):
2126 self.goto_btn.config(state='normal')
2128 def goto(self, event=None):
2129 selection = self.result_lst.curselection()
2130 if selection:
2131 modname = split(self.result_lst.get(selection[0]))[0]
2132 self.open(url=self.server.url + modname + '.html')
2134 def collapse(self):
2135 if not self.expanded: return
2136 self.result_frm.forget()
2137 self.result_scr.forget()
2138 self.result_lst.forget()
2139 self.bigwidth = self.window.winfo_width()
2140 self.bigheight = self.window.winfo_height()
2141 self.window.wm_geometry('%dx%d' % (self.minwidth, self.minheight))
2142 self.window.wm_minsize(self.minwidth, self.minheight)
2143 self.expanded = 0
2145 def expand(self):
2146 if self.expanded: return
2147 self.result_frm.pack(side='bottom', fill='x')
2148 self.result_scr.pack(side='right', fill='y')
2149 self.result_lst.pack(side='top', fill='both', expand=1)
2150 self.window.wm_geometry('%dx%d' % (self.bigwidth, self.bigheight))
2151 self.window.wm_minsize(self.minwidth, self.bigminheight)
2152 self.expanded = 1
2154 def hide(self, event=None):
2155 self.stop()
2156 self.collapse()
2158 import Tkinter
2159 try:
2160 root = Tkinter.Tk()
2161 # Tk will crash if pythonw.exe has an XP .manifest
2162 # file and the root has is not destroyed explicitly.
2163 # If the problem is ever fixed in Tk, the explicit
2164 # destroy can go.
2165 try:
2166 gui = GUI(root)
2167 root.mainloop()
2168 finally:
2169 root.destroy()
2170 except KeyboardInterrupt:
2171 pass
2173 # -------------------------------------------------- command-line interface
2175 def ispath(x):
2176 return isinstance(x, str) and find(x, os.sep) >= 0
2178 def cli():
2179 """Command-line interface (looks at sys.argv to decide what to do)."""
2180 import getopt
2181 class BadUsage: pass
2183 # Scripts don't get the current directory in their path by default.
2184 scriptdir = os.path.dirname(sys.argv[0])
2185 if scriptdir in sys.path:
2186 sys.path.remove(scriptdir)
2187 sys.path.insert(0, '.')
2189 try:
2190 opts, args = getopt.getopt(sys.argv[1:], 'gk:p:w')
2191 writing = 0
2193 for opt, val in opts:
2194 if opt == '-g':
2195 gui()
2196 return
2197 if opt == '-k':
2198 apropos(val)
2199 return
2200 if opt == '-p':
2201 try:
2202 port = int(val)
2203 except ValueError:
2204 raise BadUsage
2205 def ready(server):
2206 print 'pydoc server ready at %s' % server.url
2207 def stopped():
2208 print 'pydoc server stopped'
2209 serve(port, ready, stopped)
2210 return
2211 if opt == '-w':
2212 writing = 1
2214 if not args: raise BadUsage
2215 for arg in args:
2216 if ispath(arg) and not os.path.exists(arg):
2217 print 'file %r does not exist' % arg
2218 break
2219 try:
2220 if ispath(arg) and os.path.isfile(arg):
2221 arg = importfile(arg)
2222 if writing:
2223 if ispath(arg) and os.path.isdir(arg):
2224 writedocs(arg)
2225 else:
2226 writedoc(arg)
2227 else:
2228 help.help(arg)
2229 except ErrorDuringImport, value:
2230 print value
2232 except (getopt.error, BadUsage):
2233 cmd = os.path.basename(sys.argv[0])
2234 print """pydoc - the Python documentation tool
2236 %s <name> ...
2237 Show text documentation on something. <name> may be the name of a
2238 Python keyword, topic, function, module, or package, or a dotted
2239 reference to a class or function within a module or module in a
2240 package. If <name> contains a '%s', it is used as the path to a
2241 Python source file to document. If name is 'keywords', 'topics',
2242 or 'modules', a listing of these things is displayed.
2244 %s -k <keyword>
2245 Search for a keyword in the synopsis lines of all available modules.
2247 %s -p <port>
2248 Start an HTTP server on the given port on the local machine.
2250 %s -g
2251 Pop up a graphical interface for finding and serving documentation.
2253 %s -w <name> ...
2254 Write out the HTML documentation for a module to a file in the current
2255 directory. If <name> contains a '%s', it is treated as a filename; if
2256 it names a directory, documentation is written for all the contents.
2257 """ % (cmd, os.sep, cmd, cmd, cmd, cmd, os.sep)
2259 if __name__ == '__main__': cli()