blob: 8231739f7574ea32737aeae68154f7695202bf1c [file] [log] [blame]
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001#!/usr/bin/env python
Guido van Rossumfce538c2002-08-06 17:29:38 +00002# -*- coding: Latin-1 -*-
Ka-Ping Yee1d384632001-03-01 00:24:32 +00003"""Generate Python documentation in HTML or text for interactive use.
Ka-Ping Yeedd175342001-02-27 14:43:46 +00004
Ka-Ping Yeedd175342001-02-27 14:43:46 +00005In the Python interpreter, do "from pydoc import help" to provide online
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00006help. Calling help(thing) on a Python object documents the object.
7
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00008Or, at the shell command line outside of Python:
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00009
Ka-Ping Yee37f7b382001-03-23 00:12:53 +000010Run "pydoc <name>" to show documentation on something. <name> may be
11the name of a function, module, package, or a dotted reference to a
12class or function within a module or module in a package. If the
13argument contains a path segment delimiter (e.g. slash on Unix,
14backslash on Windows) it is treated as the path to a Python source file.
Ka-Ping Yee66efbc72001-03-01 13:55:20 +000015
Ka-Ping Yee37f7b382001-03-23 00:12:53 +000016Run "pydoc -k <keyword>" to search for a keyword in the synopsis lines
17of all available modules.
Ka-Ping Yee66efbc72001-03-01 13:55:20 +000018
Ka-Ping Yee37f7b382001-03-23 00:12:53 +000019Run "pydoc -p <port>" to start an HTTP server on a given port on the
20local machine to generate documentation web pages.
Ka-Ping Yee66efbc72001-03-01 13:55:20 +000021
Ka-Ping Yee37f7b382001-03-23 00:12:53 +000022For platforms without a command line, "pydoc -g" starts the HTTP server
23and also pops up a little window for controlling it.
24
25Run "pydoc -w <name>" to write out the HTML documentation for a module
26to a file named "<name>.html".
Skip Montanaro4997a692003-09-10 16:47:51 +000027
28Module docs for core modules are assumed to be in
29
30 http://www.python.org/doc/current/lib/
31
32This can be overridden by setting the PYTHONDOCS environment variable
33to a different URL or to a local directory containing the Library
34Reference Manual pages.
Ka-Ping Yee66efbc72001-03-01 13:55:20 +000035"""
Ka-Ping Yeedd175342001-02-27 14:43:46 +000036
37__author__ = "Ka-Ping Yee <ping@lfw.org>"
Ka-Ping Yee6f3f9a42001-02-27 22:42:36 +000038__date__ = "26 February 2001"
Jeremy Hylton3e0055f2005-10-20 19:59:25 +000039
Ka-Ping Yee09d7d9a2001-02-27 22:43:48 +000040__version__ = "$Revision$"
Martin v. Löwis6fe8f192004-11-14 10:21:04 +000041__credits__ = """Guido van Rossum, for an excellent programming language.
Ka-Ping Yee5e2b1732001-02-27 23:35:09 +000042Tommy Burnette, the original creator of manpy.
Ka-Ping Yee6f3f9a42001-02-27 22:42:36 +000043Paul Prescod, for all his work on onlinehelp.
44Richard Chamberlain, for the first implementation of textdoc.
Raymond Hettingered2dbe32005-01-01 07:51:01 +000045"""
Ka-Ping Yeedd175342001-02-27 14:43:46 +000046
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +000047# 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.
Ka-Ping Yee66efbc72001-03-01 13:55:20 +000054
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +000055import sys, imp, os, re, types, inspect, __builtin__
Ka-Ping Yeedd175342001-02-27 14:43:46 +000056from repr import Repr
Ka-Ping Yee3bda8792001-03-23 13:17:50 +000057from string import expandtabs, find, join, lower, split, strip, rfind, rstrip
Raymond Hettinger756b3f32004-01-29 06:37:52 +000058from collections import deque
Ka-Ping Yeedd175342001-02-27 14:43:46 +000059
60# --------------------------------------------------------- common routines
61
Ka-Ping Yeedd175342001-02-27 14:43:46 +000062def pathdirs():
63 """Convert sys.path into a list of absolute, existing, unique paths."""
64 dirs = []
Ka-Ping Yee1d384632001-03-01 00:24:32 +000065 normdirs = []
Ka-Ping Yeedd175342001-02-27 14:43:46 +000066 for dir in sys.path:
67 dir = os.path.abspath(dir or '.')
Ka-Ping Yee1d384632001-03-01 00:24:32 +000068 normdir = os.path.normcase(dir)
69 if normdir not in normdirs and os.path.isdir(dir):
Ka-Ping Yeedd175342001-02-27 14:43:46 +000070 dirs.append(dir)
Ka-Ping Yee1d384632001-03-01 00:24:32 +000071 normdirs.append(normdir)
Ka-Ping Yeedd175342001-02-27 14:43:46 +000072 return dirs
73
74def getdoc(object):
75 """Get the doc string or comments for an object."""
Ka-Ping Yee3bda8792001-03-23 13:17:50 +000076 result = inspect.getdoc(object) or inspect.getcomments(object)
Ka-Ping Yee239432a2001-03-02 02:45:08 +000077 return result and re.sub('^ *\n', '', rstrip(result)) or ''
Ka-Ping Yeedd175342001-02-27 14:43:46 +000078
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +000079def splitdoc(doc):
80 """Split a doc string into a synopsis line (if any) and the rest."""
81 lines = split(strip(doc), '\n')
82 if len(lines) == 1:
83 return lines[0], ''
84 elif len(lines) >= 2 and not rstrip(lines[1]):
85 return lines[0], join(lines[2:], '\n')
86 return '', join(lines, '\n')
87
Ka-Ping Yeedd175342001-02-27 14:43:46 +000088def classname(object, modname):
89 """Get a class name and qualify it with a module name if necessary."""
90 name = object.__name__
91 if object.__module__ != modname:
92 name = object.__module__ + '.' + name
93 return name
94
Ka-Ping Yeedec96e92001-04-13 09:55:49 +000095def isdata(object):
Georg Brandl94432422005-07-22 21:52:25 +000096 """Check if an object is of a type that probably means it's data."""
Ka-Ping Yeedec96e92001-04-13 09:55:49 +000097 return not (inspect.ismodule(object) or inspect.isclass(object) or
98 inspect.isroutine(object) or inspect.isframe(object) or
99 inspect.istraceback(object) or inspect.iscode(object))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000100
101def replace(text, *pairs):
102 """Do a series of global replacements on a string."""
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +0000103 while pairs:
104 text = join(split(text, pairs[0]), pairs[1])
105 pairs = pairs[2:]
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000106 return text
107
108def cram(text, maxlen):
109 """Omit part of a string if needed to make it fit in a maximum length."""
110 if len(text) > maxlen:
Raymond Hettingerfca3bb62002-10-21 04:44:11 +0000111 pre = max(0, (maxlen-3)//2)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000112 post = max(0, maxlen-3-pre)
113 return text[:pre] + '...' + text[len(text)-post:]
114 return text
115
Brett Cannon84601f12004-06-19 01:22:48 +0000116_re_stripid = re.compile(r' at 0x[0-9a-f]{6,16}(>+)$', re.IGNORECASE)
Ka-Ping Yee1d384632001-03-01 00:24:32 +0000117def stripid(text):
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000118 """Remove the hexadecimal id from a Python object representation."""
Brett Cannonc6c1f472004-06-19 01:02:51 +0000119 # The behaviour of %p is implementation-dependent in terms of case.
120 if _re_stripid.search(repr(Exception)):
121 return _re_stripid.sub(r'\1', text)
Ka-Ping Yee1d384632001-03-01 00:24:32 +0000122 return text
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000123
Brett Cannonc6c1f472004-06-19 01:02:51 +0000124def _is_some_method(obj):
125 return inspect.ismethod(obj) or inspect.ismethoddescriptor(obj)
Tim Peters536d2262001-09-20 05:13:38 +0000126
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000127def allmethods(cl):
128 methods = {}
Tim Peters536d2262001-09-20 05:13:38 +0000129 for key, value in inspect.getmembers(cl, _is_some_method):
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000130 methods[key] = 1
131 for base in cl.__bases__:
132 methods.update(allmethods(base)) # all your base are belong to us
133 for key in methods.keys():
134 methods[key] = getattr(cl, key)
135 return methods
136
Tim Petersfa26f7c2001-09-24 08:05:11 +0000137def _split_list(s, predicate):
138 """Split sequence s via predicate, and return pair ([true], [false]).
139
140 The return value is a 2-tuple of lists,
141 ([x for x in s if predicate(x)],
142 [x for x in s if not predicate(x)])
143 """
144
Tim Peters28355492001-09-23 21:29:55 +0000145 yes = []
146 no = []
Tim Petersfa26f7c2001-09-24 08:05:11 +0000147 for x in s:
148 if predicate(x):
149 yes.append(x)
Tim Peters28355492001-09-23 21:29:55 +0000150 else:
Tim Petersfa26f7c2001-09-24 08:05:11 +0000151 no.append(x)
Tim Peters28355492001-09-23 21:29:55 +0000152 return yes, no
153
Skip Montanaroa5616d22004-06-11 04:46:12 +0000154def visiblename(name, all=None):
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000155 """Decide whether to show documentation on a variable."""
156 # Certain special names are redundant.
Raymond Hettingerdbecd932005-02-06 06:57:08 +0000157 if name in ('__builtins__', '__doc__', '__file__', '__path__',
158 '__module__', '__name__', '__slots__'): return 0
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000159 # Private names are hidden, but special names are displayed.
160 if name.startswith('__') and name.endswith('__'): return 1
Skip Montanaroa5616d22004-06-11 04:46:12 +0000161 if all is not None:
162 # only document that which the programmer exported in __all__
163 return name in all
164 else:
165 return not name.startswith('_')
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000166
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +0000167def classify_class_attrs(object):
168 """Wrap inspect.classify_class_attrs, with fixup for data descriptors."""
169 def fixup((name, kind, cls, value)):
170 if inspect.isdatadescriptor(value):
171 kind = 'data descriptor'
172 return name, kind, cls, value
173 return map(fixup, inspect.classify_class_attrs(object))
174
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000175# ----------------------------------------------------- module manipulation
176
177def ispackage(path):
178 """Guess whether a path refers to a package directory."""
179 if os.path.isdir(path):
Raymond Hettingerdbecd932005-02-06 06:57:08 +0000180 for ext in ('.py', '.pyc', '.pyo'):
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000181 if os.path.isfile(os.path.join(path, '__init__' + ext)):
Tim Petersbc0e9102002-04-04 22:55:58 +0000182 return True
183 return False
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000184
185def synopsis(filename, cache={}):
186 """Get the one-line summary out of a module file."""
Raymond Hettinger32200ae2002-06-01 19:51:15 +0000187 mtime = os.stat(filename).st_mtime
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000188 lastupdate, result = cache.get(filename, (0, None))
189 if lastupdate < mtime:
190 info = inspect.getmoduleinfo(filename)
191 file = open(filename)
192 if info and 'b' in info[2]: # binary modules have to be imported
193 try: module = imp.load_module('__temp__', file, filename, info[1:])
194 except: return None
195 result = split(module.__doc__ or '', '\n')[0]
196 del sys.modules['__temp__']
197 else: # text modules can be directly examined
198 line = file.readline()
199 while line[:1] == '#' or not strip(line):
200 line = file.readline()
201 if not line: break
202 line = strip(line)
203 if line[:4] == 'r"""': line = line[1:]
204 if line[:3] == '"""':
205 line = line[3:]
206 if line[-1:] == '\\': line = line[:-1]
207 while not strip(line):
208 line = file.readline()
209 if not line: break
210 result = strip(split(line, '"""')[0])
211 else: result = None
212 file.close()
213 cache[filename] = (mtime, result)
214 return result
215
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000216class ErrorDuringImport(Exception):
217 """Errors that occurred while trying to import something to document it."""
218 def __init__(self, filename, (exc, value, tb)):
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000219 self.filename = filename
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000220 self.exc = exc
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000221 self.value = value
222 self.tb = tb
223
224 def __str__(self):
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000225 exc = self.exc
226 if type(exc) is types.ClassType:
227 exc = exc.__name__
228 return 'problem in %s - %s: %s' % (self.filename, exc, self.value)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000229
230def importfile(path):
231 """Import a Python source file or compiled file given its path."""
232 magic = imp.get_magic()
233 file = open(path, 'r')
234 if file.read(len(magic)) == magic:
235 kind = imp.PY_COMPILED
236 else:
237 kind = imp.PY_SOURCE
238 file.close()
239 filename = os.path.basename(path)
240 name, ext = os.path.splitext(filename)
241 file = open(path, 'r')
242 try:
243 module = imp.load_module(name, file, path, (ext, 'r', kind))
244 except:
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000245 raise ErrorDuringImport(path, sys.exc_info())
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000246 file.close()
247 return module
248
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000249def safeimport(path, forceload=0, cache={}):
250 """Import a module; handle errors; return None if the module isn't found.
251
252 If the module *is* found but an exception occurs, it's wrapped in an
253 ErrorDuringImport exception and reraised. Unlike __import__, if a
254 package path is specified, the module at the end of the path is returned,
255 not the package at the beginning. If the optional 'forceload' argument
256 is 1, we reload the module from disk (unless it's a dynamic extension)."""
Raymond Hettinger54f02222002-06-01 14:18:47 +0000257 if forceload and path in sys.modules:
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000258 # This is the only way to be sure. Checking the mtime of the file
259 # isn't good enough (e.g. what if the module contains a class that
260 # inherits from another module that has changed?).
261 if path not in sys.builtin_module_names:
262 # Python never loads a dynamic extension a second time from the
263 # same path, even if the file is changed or missing. Deleting
264 # the entry in sys.modules doesn't help for dynamic extensions,
265 # so we're not even going to try to keep them up to date.
266 info = inspect.getmoduleinfo(sys.modules[path].__file__)
267 if info[3] != imp.C_EXTENSION:
268 cache[path] = sys.modules[path] # prevent module from clearing
269 del sys.modules[path]
270 try:
271 module = __import__(path)
272 except:
273 # Did the error occur before or after the module was found?
274 (exc, value, tb) = info = sys.exc_info()
Raymond Hettinger54f02222002-06-01 14:18:47 +0000275 if path in sys.modules:
Fred Drakedb390c12005-10-28 14:39:47 +0000276 # An error occurred while executing the imported module.
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000277 raise ErrorDuringImport(sys.modules[path].__file__, info)
278 elif exc is SyntaxError:
279 # A SyntaxError occurred before we could execute the module.
280 raise ErrorDuringImport(value.filename, info)
281 elif exc is ImportError and \
282 split(lower(str(value)))[:2] == ['no', 'module']:
283 # The module was not found.
284 return None
285 else:
286 # Some other error occurred during the importing process.
287 raise ErrorDuringImport(path, sys.exc_info())
288 for part in split(path, '.')[1:]:
289 try: module = getattr(module, part)
290 except AttributeError: return None
291 return module
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000292
293# ---------------------------------------------------- formatter base class
294
295class Doc:
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000296 def document(self, object, name=None, *args):
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000297 """Generate documentation for an object."""
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000298 args = (object, name) + args
Brett Cannon28a4f0f2003-06-11 23:38:55 +0000299 # 'try' clause is to attempt to handle the possibility that inspect
300 # identifies something in a way that pydoc itself has issues handling;
301 # think 'super' and how it is a descriptor (which raises the exception
302 # by lacking a __name__ attribute) and an instance.
303 try:
304 if inspect.ismodule(object): return self.docmodule(*args)
305 if inspect.isclass(object): return self.docclass(*args)
306 if inspect.isroutine(object): return self.docroutine(*args)
307 except AttributeError:
308 pass
Johannes Gijsbers8de645a2004-11-07 19:16:05 +0000309 if isinstance(object, property): return self.docproperty(*args)
Guido van Rossum68468eb2003-02-27 20:14:51 +0000310 return self.docother(*args)
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000311
312 def fail(self, object, name=None, *args):
313 """Raise an exception for unimplemented types."""
314 message = "don't know how to document object%s of type %s" % (
315 name and ' ' + repr(name), type(object).__name__)
316 raise TypeError, message
317
318 docmodule = docclass = docroutine = docother = fail
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000319
Skip Montanaro4997a692003-09-10 16:47:51 +0000320 def getdocloc(self, object):
321 """Return the location of module docs or None"""
322
323 try:
324 file = inspect.getabsfile(object)
325 except TypeError:
326 file = '(built-in)'
327
328 docloc = os.environ.get("PYTHONDOCS",
329 "http://www.python.org/doc/current/lib")
330 basedir = os.path.join(sys.exec_prefix, "lib",
331 "python"+sys.version[0:3])
332 if (isinstance(object, type(os)) and
333 (object.__name__ in ('errno', 'exceptions', 'gc', 'imp',
334 'marshal', 'posix', 'signal', 'sys',
335 'thread', 'zipimport') or
336 (file.startswith(basedir) and
337 not file.startswith(os.path.join(basedir, 'site-packages'))))):
Skip Montanarof2134842004-06-07 02:40:05 +0000338 htmlfile = "module-%s.html" % object.__name__
Skip Montanaro4997a692003-09-10 16:47:51 +0000339 if docloc.startswith("http://"):
Skip Montanarof2134842004-06-07 02:40:05 +0000340 docloc = "%s/%s" % (docloc.rstrip("/"), htmlfile)
Skip Montanaro4997a692003-09-10 16:47:51 +0000341 else:
Skip Montanarof2134842004-06-07 02:40:05 +0000342 docloc = os.path.join(docloc, htmlfile)
Skip Montanaro4997a692003-09-10 16:47:51 +0000343 else:
344 docloc = None
345 return docloc
346
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000347# -------------------------------------------- HTML documentation generator
348
349class HTMLRepr(Repr):
350 """Class for safely making an HTML representation of a Python object."""
351 def __init__(self):
352 Repr.__init__(self)
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000353 self.maxlist = self.maxtuple = 20
354 self.maxdict = 10
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000355 self.maxstring = self.maxother = 100
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000356
357 def escape(self, text):
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +0000358 return replace(text, '&', '&amp;', '<', '&lt;', '>', '&gt;')
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000359
360 def repr(self, object):
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000361 return Repr.repr(self, object)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000362
363 def repr1(self, x, level):
Skip Montanaro0fe8fce2003-06-27 15:45:41 +0000364 if hasattr(type(x), '__name__'):
365 methodname = 'repr_' + join(split(type(x).__name__), '_')
366 if hasattr(self, methodname):
367 return getattr(self, methodname)(x, level)
368 return self.escape(cram(stripid(repr(x)), self.maxother))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000369
370 def repr_string(self, x, level):
Ka-Ping Yeea2fe1032001-03-02 01:19:14 +0000371 test = cram(x, self.maxstring)
372 testrepr = repr(test)
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +0000373 if '\\' in test and '\\' not in replace(testrepr, r'\\', ''):
Ka-Ping Yeea2fe1032001-03-02 01:19:14 +0000374 # Backslashes are only literal in the string and are never
375 # needed to make any special characters, so show a raw string.
376 return 'r' + testrepr[0] + self.escape(test) + testrepr[0]
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +0000377 return re.sub(r'((\\[\\abfnrtv\'"]|\\[0-9]..|\\x..|\\u....)+)',
Ka-Ping Yeea2fe1032001-03-02 01:19:14 +0000378 r'<font color="#c040c0">\1</font>',
379 self.escape(testrepr))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000380
Skip Montanarodf708782002-03-07 22:58:02 +0000381 repr_str = repr_string
382
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000383 def repr_instance(self, x, level):
384 try:
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000385 return self.escape(cram(stripid(repr(x)), self.maxstring))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000386 except:
387 return self.escape('<%s instance>' % x.__class__.__name__)
388
389 repr_unicode = repr_string
390
391class HTMLDoc(Doc):
392 """Formatter class for HTML documentation."""
393
394 # ------------------------------------------- HTML formatting utilities
395
396 _repr_instance = HTMLRepr()
397 repr = _repr_instance.repr
398 escape = _repr_instance.escape
399
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000400 def page(self, title, contents):
401 """Format an HTML page."""
402 return '''
Tim Peters59ed4482001-10-31 04:20:26 +0000403<!doctype html PUBLIC "-//W3C//DTD HTML 4.0 Transitional//EN">
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000404<html><head><title>Python: %s</title>
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000405</head><body bgcolor="#f0f0f8">
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000406%s
407</body></html>''' % (title, contents)
408
409 def heading(self, title, fgcol, bgcol, extras=''):
410 """Format a page heading."""
411 return '''
Tim Peters59ed4482001-10-31 04:20:26 +0000412<table width="100%%" cellspacing=0 cellpadding=2 border=0 summary="heading">
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000413<tr bgcolor="%s">
Tim Peters2306d242001-09-25 03:18:32 +0000414<td valign=bottom>&nbsp;<br>
415<font color="%s" face="helvetica, arial">&nbsp;<br>%s</font></td
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000416><td align=right valign=bottom
Ka-Ping Yee987ec902001-03-23 13:35:45 +0000417><font color="%s" face="helvetica, arial">%s</font></td></tr></table>
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000418 ''' % (bgcol, fgcol, title, fgcol, extras or '&nbsp;')
419
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000420 def section(self, title, fgcol, bgcol, contents, width=6,
421 prelude='', marginalia=None, gap='&nbsp;'):
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000422 """Format a section with a heading."""
423 if marginalia is None:
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +0000424 marginalia = '<tt>' + '&nbsp;' * width + '</tt>'
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000425 result = '''<p>
Tim Peters59ed4482001-10-31 04:20:26 +0000426<table width="100%%" cellspacing=0 cellpadding=2 border=0 summary="section">
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000427<tr bgcolor="%s">
Tim Peters2306d242001-09-25 03:18:32 +0000428<td colspan=3 valign=bottom>&nbsp;<br>
429<font color="%s" face="helvetica, arial">%s</font></td></tr>
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000430 ''' % (bgcol, fgcol, title)
431 if prelude:
432 result = result + '''
Ka-Ping Yee987ec902001-03-23 13:35:45 +0000433<tr bgcolor="%s"><td rowspan=2>%s</td>
434<td colspan=2>%s</td></tr>
435<tr><td>%s</td>''' % (bgcol, marginalia, prelude, gap)
436 else:
437 result = result + '''
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000438<tr><td bgcolor="%s">%s</td><td>%s</td>''' % (bgcol, marginalia, gap)
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000439
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +0000440 return result + '\n<td width="100%%">%s</td></tr></table>' % contents
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000441
442 def bigsection(self, title, *args):
443 """Format a section with a big heading."""
444 title = '<big><strong>%s</strong></big>' % title
Guido van Rossum68468eb2003-02-27 20:14:51 +0000445 return self.section(title, *args)
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000446
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000447 def preformat(self, text):
448 """Format literal preformatted text."""
449 text = self.escape(expandtabs(text))
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +0000450 return replace(text, '\n\n', '\n \n', '\n\n', '\n \n',
451 ' ', '&nbsp;', '\n', '<br>\n')
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000452
453 def multicolumn(self, list, format, cols=4):
454 """Format a list of items into a multi-column list."""
455 result = ''
456 rows = (len(list)+cols-1)/cols
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000457 for col in range(cols):
458 result = result + '<td width="%d%%" valign=top>' % (100/cols)
459 for i in range(rows*col, rows*col+rows):
460 if i < len(list):
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000461 result = result + format(list[i]) + '<br>\n'
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000462 result = result + '</td>'
Tim Peters59ed4482001-10-31 04:20:26 +0000463 return '<table width="100%%" summary="list"><tr>%s</tr></table>' % result
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000464
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000465 def grey(self, text): return '<font color="#909090">%s</font>' % text
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000466
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000467 def namelink(self, name, *dicts):
468 """Make a link for an identifier, given name-to-URL mappings."""
469 for dict in dicts:
Raymond Hettinger54f02222002-06-01 14:18:47 +0000470 if name in dict:
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000471 return '<a href="%s">%s</a>' % (dict[name], name)
472 return name
473
Ka-Ping Yeeb7a48302001-04-12 20:39:14 +0000474 def classlink(self, object, modname):
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000475 """Make a link for a class."""
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000476 name, module = object.__name__, sys.modules.get(object.__module__)
477 if hasattr(module, name) and getattr(module, name) is object:
Ka-Ping Yeeb7a48302001-04-12 20:39:14 +0000478 return '<a href="%s.html#%s">%s</a>' % (
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000479 module.__name__, name, classname(object, modname))
480 return classname(object, modname)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000481
482 def modulelink(self, object):
483 """Make a link for a module."""
484 return '<a href="%s.html">%s</a>' % (object.__name__, object.__name__)
485
486 def modpkglink(self, (name, path, ispackage, shadowed)):
487 """Make a link for a module or package to display in an index."""
488 if shadowed:
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000489 return self.grey(name)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000490 if path:
491 url = '%s.%s.html' % (path, name)
492 else:
493 url = '%s.html' % name
494 if ispackage:
495 text = '<strong>%s</strong>&nbsp;(package)' % name
496 else:
497 text = name
498 return '<a href="%s">%s</a>' % (url, text)
499
500 def markup(self, text, escape=None, funcs={}, classes={}, methods={}):
501 """Mark up some plain text, given a context of symbols to look for.
502 Each context dictionary maps object names to anchor names."""
503 escape = escape or self.escape
504 results = []
505 here = 0
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000506 pattern = re.compile(r'\b((http|ftp)://\S+[\w/]|'
507 r'RFC[- ]?(\d+)|'
Ka-Ping Yeef78a81b2001-03-27 08:13:42 +0000508 r'PEP[- ]?(\d+)|'
Neil Schemenauerd69711c2002-03-24 23:02:07 +0000509 r'(self\.)?(\w+))')
Guido van Rossum8ca162f2002-04-07 06:36:23 +0000510 while True:
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000511 match = pattern.search(text, here)
512 if not match: break
513 start, end = match.span()
514 results.append(escape(text[here:start]))
515
Ka-Ping Yeef78a81b2001-03-27 08:13:42 +0000516 all, scheme, rfc, pep, selfdot, name = match.groups()
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000517 if scheme:
Neil Schemenauercddc1a02002-03-24 23:11:21 +0000518 url = escape(all).replace('"', '&quot;')
519 results.append('<a href="%s">%s</a>' % (url, url))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000520 elif rfc:
Ka-Ping Yeef78a81b2001-03-27 08:13:42 +0000521 url = 'http://www.rfc-editor.org/rfc/rfc%d.txt' % int(rfc)
522 results.append('<a href="%s">%s</a>' % (url, escape(all)))
523 elif pep:
524 url = 'http://www.python.org/peps/pep-%04d.html' % int(pep)
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000525 results.append('<a href="%s">%s</a>' % (url, escape(all)))
526 elif text[end:end+1] == '(':
527 results.append(self.namelink(name, methods, funcs, classes))
528 elif selfdot:
529 results.append('self.<strong>%s</strong>' % name)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000530 else:
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000531 results.append(self.namelink(name, classes))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000532 here = end
533 results.append(escape(text[here:]))
534 return join(results, '')
535
536 # ---------------------------------------------- type-specific routines
537
Ka-Ping Yeeb7a48302001-04-12 20:39:14 +0000538 def formattree(self, tree, modname, parent=None):
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000539 """Produce HTML for a class tree as given by inspect.getclasstree()."""
540 result = ''
541 for entry in tree:
542 if type(entry) is type(()):
543 c, bases = entry
Tim Peters2306d242001-09-25 03:18:32 +0000544 result = result + '<dt><font face="helvetica, arial">'
Ka-Ping Yeeb7a48302001-04-12 20:39:14 +0000545 result = result + self.classlink(c, modname)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000546 if bases and bases != (parent,):
547 parents = []
548 for base in bases:
Ka-Ping Yeeb7a48302001-04-12 20:39:14 +0000549 parents.append(self.classlink(base, modname))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000550 result = result + '(' + join(parents, ', ') + ')'
Tim Peters2306d242001-09-25 03:18:32 +0000551 result = result + '\n</font></dt>'
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000552 elif type(entry) is type([]):
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000553 result = result + '<dd>\n%s</dd>\n' % self.formattree(
Ka-Ping Yeeb7a48302001-04-12 20:39:14 +0000554 entry, modname, c)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000555 return '<dl>\n%s</dl>\n' % result
556
Tim Peters8dd7ade2001-10-18 19:56:17 +0000557 def docmodule(self, object, name=None, mod=None, *ignored):
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000558 """Produce HTML documentation for a module object."""
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000559 name = object.__name__ # ignore the passed-in name
Skip Montanaroa5616d22004-06-11 04:46:12 +0000560 try:
561 all = object.__all__
562 except AttributeError:
563 all = None
Ka-Ping Yee66efbc72001-03-01 13:55:20 +0000564 parts = split(name, '.')
565 links = []
566 for i in range(len(parts)-1):
567 links.append(
568 '<a href="%s.html"><font color="#ffffff">%s</font></a>' %
569 (join(parts[:i+1], '.'), parts[i]))
570 linkedname = join(links + parts[-1:], '.')
571 head = '<big><big><strong>%s</strong></big></big>' % linkedname
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000572 try:
Ka-Ping Yee239432a2001-03-02 02:45:08 +0000573 path = inspect.getabsfile(object)
Ka-Ping Yee6191a232001-04-13 15:00:27 +0000574 url = path
575 if sys.platform == 'win32':
576 import nturl2path
577 url = nturl2path.pathname2url(path)
578 filelink = '<a href="file:%s">%s</a>' % (url, path)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000579 except TypeError:
580 filelink = '(built-in)'
Ka-Ping Yee6f3f9a42001-02-27 22:42:36 +0000581 info = []
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000582 if hasattr(object, '__version__'):
Ka-Ping Yee6f3f9a42001-02-27 22:42:36 +0000583 version = str(object.__version__)
Ka-Ping Yee40c49912001-02-27 22:46:01 +0000584 if version[:11] == '$' + 'Revision: ' and version[-1:] == '$':
585 version = strip(version[11:-1])
Ka-Ping Yee1d384632001-03-01 00:24:32 +0000586 info.append('version %s' % self.escape(version))
Ka-Ping Yee6f3f9a42001-02-27 22:42:36 +0000587 if hasattr(object, '__date__'):
588 info.append(self.escape(str(object.__date__)))
589 if info:
590 head = head + ' (%s)' % join(info, ', ')
Skip Montanaro4997a692003-09-10 16:47:51 +0000591 docloc = self.getdocloc(object)
592 if docloc is not None:
593 docloc = '<br><a href="%(docloc)s">Module Docs</a>' % locals()
594 else:
595 docloc = ''
Ka-Ping Yee66efbc72001-03-01 13:55:20 +0000596 result = self.heading(
Skip Montanaro4997a692003-09-10 16:47:51 +0000597 head, '#ffffff', '#7799ee',
598 '<a href=".">index</a><br>' + filelink + docloc)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000599
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000600 modules = inspect.getmembers(object, inspect.ismodule)
601
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000602 classes, cdict = [], {}
603 for key, value in inspect.getmembers(object, inspect.isclass):
Johannes Gijsbers4c11f602004-08-30 14:13:04 +0000604 # if __all__ exists, believe it. Otherwise use old heuristic.
605 if (all is not None or
606 (inspect.getmodule(value) or object) is object):
Skip Montanaroa5616d22004-06-11 04:46:12 +0000607 if visiblename(key, all):
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000608 classes.append((key, value))
609 cdict[key] = cdict[value] = '#' + key
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000610 for key, value in classes:
611 for base in value.__bases__:
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000612 key, modname = base.__name__, base.__module__
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000613 module = sys.modules.get(modname)
614 if modname != name and module and hasattr(module, key):
615 if getattr(module, key) is base:
Raymond Hettinger54f02222002-06-01 14:18:47 +0000616 if not key in cdict:
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000617 cdict[key] = cdict[base] = modname + '.html#' + key
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000618 funcs, fdict = [], {}
619 for key, value in inspect.getmembers(object, inspect.isroutine):
Johannes Gijsbers4c11f602004-08-30 14:13:04 +0000620 # if __all__ exists, believe it. Otherwise use old heuristic.
621 if (all is not None or
622 inspect.isbuiltin(value) or inspect.getmodule(value) is object):
Skip Montanaroa5616d22004-06-11 04:46:12 +0000623 if visiblename(key, all):
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000624 funcs.append((key, value))
625 fdict[key] = '#-' + key
626 if inspect.isfunction(value): fdict[value] = fdict[key]
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000627 data = []
628 for key, value in inspect.getmembers(object, isdata):
Skip Montanaroa5616d22004-06-11 04:46:12 +0000629 if visiblename(key, all):
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000630 data.append((key, value))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000631
632 doc = self.markup(getdoc(object), self.preformat, fdict, cdict)
633 doc = doc and '<tt>%s</tt>' % doc
Tim Peters2306d242001-09-25 03:18:32 +0000634 result = result + '<p>%s</p>\n' % doc
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000635
636 if hasattr(object, '__path__'):
637 modpkgs = []
638 modnames = []
639 for file in os.listdir(object.__path__[0]):
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +0000640 path = os.path.join(object.__path__[0], file)
641 modname = inspect.getmodulename(file)
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000642 if modname != '__init__':
643 if modname and modname not in modnames:
644 modpkgs.append((modname, name, 0, 0))
645 modnames.append(modname)
646 elif ispackage(path):
647 modpkgs.append((file, name, 1, 0))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000648 modpkgs.sort()
649 contents = self.multicolumn(modpkgs, self.modpkglink)
650 result = result + self.bigsection(
651 'Package Contents', '#ffffff', '#aa55cc', contents)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000652 elif modules:
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000653 contents = self.multicolumn(
654 modules, lambda (key, value), s=self: s.modulelink(value))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000655 result = result + self.bigsection(
656 'Modules', '#fffff', '#aa55cc', contents)
657
658 if classes:
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000659 classlist = map(lambda (key, value): value, classes)
Ka-Ping Yeeb7a48302001-04-12 20:39:14 +0000660 contents = [
661 self.formattree(inspect.getclasstree(classlist, 1), name)]
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000662 for key, value in classes:
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +0000663 contents.append(self.document(value, key, name, fdict, cdict))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000664 result = result + self.bigsection(
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000665 'Classes', '#ffffff', '#ee77aa', join(contents))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000666 if funcs:
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000667 contents = []
668 for key, value in funcs:
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +0000669 contents.append(self.document(value, key, name, fdict, cdict))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000670 result = result + self.bigsection(
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000671 'Functions', '#ffffff', '#eeaa77', join(contents))
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000672 if data:
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000673 contents = []
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000674 for key, value in data:
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000675 contents.append(self.document(value, key))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000676 result = result + self.bigsection(
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000677 'Data', '#ffffff', '#55aa55', join(contents, '<br>\n'))
Ka-Ping Yee6f3f9a42001-02-27 22:42:36 +0000678 if hasattr(object, '__author__'):
679 contents = self.markup(str(object.__author__), self.preformat)
680 result = result + self.bigsection(
681 'Author', '#ffffff', '#7799ee', contents)
Ka-Ping Yee6f3f9a42001-02-27 22:42:36 +0000682 if hasattr(object, '__credits__'):
683 contents = self.markup(str(object.__credits__), self.preformat)
684 result = result + self.bigsection(
685 'Credits', '#ffffff', '#7799ee', contents)
686
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000687 return result
688
Tim Peters8dd7ade2001-10-18 19:56:17 +0000689 def docclass(self, object, name=None, mod=None, funcs={}, classes={},
690 *ignored):
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000691 """Produce HTML documentation for a class object."""
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000692 realname = object.__name__
693 name = name or realname
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000694 bases = object.__bases__
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000695
Tim Petersb47879b2001-09-24 04:47:19 +0000696 contents = []
697 push = contents.append
698
Tim Petersfa26f7c2001-09-24 08:05:11 +0000699 # Cute little class to pump out a horizontal rule between sections.
700 class HorizontalRule:
701 def __init__(self):
702 self.needone = 0
703 def maybe(self):
704 if self.needone:
705 push('<hr>\n')
706 self.needone = 1
707 hr = HorizontalRule()
708
Tim Petersc86f6ca2001-09-26 21:31:51 +0000709 # List the mro, if non-trivial.
Raymond Hettinger756b3f32004-01-29 06:37:52 +0000710 mro = deque(inspect.getmro(object))
Tim Petersc86f6ca2001-09-26 21:31:51 +0000711 if len(mro) > 2:
712 hr.maybe()
713 push('<dl><dt>Method resolution order:</dt>\n')
714 for base in mro:
715 push('<dd>%s</dd>\n' % self.classlink(base,
716 object.__module__))
717 push('</dl>\n')
718
Tim Petersb47879b2001-09-24 04:47:19 +0000719 def spill(msg, attrs, predicate):
Tim Petersfa26f7c2001-09-24 08:05:11 +0000720 ok, attrs = _split_list(attrs, predicate)
Tim Petersb47879b2001-09-24 04:47:19 +0000721 if ok:
Tim Petersfa26f7c2001-09-24 08:05:11 +0000722 hr.maybe()
Tim Petersb47879b2001-09-24 04:47:19 +0000723 push(msg)
724 for name, kind, homecls, value in ok:
725 push(self.document(getattr(object, name), name, mod,
726 funcs, classes, mdict, object))
727 push('\n')
728 return attrs
729
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +0000730 def spilldescriptors(msg, attrs, predicate):
Tim Petersfa26f7c2001-09-24 08:05:11 +0000731 ok, attrs = _split_list(attrs, predicate)
Tim Petersb47879b2001-09-24 04:47:19 +0000732 if ok:
Tim Petersfa26f7c2001-09-24 08:05:11 +0000733 hr.maybe()
Tim Petersb47879b2001-09-24 04:47:19 +0000734 push(msg)
735 for name, kind, homecls, value in ok:
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +0000736 push(self._docdescriptor(name, value, mod))
Tim Petersb47879b2001-09-24 04:47:19 +0000737 return attrs
738
Tim Petersfa26f7c2001-09-24 08:05:11 +0000739 def spilldata(msg, attrs, predicate):
740 ok, attrs = _split_list(attrs, predicate)
Tim Petersb47879b2001-09-24 04:47:19 +0000741 if ok:
Tim Petersfa26f7c2001-09-24 08:05:11 +0000742 hr.maybe()
Tim Petersb47879b2001-09-24 04:47:19 +0000743 push(msg)
744 for name, kind, homecls, value in ok:
745 base = self.docother(getattr(object, name), name, mod)
Martin v. Löwise59e2ba2003-05-03 09:09:02 +0000746 if callable(value) or inspect.isdatadescriptor(value):
Guido van Rossum5e355b22002-05-21 20:56:15 +0000747 doc = getattr(value, "__doc__", None)
748 else:
749 doc = None
Tim Petersb47879b2001-09-24 04:47:19 +0000750 if doc is None:
751 push('<dl><dt>%s</dl>\n' % base)
752 else:
753 doc = self.markup(getdoc(value), self.preformat,
754 funcs, classes, mdict)
Tim Peters2306d242001-09-25 03:18:32 +0000755 doc = '<dd><tt>%s</tt>' % doc
Tim Petersb47879b2001-09-24 04:47:19 +0000756 push('<dl><dt>%s%s</dl>\n' % (base, doc))
757 push('\n')
758 return attrs
759
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000760 attrs = filter(lambda (name, kind, cls, value): visiblename(name),
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +0000761 classify_class_attrs(object))
Tim Petersb47879b2001-09-24 04:47:19 +0000762 mdict = {}
763 for key, kind, homecls, value in attrs:
764 mdict[key] = anchor = '#' + name + '-' + key
765 value = getattr(object, key)
766 try:
767 # The value may not be hashable (e.g., a data attr with
768 # a dict or list value).
769 mdict[value] = anchor
770 except TypeError:
771 pass
772
Tim Petersfa26f7c2001-09-24 08:05:11 +0000773 while attrs:
Tim Peters351e3622001-09-27 03:29:51 +0000774 if mro:
Raymond Hettinger756b3f32004-01-29 06:37:52 +0000775 thisclass = mro.popleft()
Tim Peters351e3622001-09-27 03:29:51 +0000776 else:
777 thisclass = attrs[0][2]
Tim Petersfa26f7c2001-09-24 08:05:11 +0000778 attrs, inherited = _split_list(attrs, lambda t: t[2] is thisclass)
779
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000780 if thisclass is __builtin__.object:
781 attrs = inherited
782 continue
783 elif thisclass is object:
784 tag = 'defined here'
Tim Petersb47879b2001-09-24 04:47:19 +0000785 else:
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000786 tag = 'inherited from %s' % self.classlink(thisclass,
787 object.__module__)
Tim Petersb47879b2001-09-24 04:47:19 +0000788 tag += ':<br>\n'
789
790 # Sort attrs by name.
Raymond Hettinger6b59f5f2003-10-16 05:53:16 +0000791 attrs.sort(key=lambda t: t[0])
Tim Petersb47879b2001-09-24 04:47:19 +0000792
793 # Pump out the attrs, segregated by kind.
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000794 attrs = spill('Methods %s' % tag, attrs,
Tim Petersb47879b2001-09-24 04:47:19 +0000795 lambda t: t[1] == 'method')
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000796 attrs = spill('Class methods %s' % tag, attrs,
Tim Petersb47879b2001-09-24 04:47:19 +0000797 lambda t: t[1] == 'class method')
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000798 attrs = spill('Static methods %s' % tag, attrs,
Tim Petersb47879b2001-09-24 04:47:19 +0000799 lambda t: t[1] == 'static method')
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +0000800 attrs = spilldescriptors('Data descriptors %s' % tag, attrs,
801 lambda t: t[1] == 'data descriptor')
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000802 attrs = spilldata('Data and other attributes %s' % tag, attrs,
Tim Petersfa26f7c2001-09-24 08:05:11 +0000803 lambda t: t[1] == 'data')
Tim Petersb47879b2001-09-24 04:47:19 +0000804 assert attrs == []
Tim Peters351e3622001-09-27 03:29:51 +0000805 attrs = inherited
Tim Petersb47879b2001-09-24 04:47:19 +0000806
807 contents = ''.join(contents)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000808
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000809 if name == realname:
810 title = '<a name="%s">class <strong>%s</strong></a>' % (
811 name, realname)
812 else:
813 title = '<strong>%s</strong> = <a name="%s">class %s</a>' % (
814 name, name, realname)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000815 if bases:
816 parents = []
817 for base in bases:
Ka-Ping Yeeb7a48302001-04-12 20:39:14 +0000818 parents.append(self.classlink(base, object.__module__))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000819 title = title + '(%s)' % join(parents, ', ')
Tim Peters2306d242001-09-25 03:18:32 +0000820 doc = self.markup(getdoc(object), self.preformat, funcs, classes, mdict)
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000821 doc = doc and '<tt>%s<br>&nbsp;</tt>' % doc
Tim Petersc86f6ca2001-09-26 21:31:51 +0000822
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000823 return self.section(title, '#000000', '#ffc8d8', contents, 3, doc)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000824
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000825 def formatvalue(self, object):
826 """Format an argument default value as text."""
Tim Peters2306d242001-09-25 03:18:32 +0000827 return self.grey('=' + self.repr(object))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000828
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +0000829 def docroutine(self, object, name=None, mod=None,
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000830 funcs={}, classes={}, methods={}, cl=None):
Ka-Ping Yee66efbc72001-03-01 13:55:20 +0000831 """Produce HTML documentation for a function or method object."""
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000832 realname = object.__name__
833 name = name or realname
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000834 anchor = (cl and cl.__name__ or '') + '-' + name
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000835 note = ''
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000836 skipdocs = 0
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000837 if inspect.ismethod(object):
Ka-Ping Yee6dcfa382001-04-12 20:27:31 +0000838 imclass = object.im_class
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000839 if cl:
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +0000840 if imclass is not cl:
Ka-Ping Yeeb7a48302001-04-12 20:39:14 +0000841 note = ' from ' + self.classlink(imclass, mod)
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000842 else:
Ka-Ping Yeeb7a48302001-04-12 20:39:14 +0000843 if object.im_self:
844 note = ' method of %s instance' % self.classlink(
845 object.im_self.__class__, mod)
846 else:
847 note = ' unbound %s method' % self.classlink(imclass,mod)
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000848 object = object.im_func
849
850 if name == realname:
851 title = '<a name="%s"><strong>%s</strong></a>' % (anchor, realname)
852 else:
Raymond Hettinger54f02222002-06-01 14:18:47 +0000853 if (cl and realname in cl.__dict__ and
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000854 cl.__dict__[realname] is object):
Ka-Ping Yeee280c062001-03-23 14:05:53 +0000855 reallink = '<a href="#%s">%s</a>' % (
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000856 cl.__name__ + '-' + realname, realname)
857 skipdocs = 1
858 else:
859 reallink = realname
860 title = '<a name="%s"><strong>%s</strong></a> = %s' % (
861 anchor, name, reallink)
Tim Peters4bcfa312001-09-20 06:08:24 +0000862 if inspect.isfunction(object):
Ka-Ping Yee66efbc72001-03-01 13:55:20 +0000863 args, varargs, varkw, defaults = inspect.getargspec(object)
864 argspec = inspect.formatargspec(
865 args, varargs, varkw, defaults, formatvalue=self.formatvalue)
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000866 if realname == '<lambda>':
Tim Peters59ed4482001-10-31 04:20:26 +0000867 title = '<strong>%s</strong> <em>lambda</em> ' % name
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000868 argspec = argspec[1:-1] # remove parentheses
Tim Peters4bcfa312001-09-20 06:08:24 +0000869 else:
870 argspec = '(...)'
Ka-Ping Yee66efbc72001-03-01 13:55:20 +0000871
Tim Peters2306d242001-09-25 03:18:32 +0000872 decl = title + argspec + (note and self.grey(
873 '<font face="helvetica, arial">%s</font>' % note))
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000874
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000875 if skipdocs:
Tim Peters2306d242001-09-25 03:18:32 +0000876 return '<dl><dt>%s</dt></dl>\n' % decl
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000877 else:
878 doc = self.markup(
879 getdoc(object), self.preformat, funcs, classes, methods)
Tim Peters2306d242001-09-25 03:18:32 +0000880 doc = doc and '<dd><tt>%s</tt></dd>' % doc
881 return '<dl><dt>%s</dt>%s</dl>\n' % (decl, doc)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000882
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +0000883 def _docdescriptor(self, name, value, mod):
Johannes Gijsbers8de645a2004-11-07 19:16:05 +0000884 results = []
885 push = results.append
886
887 if name:
888 push('<dl><dt><strong>%s</strong></dt>\n' % name)
889 if value.__doc__ is not None:
Ka-Ping Yeebba6acc2005-02-19 22:58:26 +0000890 doc = self.markup(getdoc(value), self.preformat)
Johannes Gijsbers8de645a2004-11-07 19:16:05 +0000891 push('<dd><tt>%s</tt></dd>\n' % doc)
Johannes Gijsbers8de645a2004-11-07 19:16:05 +0000892 push('</dl>\n')
893
894 return ''.join(results)
895
896 def docproperty(self, object, name=None, mod=None, cl=None):
897 """Produce html documentation for a property."""
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +0000898 return self._docdescriptor(name, object, mod)
Johannes Gijsbers8de645a2004-11-07 19:16:05 +0000899
Tim Peters8dd7ade2001-10-18 19:56:17 +0000900 def docother(self, object, name=None, mod=None, *ignored):
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000901 """Produce HTML documentation for a data object."""
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +0000902 lhs = name and '<strong>%s</strong> = ' % name or ''
903 return lhs + self.repr(object)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000904
905 def index(self, dir, shadowed=None):
906 """Generate an HTML index for a directory of modules."""
907 modpkgs = []
908 if shadowed is None: shadowed = {}
909 seen = {}
910 files = os.listdir(dir)
911
912 def found(name, ispackage,
913 modpkgs=modpkgs, shadowed=shadowed, seen=seen):
Raymond Hettinger4f759d82002-11-02 02:02:46 +0000914 if name not in seen:
915 modpkgs.append((name, '', ispackage, name in shadowed))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000916 seen[name] = 1
917 shadowed[name] = 1
918
919 # Package spam/__init__.py takes precedence over module spam.py.
920 for file in files:
921 path = os.path.join(dir, file)
922 if ispackage(path): found(file, 1)
923 for file in files:
924 path = os.path.join(dir, file)
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +0000925 if os.path.isfile(path):
926 modname = inspect.getmodulename(file)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000927 if modname: found(modname, 0)
928
929 modpkgs.sort()
930 contents = self.multicolumn(modpkgs, self.modpkglink)
931 return self.bigsection(dir, '#ffffff', '#ee77aa', contents)
932
933# -------------------------------------------- text documentation generator
934
935class TextRepr(Repr):
936 """Class for safely making a text representation of a Python object."""
937 def __init__(self):
938 Repr.__init__(self)
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000939 self.maxlist = self.maxtuple = 20
940 self.maxdict = 10
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000941 self.maxstring = self.maxother = 100
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000942
943 def repr1(self, x, level):
Skip Montanaro0fe8fce2003-06-27 15:45:41 +0000944 if hasattr(type(x), '__name__'):
945 methodname = 'repr_' + join(split(type(x).__name__), '_')
946 if hasattr(self, methodname):
947 return getattr(self, methodname)(x, level)
948 return cram(stripid(repr(x)), self.maxother)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000949
Ka-Ping Yeea2fe1032001-03-02 01:19:14 +0000950 def repr_string(self, x, level):
951 test = cram(x, self.maxstring)
952 testrepr = repr(test)
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +0000953 if '\\' in test and '\\' not in replace(testrepr, r'\\', ''):
Ka-Ping Yeea2fe1032001-03-02 01:19:14 +0000954 # Backslashes are only literal in the string and are never
955 # needed to make any special characters, so show a raw string.
956 return 'r' + testrepr[0] + test + testrepr[0]
957 return testrepr
958
Skip Montanarodf708782002-03-07 22:58:02 +0000959 repr_str = repr_string
960
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000961 def repr_instance(self, x, level):
962 try:
Ka-Ping Yee1d384632001-03-01 00:24:32 +0000963 return cram(stripid(repr(x)), self.maxstring)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000964 except:
965 return '<%s instance>' % x.__class__.__name__
966
967class TextDoc(Doc):
968 """Formatter class for text documentation."""
969
970 # ------------------------------------------- text formatting utilities
971
972 _repr_instance = TextRepr()
973 repr = _repr_instance.repr
974
975 def bold(self, text):
976 """Format a string in bold by overstriking."""
977 return join(map(lambda ch: ch + '\b' + ch, text), '')
978
979 def indent(self, text, prefix=' '):
980 """Indent text by prepending a given prefix to each line."""
981 if not text: return ''
982 lines = split(text, '\n')
983 lines = map(lambda line, prefix=prefix: prefix + line, lines)
984 if lines: lines[-1] = rstrip(lines[-1])
985 return join(lines, '\n')
986
987 def section(self, title, contents):
988 """Format a section with a given heading."""
989 return self.bold(title) + '\n' + rstrip(self.indent(contents)) + '\n\n'
990
991 # ---------------------------------------------- type-specific routines
992
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000993 def formattree(self, tree, modname, parent=None, prefix=''):
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000994 """Render in text a class tree as returned by inspect.getclasstree()."""
995 result = ''
996 for entry in tree:
997 if type(entry) is type(()):
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000998 c, bases = entry
999 result = result + prefix + classname(c, modname)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001000 if bases and bases != (parent,):
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001001 parents = map(lambda c, m=modname: classname(c, m), bases)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001002 result = result + '(%s)' % join(parents, ', ')
1003 result = result + '\n'
1004 elif type(entry) is type([]):
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001005 result = result + self.formattree(
1006 entry, modname, c, prefix + ' ')
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001007 return result
1008
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001009 def docmodule(self, object, name=None, mod=None):
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001010 """Produce text documentation for a given module object."""
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001011 name = object.__name__ # ignore the passed-in name
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +00001012 synop, desc = splitdoc(getdoc(object))
1013 result = self.section('NAME', name + (synop and ' - ' + synop))
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001014
1015 try:
Skip Montanaroa5616d22004-06-11 04:46:12 +00001016 all = object.__all__
1017 except AttributeError:
1018 all = None
1019
1020 try:
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001021 file = inspect.getabsfile(object)
1022 except TypeError:
1023 file = '(built-in)'
Ka-Ping Yee239432a2001-03-02 02:45:08 +00001024 result = result + self.section('FILE', file)
Skip Montanaro4997a692003-09-10 16:47:51 +00001025
1026 docloc = self.getdocloc(object)
1027 if docloc is not None:
1028 result = result + self.section('MODULE DOCS', docloc)
1029
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +00001030 if desc:
1031 result = result + self.section('DESCRIPTION', desc)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001032
1033 classes = []
1034 for key, value in inspect.getmembers(object, inspect.isclass):
Johannes Gijsbers4c11f602004-08-30 14:13:04 +00001035 # if __all__ exists, believe it. Otherwise use old heuristic.
1036 if (all is not None
1037 or (inspect.getmodule(value) or object) is object):
Skip Montanaroa5616d22004-06-11 04:46:12 +00001038 if visiblename(key, all):
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +00001039 classes.append((key, value))
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001040 funcs = []
1041 for key, value in inspect.getmembers(object, inspect.isroutine):
Johannes Gijsbers4c11f602004-08-30 14:13:04 +00001042 # if __all__ exists, believe it. Otherwise use old heuristic.
1043 if (all is not None or
1044 inspect.isbuiltin(value) or inspect.getmodule(value) is object):
Skip Montanaroa5616d22004-06-11 04:46:12 +00001045 if visiblename(key, all):
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +00001046 funcs.append((key, value))
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001047 data = []
1048 for key, value in inspect.getmembers(object, isdata):
Skip Montanaroa5616d22004-06-11 04:46:12 +00001049 if visiblename(key, all):
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001050 data.append((key, value))
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001051
1052 if hasattr(object, '__path__'):
1053 modpkgs = []
1054 for file in os.listdir(object.__path__[0]):
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +00001055 path = os.path.join(object.__path__[0], file)
1056 modname = inspect.getmodulename(file)
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +00001057 if modname != '__init__':
1058 if modname and modname not in modpkgs:
1059 modpkgs.append(modname)
1060 elif ispackage(path):
1061 modpkgs.append(file + ' (package)')
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001062 modpkgs.sort()
1063 result = result + self.section(
1064 'PACKAGE CONTENTS', join(modpkgs, '\n'))
1065
1066 if classes:
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001067 classlist = map(lambda (key, value): value, classes)
1068 contents = [self.formattree(
1069 inspect.getclasstree(classlist, 1), name)]
1070 for key, value in classes:
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001071 contents.append(self.document(value, key, name))
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001072 result = result + self.section('CLASSES', join(contents, '\n'))
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001073
1074 if funcs:
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001075 contents = []
1076 for key, value in funcs:
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001077 contents.append(self.document(value, key, name))
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001078 result = result + self.section('FUNCTIONS', join(contents, '\n'))
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001079
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001080 if data:
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001081 contents = []
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001082 for key, value in data:
Georg Brandl8b813db2005-10-01 16:32:31 +00001083 contents.append(self.docother(value, key, name, maxlen=70))
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001084 result = result + self.section('DATA', join(contents, '\n'))
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001085
1086 if hasattr(object, '__version__'):
1087 version = str(object.__version__)
Ka-Ping Yee1d384632001-03-01 00:24:32 +00001088 if version[:11] == '$' + 'Revision: ' and version[-1:] == '$':
1089 version = strip(version[11:-1])
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001090 result = result + self.section('VERSION', version)
Ka-Ping Yee6f3f9a42001-02-27 22:42:36 +00001091 if hasattr(object, '__date__'):
1092 result = result + self.section('DATE', str(object.__date__))
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001093 if hasattr(object, '__author__'):
Ka-Ping Yee6f3f9a42001-02-27 22:42:36 +00001094 result = result + self.section('AUTHOR', str(object.__author__))
1095 if hasattr(object, '__credits__'):
1096 result = result + self.section('CREDITS', str(object.__credits__))
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001097 return result
1098
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001099 def docclass(self, object, name=None, mod=None):
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001100 """Produce text documentation for a given class object."""
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001101 realname = object.__name__
1102 name = name or realname
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001103 bases = object.__bases__
1104
Tim Petersc86f6ca2001-09-26 21:31:51 +00001105 def makename(c, m=object.__module__):
1106 return classname(c, m)
1107
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001108 if name == realname:
1109 title = 'class ' + self.bold(realname)
1110 else:
1111 title = self.bold(name) + ' = class ' + realname
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001112 if bases:
Ka-Ping Yee3bda8792001-03-23 13:17:50 +00001113 parents = map(makename, bases)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001114 title = title + '(%s)' % join(parents, ', ')
1115
1116 doc = getdoc(object)
Tim Peters28355492001-09-23 21:29:55 +00001117 contents = doc and [doc + '\n'] or []
1118 push = contents.append
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001119
Tim Petersc86f6ca2001-09-26 21:31:51 +00001120 # List the mro, if non-trivial.
Raymond Hettinger756b3f32004-01-29 06:37:52 +00001121 mro = deque(inspect.getmro(object))
Tim Petersc86f6ca2001-09-26 21:31:51 +00001122 if len(mro) > 2:
1123 push("Method resolution order:")
1124 for base in mro:
1125 push(' ' + makename(base))
1126 push('')
1127
Tim Petersf4aad8e2001-09-24 22:40:47 +00001128 # Cute little class to pump out a horizontal rule between sections.
1129 class HorizontalRule:
1130 def __init__(self):
1131 self.needone = 0
1132 def maybe(self):
1133 if self.needone:
1134 push('-' * 70)
1135 self.needone = 1
1136 hr = HorizontalRule()
1137
Tim Peters28355492001-09-23 21:29:55 +00001138 def spill(msg, attrs, predicate):
Tim Petersfa26f7c2001-09-24 08:05:11 +00001139 ok, attrs = _split_list(attrs, predicate)
Tim Peters28355492001-09-23 21:29:55 +00001140 if ok:
Tim Petersf4aad8e2001-09-24 22:40:47 +00001141 hr.maybe()
Tim Peters28355492001-09-23 21:29:55 +00001142 push(msg)
1143 for name, kind, homecls, value in ok:
1144 push(self.document(getattr(object, name),
1145 name, mod, object))
1146 return attrs
1147
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +00001148 def spilldescriptors(msg, attrs, predicate):
Tim Petersfa26f7c2001-09-24 08:05:11 +00001149 ok, attrs = _split_list(attrs, predicate)
Tim Peters28355492001-09-23 21:29:55 +00001150 if ok:
Tim Petersf4aad8e2001-09-24 22:40:47 +00001151 hr.maybe()
Tim Peters28355492001-09-23 21:29:55 +00001152 push(msg)
1153 for name, kind, homecls, value in ok:
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +00001154 push(self._docdescriptor(name, value, mod))
Tim Peters28355492001-09-23 21:29:55 +00001155 return attrs
Tim Petersb47879b2001-09-24 04:47:19 +00001156
Tim Petersfa26f7c2001-09-24 08:05:11 +00001157 def spilldata(msg, attrs, predicate):
1158 ok, attrs = _split_list(attrs, predicate)
Tim Peters28355492001-09-23 21:29:55 +00001159 if ok:
Tim Petersf4aad8e2001-09-24 22:40:47 +00001160 hr.maybe()
Tim Peters28355492001-09-23 21:29:55 +00001161 push(msg)
1162 for name, kind, homecls, value in ok:
Martin v. Löwise59e2ba2003-05-03 09:09:02 +00001163 if callable(value) or inspect.isdatadescriptor(value):
Ka-Ping Yeebba6acc2005-02-19 22:58:26 +00001164 doc = getdoc(value)
Guido van Rossum5e355b22002-05-21 20:56:15 +00001165 else:
1166 doc = None
Tim Peters28355492001-09-23 21:29:55 +00001167 push(self.docother(getattr(object, name),
Georg Brandl8b813db2005-10-01 16:32:31 +00001168 name, mod, maxlen=70, doc=doc) + '\n')
Tim Peters28355492001-09-23 21:29:55 +00001169 return attrs
1170
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +00001171 attrs = filter(lambda (name, kind, cls, value): visiblename(name),
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +00001172 classify_class_attrs(object))
Tim Petersfa26f7c2001-09-24 08:05:11 +00001173 while attrs:
Tim Peters351e3622001-09-27 03:29:51 +00001174 if mro:
Raymond Hettinger756b3f32004-01-29 06:37:52 +00001175 thisclass = mro.popleft()
Tim Peters351e3622001-09-27 03:29:51 +00001176 else:
1177 thisclass = attrs[0][2]
Tim Petersfa26f7c2001-09-24 08:05:11 +00001178 attrs, inherited = _split_list(attrs, lambda t: t[2] is thisclass)
1179
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +00001180 if thisclass is __builtin__.object:
1181 attrs = inherited
1182 continue
1183 elif thisclass is object:
Tim Peters28355492001-09-23 21:29:55 +00001184 tag = "defined here"
1185 else:
Tim Petersfa26f7c2001-09-24 08:05:11 +00001186 tag = "inherited from %s" % classname(thisclass,
1187 object.__module__)
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +00001188 filter(lambda t: not t[0].startswith('_'), attrs)
Tim Peters28355492001-09-23 21:29:55 +00001189
1190 # Sort attrs by name.
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +00001191 attrs.sort()
Tim Peters28355492001-09-23 21:29:55 +00001192
1193 # Pump out the attrs, segregated by kind.
Tim Petersf4aad8e2001-09-24 22:40:47 +00001194 attrs = spill("Methods %s:\n" % tag, attrs,
Tim Peters28355492001-09-23 21:29:55 +00001195 lambda t: t[1] == 'method')
Tim Petersf4aad8e2001-09-24 22:40:47 +00001196 attrs = spill("Class methods %s:\n" % tag, attrs,
Tim Peters28355492001-09-23 21:29:55 +00001197 lambda t: t[1] == 'class method')
Tim Petersf4aad8e2001-09-24 22:40:47 +00001198 attrs = spill("Static methods %s:\n" % tag, attrs,
Tim Peters28355492001-09-23 21:29:55 +00001199 lambda t: t[1] == 'static method')
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +00001200 attrs = spilldescriptors("Data descriptors %s:\n" % tag, attrs,
1201 lambda t: t[1] == 'data descriptor')
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +00001202 attrs = spilldata("Data and other attributes %s:\n" % tag, attrs,
1203 lambda t: t[1] == 'data')
Tim Peters28355492001-09-23 21:29:55 +00001204 assert attrs == []
Tim Peters351e3622001-09-27 03:29:51 +00001205 attrs = inherited
Tim Peters28355492001-09-23 21:29:55 +00001206
1207 contents = '\n'.join(contents)
1208 if not contents:
1209 return title + '\n'
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001210 return title + '\n' + self.indent(rstrip(contents), ' | ') + '\n'
1211
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001212 def formatvalue(self, object):
1213 """Format an argument default value as text."""
1214 return '=' + self.repr(object)
1215
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001216 def docroutine(self, object, name=None, mod=None, cl=None):
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001217 """Produce text documentation for a function or method object."""
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001218 realname = object.__name__
1219 name = name or realname
1220 note = ''
Ka-Ping Yee3bda8792001-03-23 13:17:50 +00001221 skipdocs = 0
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001222 if inspect.ismethod(object):
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001223 imclass = object.im_class
Ka-Ping Yee3bda8792001-03-23 13:17:50 +00001224 if cl:
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001225 if imclass is not cl:
1226 note = ' from ' + classname(imclass, mod)
Ka-Ping Yee3bda8792001-03-23 13:17:50 +00001227 else:
Ka-Ping Yeeb7a48302001-04-12 20:39:14 +00001228 if object.im_self:
1229 note = ' method of %s instance' % classname(
1230 object.im_self.__class__, mod)
1231 else:
1232 note = ' unbound %s method' % classname(imclass,mod)
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001233 object = object.im_func
1234
1235 if name == realname:
1236 title = self.bold(realname)
1237 else:
Raymond Hettinger54f02222002-06-01 14:18:47 +00001238 if (cl and realname in cl.__dict__ and
Ka-Ping Yee3bda8792001-03-23 13:17:50 +00001239 cl.__dict__[realname] is object):
1240 skipdocs = 1
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001241 title = self.bold(name) + ' = ' + realname
Tim Peters4bcfa312001-09-20 06:08:24 +00001242 if inspect.isfunction(object):
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001243 args, varargs, varkw, defaults = inspect.getargspec(object)
1244 argspec = inspect.formatargspec(
1245 args, varargs, varkw, defaults, formatvalue=self.formatvalue)
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001246 if realname == '<lambda>':
1247 title = 'lambda'
1248 argspec = argspec[1:-1] # remove parentheses
Tim Peters4bcfa312001-09-20 06:08:24 +00001249 else:
1250 argspec = '(...)'
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001251 decl = title + argspec + note
1252
Ka-Ping Yee3bda8792001-03-23 13:17:50 +00001253 if skipdocs:
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001254 return decl + '\n'
Ka-Ping Yee3bda8792001-03-23 13:17:50 +00001255 else:
1256 doc = getdoc(object) or ''
1257 return decl + '\n' + (doc and rstrip(self.indent(doc)) + '\n')
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001258
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +00001259 def _docdescriptor(self, name, value, mod):
Johannes Gijsbers8de645a2004-11-07 19:16:05 +00001260 results = []
1261 push = results.append
1262
1263 if name:
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +00001264 push(self.bold(name))
1265 push('\n')
Johannes Gijsbers8de645a2004-11-07 19:16:05 +00001266 doc = getdoc(value) or ''
1267 if doc:
1268 push(self.indent(doc))
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +00001269 push('\n')
1270 return ''.join(results)
Johannes Gijsbers8de645a2004-11-07 19:16:05 +00001271
1272 def docproperty(self, object, name=None, mod=None, cl=None):
1273 """Produce text documentation for a property."""
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +00001274 return self._docdescriptor(name, object, mod)
Johannes Gijsbers8de645a2004-11-07 19:16:05 +00001275
Georg Brandl8b813db2005-10-01 16:32:31 +00001276 def docother(self, object, name=None, mod=None, parent=None, maxlen=None, doc=None):
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001277 """Produce text documentation for a data object."""
1278 repr = self.repr(object)
1279 if maxlen:
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001280 line = (name and name + ' = ' or '') + repr
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001281 chop = maxlen - len(line)
1282 if chop < 0: repr = repr[:chop] + '...'
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001283 line = (name and self.bold(name) + ' = ' or '') + repr
Tim Peters28355492001-09-23 21:29:55 +00001284 if doc is not None:
1285 line += '\n' + self.indent(str(doc))
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001286 return line
1287
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001288# --------------------------------------------------------- user interfaces
1289
1290def pager(text):
1291 """The first time this is called, determine what kind of pager to use."""
1292 global pager
1293 pager = getpager()
1294 pager(text)
1295
1296def getpager():
1297 """Decide what method to use for paging through text."""
1298 if type(sys.stdout) is not types.FileType:
1299 return plainpager
1300 if not sys.stdin.isatty() or not sys.stdout.isatty():
1301 return plainpager
Raymond Hettinger54f02222002-06-01 14:18:47 +00001302 if 'PAGER' in os.environ:
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +00001303 if sys.platform == 'win32': # pipes completely broken in Windows
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001304 return lambda text: tempfilepager(plain(text), os.environ['PAGER'])
Raymond Hettingerdbecd932005-02-06 06:57:08 +00001305 elif os.environ.get('TERM') in ('dumb', 'emacs'):
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001306 return lambda text: pipepager(plain(text), os.environ['PAGER'])
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +00001307 else:
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001308 return lambda text: pipepager(text, os.environ['PAGER'])
Ka-Ping Yeea487e4e2005-11-05 04:49:18 +00001309 if os.environ.get('TERM') in ('dumb', 'emacs'):
1310 return plainpager
Andrew MacIntyre54e0eab2002-03-03 03:12:30 +00001311 if sys.platform == 'win32' or sys.platform.startswith('os2'):
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001312 return lambda text: tempfilepager(plain(text), 'more <')
Skip Montanarod404bee2002-09-26 21:44:57 +00001313 if hasattr(os, 'system') and os.system('(less) 2>/dev/null') == 0:
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001314 return lambda text: pipepager(text, 'less')
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001315
1316 import tempfile
Guido van Rossum3b0a3292002-08-09 16:38:32 +00001317 (fd, filename) = tempfile.mkstemp()
1318 os.close(fd)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001319 try:
1320 if hasattr(os, 'system') and os.system('more %s' % filename) == 0:
1321 return lambda text: pipepager(text, 'more')
1322 else:
1323 return ttypager
1324 finally:
1325 os.unlink(filename)
1326
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001327def plain(text):
1328 """Remove boldface formatting from text."""
1329 return re.sub('.\b', '', text)
1330
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001331def pipepager(text, cmd):
1332 """Page through text by feeding it to another program."""
1333 pipe = os.popen(cmd, 'w')
1334 try:
1335 pipe.write(text)
1336 pipe.close()
1337 except IOError:
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001338 pass # Ignore broken pipes caused by quitting the pager program.
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001339
1340def tempfilepager(text, cmd):
1341 """Page through text by invoking a program on a temporary file."""
1342 import tempfile
Tim Peters550e4e52003-02-07 01:53:46 +00001343 filename = tempfile.mktemp()
1344 file = open(filename, 'w')
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001345 file.write(text)
1346 file.close()
1347 try:
Ka-Ping Yeec92cdf72001-03-02 05:54:35 +00001348 os.system(cmd + ' ' + filename)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001349 finally:
1350 os.unlink(filename)
1351
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001352def ttypager(text):
1353 """Page through text on a text terminal."""
1354 lines = split(plain(text), '\n')
1355 try:
1356 import tty
1357 fd = sys.stdin.fileno()
1358 old = tty.tcgetattr(fd)
1359 tty.setcbreak(fd)
1360 getchar = lambda: sys.stdin.read(1)
Ka-Ping Yee457aab22001-02-27 23:36:29 +00001361 except (ImportError, AttributeError):
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001362 tty = None
1363 getchar = lambda: sys.stdin.readline()[:-1][:1]
1364
1365 try:
1366 r = inc = os.environ.get('LINES', 25) - 1
1367 sys.stdout.write(join(lines[:inc], '\n') + '\n')
1368 while lines[r:]:
1369 sys.stdout.write('-- more --')
1370 sys.stdout.flush()
1371 c = getchar()
1372
Raymond Hettingerdbecd932005-02-06 06:57:08 +00001373 if c in ('q', 'Q'):
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001374 sys.stdout.write('\r \r')
1375 break
Raymond Hettingerdbecd932005-02-06 06:57:08 +00001376 elif c in ('\r', '\n'):
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001377 sys.stdout.write('\r \r' + lines[r] + '\n')
1378 r = r + 1
1379 continue
Raymond Hettingerdbecd932005-02-06 06:57:08 +00001380 if c in ('b', 'B', '\x1b'):
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001381 r = r - inc - inc
1382 if r < 0: r = 0
1383 sys.stdout.write('\n' + join(lines[r:r+inc], '\n') + '\n')
1384 r = r + inc
1385
1386 finally:
1387 if tty:
1388 tty.tcsetattr(fd, tty.TCSAFLUSH, old)
1389
1390def plainpager(text):
1391 """Simply print unformatted text. This is the ultimate fallback."""
1392 sys.stdout.write(plain(text))
1393
1394def describe(thing):
Ka-Ping Yee3bda8792001-03-23 13:17:50 +00001395 """Produce a short description of the given thing."""
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001396 if inspect.ismodule(thing):
1397 if thing.__name__ in sys.builtin_module_names:
1398 return 'built-in module ' + thing.__name__
1399 if hasattr(thing, '__path__'):
1400 return 'package ' + thing.__name__
1401 else:
1402 return 'module ' + thing.__name__
1403 if inspect.isbuiltin(thing):
1404 return 'built-in function ' + thing.__name__
1405 if inspect.isclass(thing):
1406 return 'class ' + thing.__name__
1407 if inspect.isfunction(thing):
1408 return 'function ' + thing.__name__
1409 if inspect.ismethod(thing):
1410 return 'method ' + thing.__name__
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001411 if type(thing) is types.InstanceType:
1412 return 'instance of ' + thing.__class__.__name__
1413 return type(thing).__name__
1414
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001415def locate(path, forceload=0):
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001416 """Locate an object by name or dotted path, importing as necessary."""
Guido van Rossum97dede02003-02-16 01:12:32 +00001417 parts = [part for part in split(path, '.') if part]
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001418 module, n = None, 0
1419 while n < len(parts):
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001420 nextmodule = safeimport(join(parts[:n+1], '.'), forceload)
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001421 if nextmodule: module, n = nextmodule, n + 1
1422 else: break
1423 if module:
1424 object = module
1425 for part in parts[n:]:
1426 try: object = getattr(object, part)
1427 except AttributeError: return None
1428 return object
1429 else:
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001430 if hasattr(__builtin__, path):
1431 return getattr(__builtin__, path)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001432
1433# --------------------------------------- interactive interpreter interface
1434
1435text = TextDoc()
1436html = HTMLDoc()
1437
Ka-Ping Yee45daeb02002-08-11 15:11:33 +00001438def resolve(thing, forceload=0):
1439 """Given an object or a path to an object, get the object and its name."""
1440 if isinstance(thing, str):
1441 object = locate(thing, forceload)
1442 if not object:
1443 raise ImportError, 'no Python documentation found for %r' % thing
1444 return object, thing
1445 else:
1446 return thing, getattr(thing, '__name__', None)
1447
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001448def doc(thing, title='Python Library Documentation: %s', forceload=0):
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001449 """Display text documentation, given an object or a path to an object."""
Ka-Ping Yee45daeb02002-08-11 15:11:33 +00001450 try:
1451 object, name = resolve(thing, forceload)
1452 desc = describe(object)
1453 module = inspect.getmodule(object)
1454 if name and '.' in name:
1455 desc += ' in ' + name[:name.rfind('.')]
1456 elif module and module is not object:
1457 desc += ' in module ' + module.__name__
Ka-Ping Yeebba6acc2005-02-19 22:58:26 +00001458 if not (inspect.ismodule(object) or
1459 inspect.isclass(object) or
1460 inspect.isroutine(object) or
1461 isinstance(object, property)):
1462 # If the passed object is a piece of data or an instance,
1463 # document its available methods instead of its value.
1464 object = type(object)
1465 desc += ' object'
Ka-Ping Yee45daeb02002-08-11 15:11:33 +00001466 pager(title % desc + '\n\n' + text.document(object, name))
1467 except (ImportError, ErrorDuringImport), value:
1468 print value
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001469
Ka-Ping Yee45daeb02002-08-11 15:11:33 +00001470def writedoc(thing, forceload=0):
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001471 """Write HTML documentation to a file in the current directory."""
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001472 try:
Ka-Ping Yee45daeb02002-08-11 15:11:33 +00001473 object, name = resolve(thing, forceload)
1474 page = html.page(describe(object), html.document(object, name))
1475 file = open(name + '.html', 'w')
1476 file.write(page)
1477 file.close()
1478 print 'wrote', name + '.html'
1479 except (ImportError, ErrorDuringImport), value:
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001480 print value
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001481
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001482def writedocs(dir, pkgpath='', done=None):
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001483 """Write out HTML documentation for all modules in a directory tree."""
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001484 if done is None: done = {}
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001485 for file in os.listdir(dir):
1486 path = os.path.join(dir, file)
1487 if ispackage(path):
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001488 writedocs(path, pkgpath + file + '.', done)
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001489 elif os.path.isfile(path):
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +00001490 modname = inspect.getmodulename(path)
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001491 if modname:
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +00001492 if modname == '__init__':
1493 modname = pkgpath[:-1] # remove trailing period
1494 else:
1495 modname = pkgpath + modname
1496 if modname not in done:
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001497 done[modname] = 1
1498 writedoc(modname)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001499
1500class Helper:
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001501 keywords = {
1502 'and': 'BOOLEAN',
Ka-Ping Yee79c009d2001-04-13 10:53:25 +00001503 'assert': ('ref/assert', ''),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001504 'break': ('ref/break', 'while for'),
1505 'class': ('ref/class', 'CLASSES SPECIALMETHODS'),
1506 'continue': ('ref/continue', 'while for'),
1507 'def': ('ref/function', ''),
1508 'del': ('ref/del', 'BASICMETHODS'),
1509 'elif': 'if',
1510 'else': ('ref/if', 'while for'),
1511 'except': 'try',
1512 'exec': ('ref/exec', ''),
1513 'finally': 'try',
1514 'for': ('ref/for', 'break continue while'),
1515 'from': 'import',
1516 'global': ('ref/global', 'NAMESPACES'),
1517 'if': ('ref/if', 'TRUTHVALUE'),
1518 'import': ('ref/import', 'MODULES'),
1519 'in': ('ref/comparisons', 'SEQUENCEMETHODS2'),
1520 'is': 'COMPARISON',
Neal Norwitz742dde42003-03-30 20:31:34 +00001521 'lambda': ('ref/lambdas', 'FUNCTIONS'),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001522 'not': 'BOOLEAN',
1523 'or': 'BOOLEAN',
Neal Norwitz742dde42003-03-30 20:31:34 +00001524 'pass': ('ref/pass', ''),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001525 'print': ('ref/print', ''),
1526 'raise': ('ref/raise', 'EXCEPTIONS'),
Ka-Ping Yee79c009d2001-04-13 10:53:25 +00001527 'return': ('ref/return', 'FUNCTIONS'),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001528 'try': ('ref/try', 'EXCEPTIONS'),
1529 'while': ('ref/while', 'break continue if TRUTHVALUE'),
Tim Petersfb05c4e2002-10-30 05:21:00 +00001530 'yield': ('ref/yield', ''),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001531 }
1532
1533 topics = {
1534 'TYPES': ('ref/types', 'STRINGS UNICODE NUMBERS SEQUENCES MAPPINGS FUNCTIONS CLASSES MODULES FILES inspect'),
Ka-Ping Yee79c009d2001-04-13 10:53:25 +00001535 'STRINGS': ('ref/strings', 'str UNICODE SEQUENCES STRINGMETHODS FORMATTING TYPES'),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001536 'STRINGMETHODS': ('lib/string-methods', 'STRINGS FORMATTING'),
1537 'FORMATTING': ('lib/typesseq-strings', 'OPERATORS'),
Neal Norwitz54f871e2003-05-26 13:49:54 +00001538 'UNICODE': ('ref/strings', 'encodings unicode SEQUENCES STRINGMETHODS FORMATTING TYPES'),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001539 'NUMBERS': ('ref/numbers', 'INTEGER FLOAT COMPLEX TYPES'),
1540 'INTEGER': ('ref/integers', 'int range'),
1541 'FLOAT': ('ref/floating', 'float math'),
1542 'COMPLEX': ('ref/imaginary', 'complex cmath'),
Ka-Ping Yee79c009d2001-04-13 10:53:25 +00001543 'SEQUENCES': ('lib/typesseq', 'STRINGMETHODS FORMATTING xrange LISTS'),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001544 'MAPPINGS': 'DICTIONARIES',
1545 'FUNCTIONS': ('lib/typesfunctions', 'def TYPES'),
1546 'METHODS': ('lib/typesmethods', 'class def CLASSES TYPES'),
1547 'CODEOBJECTS': ('lib/bltin-code-objects', 'compile FUNCTIONS TYPES'),
Ka-Ping Yee79c009d2001-04-13 10:53:25 +00001548 'TYPEOBJECTS': ('lib/bltin-type-objects', 'types TYPES'),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001549 'FRAMEOBJECTS': 'TYPES',
1550 'TRACEBACKS': 'TYPES',
1551 'NONE': ('lib/bltin-null-object', ''),
1552 'ELLIPSIS': ('lib/bltin-ellipsis-object', 'SLICINGS'),
1553 'FILES': ('lib/bltin-file-objects', ''),
1554 'SPECIALATTRIBUTES': ('lib/specialattrs', ''),
1555 'CLASSES': ('ref/types', 'class SPECIALMETHODS PRIVATENAMES'),
1556 'MODULES': ('lib/typesmodules', 'import'),
1557 'PACKAGES': 'import',
1558 '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'),
1559 'OPERATORS': 'EXPRESSIONS',
1560 'PRECEDENCE': 'EXPRESSIONS',
1561 'OBJECTS': ('ref/objects', 'TYPES'),
1562 'SPECIALMETHODS': ('ref/specialnames', 'BASICMETHODS ATTRIBUTEMETHODS CALLABLEMETHODS SEQUENCEMETHODS1 MAPPINGMETHODS SEQUENCEMETHODS2 NUMBERMETHODS CLASSES'),
Ka-Ping Yee79c009d2001-04-13 10:53:25 +00001563 'BASICMETHODS': ('ref/customization', 'cmp hash repr str SPECIALMETHODS'),
1564 'ATTRIBUTEMETHODS': ('ref/attribute-access', 'ATTRIBUTES SPECIALMETHODS'),
1565 'CALLABLEMETHODS': ('ref/callable-types', 'CALLS SPECIALMETHODS'),
1566 'SEQUENCEMETHODS1': ('ref/sequence-types', 'SEQUENCES SEQUENCEMETHODS2 SPECIALMETHODS'),
1567 'SEQUENCEMETHODS2': ('ref/sequence-methods', 'SEQUENCES SEQUENCEMETHODS1 SPECIALMETHODS'),
1568 'MAPPINGMETHODS': ('ref/sequence-types', 'MAPPINGS SPECIALMETHODS'),
1569 'NUMBERMETHODS': ('ref/numeric-types', 'NUMBERS AUGMENTEDASSIGNMENT SPECIALMETHODS'),
Neal Norwitz54f871e2003-05-26 13:49:54 +00001570 'EXECUTION': ('ref/execmodel', 'NAMESPACES DYNAMICFEATURES EXCEPTIONS'),
1571 'NAMESPACES': ('ref/naming', 'global ASSIGNMENT DELETION DYNAMICFEATURES'),
1572 'DYNAMICFEATURES': ('ref/dynamic-features', ''),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001573 'SCOPING': 'NAMESPACES',
1574 'FRAMES': 'NAMESPACES',
1575 'EXCEPTIONS': ('ref/exceptions', 'try except finally raise'),
Neal Norwitz54f871e2003-05-26 13:49:54 +00001576 'COERCIONS': ('ref/coercion-rules','CONVERSIONS'),
1577 'CONVERSIONS': ('ref/conversions', 'COERCIONS'),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001578 'IDENTIFIERS': ('ref/identifiers', 'keywords SPECIALIDENTIFIERS'),
1579 'SPECIALIDENTIFIERS': ('ref/id-classes', ''),
Ka-Ping Yee79c009d2001-04-13 10:53:25 +00001580 'PRIVATENAMES': ('ref/atom-identifiers', ''),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001581 'LITERALS': ('ref/atom-literals', 'STRINGS BACKQUOTES NUMBERS TUPLELITERALS LISTLITERALS DICTIONARYLITERALS'),
1582 'TUPLES': 'SEQUENCES',
Ka-Ping Yee79c009d2001-04-13 10:53:25 +00001583 'TUPLELITERALS': ('ref/exprlists', 'TUPLES LITERALS'),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001584 'LISTS': ('lib/typesseq-mutable', 'LISTLITERALS'),
Ka-Ping Yee79c009d2001-04-13 10:53:25 +00001585 'LISTLITERALS': ('ref/lists', 'LISTS LITERALS'),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001586 'DICTIONARIES': ('lib/typesmapping', 'DICTIONARYLITERALS'),
Ka-Ping Yee79c009d2001-04-13 10:53:25 +00001587 'DICTIONARYLITERALS': ('ref/dict', 'DICTIONARIES LITERALS'),
1588 'BACKQUOTES': ('ref/string-conversions', 'repr str STRINGS LITERALS'),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001589 'ATTRIBUTES': ('ref/attribute-references', 'getattr hasattr setattr ATTRIBUTEMETHODS'),
1590 'SUBSCRIPTS': ('ref/subscriptions', 'SEQUENCEMETHODS1'),
1591 'SLICINGS': ('ref/slicings', 'SEQUENCEMETHODS2'),
1592 'CALLS': ('ref/calls', 'EXPRESSIONS'),
1593 'POWER': ('ref/power', 'EXPRESSIONS'),
1594 'UNARY': ('ref/unary', 'EXPRESSIONS'),
1595 'BINARY': ('ref/binary', 'EXPRESSIONS'),
1596 'SHIFTING': ('ref/shifting', 'EXPRESSIONS'),
1597 'BITWISE': ('ref/bitwise', 'EXPRESSIONS'),
1598 'COMPARISON': ('ref/comparisons', 'EXPRESSIONS BASICMETHODS'),
Neal Norwitzf98159c2003-02-07 20:49:40 +00001599 'BOOLEAN': ('ref/Booleans', 'EXPRESSIONS TRUTHVALUE'),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001600 'ASSERTION': 'assert',
1601 'ASSIGNMENT': ('ref/assignment', 'AUGMENTEDASSIGNMENT'),
Ka-Ping Yee79c009d2001-04-13 10:53:25 +00001602 'AUGMENTEDASSIGNMENT': ('ref/augassign', 'NUMBERMETHODS'),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001603 'DELETION': 'del',
1604 'PRINTING': 'print',
1605 'RETURNING': 'return',
1606 'IMPORTING': 'import',
1607 'CONDITIONAL': 'if',
1608 'LOOPING': ('ref/compound', 'for while break continue'),
1609 'TRUTHVALUE': ('lib/truth', 'if while and or not BASICMETHODS'),
Ka-Ping Yee79c009d2001-04-13 10:53:25 +00001610 'DEBUGGING': ('lib/module-pdb', 'pdb'),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001611 }
1612
1613 def __init__(self, input, output):
1614 self.input = input
1615 self.output = output
1616 self.docdir = None
1617 execdir = os.path.dirname(sys.executable)
1618 homedir = os.environ.get('PYTHONHOME')
1619 for dir in [os.environ.get('PYTHONDOCS'),
1620 homedir and os.path.join(homedir, 'doc'),
1621 os.path.join(execdir, 'doc'),
1622 '/usr/doc/python-docs-' + split(sys.version)[0],
1623 '/usr/doc/python-' + split(sys.version)[0],
1624 '/usr/doc/python-docs-' + sys.version[:3],
Jack Jansenb2628b02002-08-23 08:40:42 +00001625 '/usr/doc/python-' + sys.version[:3],
1626 os.path.join(sys.prefix, 'Resources/English.lproj/Documentation')]:
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001627 if dir and os.path.isdir(os.path.join(dir, 'lib')):
1628 self.docdir = dir
1629
Ka-Ping Yee79c009d2001-04-13 10:53:25 +00001630 def __repr__(self):
Ka-Ping Yee9bc576b2001-04-13 13:57:31 +00001631 if inspect.stack()[1][3] == '?':
Ka-Ping Yee79c009d2001-04-13 10:53:25 +00001632 self()
1633 return ''
Ka-Ping Yee9bc576b2001-04-13 13:57:31 +00001634 return '<pydoc.Helper instance>'
Ka-Ping Yee79c009d2001-04-13 10:53:25 +00001635
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001636 def __call__(self, request=None):
1637 if request is not None:
1638 self.help(request)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001639 else:
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001640 self.intro()
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001641 self.interact()
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001642 self.output.write('''
Fred Drakee61967f2001-05-10 18:41:02 +00001643You are now leaving help and returning to the Python interpreter.
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001644If you want to ask for help on a particular object directly from the
1645interpreter, you can type "help(object)". Executing "help('string')"
1646has the same effect as typing a particular string at the help> prompt.
1647''')
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001648
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001649 def interact(self):
1650 self.output.write('\n')
Guido van Rossum8ca162f2002-04-07 06:36:23 +00001651 while True:
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001652 try:
Johannes Gijsberse7691d32004-08-17 13:21:53 +00001653 request = self.getline('help> ')
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001654 if not request: break
Johannes Gijsberse7691d32004-08-17 13:21:53 +00001655 except (KeyboardInterrupt, EOFError):
1656 break
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001657 request = strip(replace(request, '"', '', "'", ''))
Raymond Hettingerdbecd932005-02-06 06:57:08 +00001658 if lower(request) in ('q', 'quit'): break
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001659 self.help(request)
1660
Johannes Gijsberse7691d32004-08-17 13:21:53 +00001661 def getline(self, prompt):
1662 """Read one line, using raw_input when available."""
1663 if self.input is sys.stdin:
1664 return raw_input(prompt)
1665 else:
1666 self.output.write(prompt)
1667 self.output.flush()
1668 return self.input.readline()
1669
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001670 def help(self, request):
1671 if type(request) is type(''):
1672 if request == 'help': self.intro()
1673 elif request == 'keywords': self.listkeywords()
1674 elif request == 'topics': self.listtopics()
1675 elif request == 'modules': self.listmodules()
1676 elif request[:8] == 'modules ':
1677 self.listmodules(split(request)[1])
Raymond Hettinger54f02222002-06-01 14:18:47 +00001678 elif request in self.keywords: self.showtopic(request)
1679 elif request in self.topics: self.showtopic(request)
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001680 elif request: doc(request, 'Help on %s:')
1681 elif isinstance(request, Helper): self()
1682 else: doc(request, 'Help on %s:')
1683 self.output.write('\n')
1684
1685 def intro(self):
1686 self.output.write('''
1687Welcome to Python %s! This is the online help utility.
1688
1689If this is your first time using Python, you should definitely check out
1690the tutorial on the Internet at http://www.python.org/doc/tut/.
1691
1692Enter the name of any module, keyword, or topic to get help on writing
1693Python programs and using Python modules. To quit this help utility and
1694return to the interpreter, just type "quit".
1695
1696To get a list of available modules, keywords, or topics, type "modules",
1697"keywords", or "topics". Each module also comes with a one-line summary
1698of what it does; to list the modules whose summaries contain a given word
1699such as "spam", type "modules spam".
1700''' % sys.version[:3])
1701
1702 def list(self, items, columns=4, width=80):
1703 items = items[:]
1704 items.sort()
1705 colw = width / columns
1706 rows = (len(items) + columns - 1) / columns
1707 for row in range(rows):
1708 for col in range(columns):
1709 i = col * rows + row
1710 if i < len(items):
1711 self.output.write(items[i])
1712 if col < columns - 1:
1713 self.output.write(' ' + ' ' * (colw-1 - len(items[i])))
1714 self.output.write('\n')
1715
1716 def listkeywords(self):
1717 self.output.write('''
1718Here is a list of the Python keywords. Enter any keyword to get more help.
1719
1720''')
1721 self.list(self.keywords.keys())
1722
1723 def listtopics(self):
1724 self.output.write('''
1725Here is a list of available topics. Enter any topic name to get more help.
1726
1727''')
1728 self.list(self.topics.keys())
1729
1730 def showtopic(self, topic):
1731 if not self.docdir:
1732 self.output.write('''
1733Sorry, topic and keyword documentation is not available because the Python
1734HTML documentation files could not be found. If you have installed them,
1735please set the environment variable PYTHONDOCS to indicate their location.
1736''')
1737 return
1738 target = self.topics.get(topic, self.keywords.get(topic))
1739 if not target:
1740 self.output.write('no documentation found for %s\n' % repr(topic))
1741 return
1742 if type(target) is type(''):
1743 return self.showtopic(target)
1744
1745 filename, xrefs = target
1746 filename = self.docdir + '/' + filename + '.html'
1747 try:
1748 file = open(filename)
1749 except:
1750 self.output.write('could not read docs from %s\n' % filename)
1751 return
1752
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001753 divpat = re.compile('<div[^>]*navigat.*?</div.*?>', re.I | re.S)
1754 addrpat = re.compile('<address.*?>.*?</address.*?>', re.I | re.S)
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001755 document = re.sub(addrpat, '', re.sub(divpat, '', file.read()))
1756 file.close()
1757
1758 import htmllib, formatter, StringIO
1759 buffer = StringIO.StringIO()
1760 parser = htmllib.HTMLParser(
1761 formatter.AbstractFormatter(formatter.DumbWriter(buffer)))
1762 parser.start_table = parser.do_p
1763 parser.end_table = lambda parser=parser: parser.do_p({})
1764 parser.start_tr = parser.do_br
1765 parser.start_td = parser.start_th = lambda a, b=buffer: b.write('\t')
1766 parser.feed(document)
1767 buffer = replace(buffer.getvalue(), '\xa0', ' ', '\n', '\n ')
1768 pager(' ' + strip(buffer) + '\n')
Ka-Ping Yeeda793892001-04-13 11:02:51 +00001769 if xrefs:
1770 buffer = StringIO.StringIO()
1771 formatter.DumbWriter(buffer).send_flowing_data(
1772 'Related help topics: ' + join(split(xrefs), ', ') + '\n')
1773 self.output.write('\n%s\n' % buffer.getvalue())
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001774
1775 def listmodules(self, key=''):
1776 if key:
1777 self.output.write('''
1778Here is a list of matching modules. Enter any module name to get more help.
1779
1780''')
1781 apropos(key)
1782 else:
1783 self.output.write('''
1784Please wait a moment while I gather a list of all available modules...
1785
1786''')
1787 modules = {}
1788 def callback(path, modname, desc, modules=modules):
1789 if modname and modname[-9:] == '.__init__':
1790 modname = modname[:-9] + ' (package)'
1791 if find(modname, '.') < 0:
1792 modules[modname] = 1
1793 ModuleScanner().run(callback)
1794 self.list(modules.keys())
1795 self.output.write('''
1796Enter any module name to get more help. Or, type "modules spam" to search
1797for modules whose descriptions contain the word "spam".
1798''')
1799
1800help = Helper(sys.stdin, sys.stdout)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001801
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001802class Scanner:
1803 """A generic tree iterator."""
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001804 def __init__(self, roots, children, descendp):
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001805 self.roots = roots[:]
1806 self.state = []
1807 self.children = children
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001808 self.descendp = descendp
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001809
1810 def next(self):
1811 if not self.state:
1812 if not self.roots:
1813 return None
1814 root = self.roots.pop(0)
1815 self.state = [(root, self.children(root))]
1816 node, children = self.state[-1]
1817 if not children:
1818 self.state.pop()
1819 return self.next()
1820 child = children.pop(0)
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001821 if self.descendp(child):
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001822 self.state.append((child, self.children(child)))
1823 return child
1824
1825class ModuleScanner(Scanner):
1826 """An interruptible scanner that searches module synopses."""
1827 def __init__(self):
1828 roots = map(lambda dir: (dir, ''), pathdirs())
Ka-Ping Yeeeca15c12001-04-13 13:53:07 +00001829 Scanner.__init__(self, roots, self.submodules, self.isnewpackage)
Raymond Hettinger32200ae2002-06-01 19:51:15 +00001830 self.inodes = map(lambda (dir, pkg): os.stat(dir).st_ino, roots)
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001831
1832 def submodules(self, (dir, package)):
1833 children = []
1834 for file in os.listdir(dir):
1835 path = os.path.join(dir, file)
Tim Peters30edd232001-03-16 08:29:48 +00001836 if ispackage(path):
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001837 children.append((path, package + (package and '.') + file))
1838 else:
1839 children.append((path, package))
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001840 children.sort() # so that spam.py comes before spam.pyc or spam.pyo
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001841 return children
1842
Ka-Ping Yeeeca15c12001-04-13 13:53:07 +00001843 def isnewpackage(self, (dir, package)):
Raymond Hettinger32200ae2002-06-01 19:51:15 +00001844 inode = os.path.exists(dir) and os.stat(dir).st_ino
Ka-Ping Yeeeca15c12001-04-13 13:53:07 +00001845 if not (os.path.islink(dir) and inode in self.inodes):
Ka-Ping Yee6191a232001-04-13 15:00:27 +00001846 self.inodes.append(inode) # detect circular symbolic links
Ka-Ping Yeeeca15c12001-04-13 13:53:07 +00001847 return ispackage(dir)
Guido van Rossum8ca162f2002-04-07 06:36:23 +00001848 return False
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001849
Ka-Ping Yee66246962001-04-12 11:59:50 +00001850 def run(self, callback, key=None, completer=None):
1851 if key: key = lower(key)
Guido van Rossum8ca162f2002-04-07 06:36:23 +00001852 self.quit = False
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001853 seen = {}
1854
1855 for modname in sys.builtin_module_names:
Ka-Ping Yee239432a2001-03-02 02:45:08 +00001856 if modname != '__main__':
1857 seen[modname] = 1
Ka-Ping Yee66246962001-04-12 11:59:50 +00001858 if key is None:
1859 callback(None, modname, '')
1860 else:
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001861 desc = split(__import__(modname).__doc__ or '', '\n')[0]
Ka-Ping Yee66246962001-04-12 11:59:50 +00001862 if find(lower(modname + ' - ' + desc), key) >= 0:
1863 callback(None, modname, desc)
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001864
1865 while not self.quit:
1866 node = self.next()
1867 if not node: break
1868 path, package = node
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +00001869 modname = inspect.getmodulename(path)
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001870 if os.path.isfile(path) and modname:
1871 modname = package + (package and '.') + modname
Raymond Hettinger54f02222002-06-01 14:18:47 +00001872 if not modname in seen:
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001873 seen[modname] = 1 # if we see spam.py, skip spam.pyc
Ka-Ping Yee66246962001-04-12 11:59:50 +00001874 if key is None:
1875 callback(path, modname, '')
1876 else:
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001877 desc = synopsis(path) or ''
1878 if find(lower(modname + ' - ' + desc), key) >= 0:
1879 callback(path, modname, desc)
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001880 if completer: completer()
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001881
1882def apropos(key):
1883 """Print all the one-line module summaries that contain a substring."""
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001884 def callback(path, modname, desc):
1885 if modname[-9:] == '.__init__':
1886 modname = modname[:-9] + ' (package)'
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +00001887 print modname, desc and '- ' + desc
1888 try: import warnings
1889 except ImportError: pass
1890 else: warnings.filterwarnings('ignore') # ignore problems during import
Ka-Ping Yee66246962001-04-12 11:59:50 +00001891 ModuleScanner().run(callback, key)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001892
1893# --------------------------------------------------- web browser interface
1894
Ka-Ping Yee66246962001-04-12 11:59:50 +00001895def serve(port, callback=None, completer=None):
Ka-Ping Yeefd540692001-04-12 12:54:36 +00001896 import BaseHTTPServer, mimetools, select
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001897
1898 # Patch up mimetools.Message so it doesn't break if rfc822 is reloaded.
1899 class Message(mimetools.Message):
1900 def __init__(self, fp, seekable=1):
1901 Message = self.__class__
1902 Message.__bases__[0].__bases__[0].__init__(self, fp, seekable)
1903 self.encodingheader = self.getheader('content-transfer-encoding')
1904 self.typeheader = self.getheader('content-type')
1905 self.parsetype()
1906 self.parseplist()
1907
1908 class DocHandler(BaseHTTPServer.BaseHTTPRequestHandler):
1909 def send_document(self, title, contents):
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001910 try:
1911 self.send_response(200)
1912 self.send_header('Content-Type', 'text/html')
1913 self.end_headers()
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001914 self.wfile.write(html.page(title, contents))
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001915 except IOError: pass
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001916
1917 def do_GET(self):
1918 path = self.path
1919 if path[-5:] == '.html': path = path[:-5]
1920 if path[:1] == '/': path = path[1:]
1921 if path and path != '.':
1922 try:
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001923 obj = locate(path, forceload=1)
Ka-Ping Yee3bda8792001-03-23 13:17:50 +00001924 except ErrorDuringImport, value:
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001925 self.send_document(path, html.escape(str(value)))
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001926 return
Ka-Ping Yee3bda8792001-03-23 13:17:50 +00001927 if obj:
1928 self.send_document(describe(obj), html.document(obj, path))
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001929 else:
1930 self.send_document(path,
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001931'no Python documentation found for %s' % repr(path))
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001932 else:
1933 heading = html.heading(
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001934'<big><big><strong>Python: Index of Modules</strong></big></big>',
1935'#ffffff', '#7799ee')
Ka-Ping Yee239432a2001-03-02 02:45:08 +00001936 def bltinlink(name):
1937 return '<a href="%s.html">%s</a>' % (name, name)
Ka-Ping Yee3bda8792001-03-23 13:17:50 +00001938 names = filter(lambda x: x != '__main__',
1939 sys.builtin_module_names)
Ka-Ping Yee239432a2001-03-02 02:45:08 +00001940 contents = html.multicolumn(names, bltinlink)
1941 indices = ['<p>' + html.bigsection(
1942 'Built-in Modules', '#ffffff', '#ee77aa', contents)]
1943
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001944 seen = {}
1945 for dir in pathdirs():
1946 indices.append(html.index(dir, seen))
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001947 contents = heading + join(indices) + '''<p align=right>
Tim Peters2306d242001-09-25 03:18:32 +00001948<font color="#909090" face="helvetica, arial"><strong>
1949pydoc</strong> by Ka-Ping Yee &lt;ping@lfw.org&gt;</font>'''
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001950 self.send_document('Index of Modules', contents)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001951
1952 def log_message(self, *args): pass
1953
Ka-Ping Yeefd540692001-04-12 12:54:36 +00001954 class DocServer(BaseHTTPServer.HTTPServer):
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001955 def __init__(self, port, callback):
Ka-Ping Yeedb8ed152001-03-02 05:58:17 +00001956 host = (sys.platform == 'mac') and '127.0.0.1' or 'localhost'
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001957 self.address = ('', port)
Ka-Ping Yeedb8ed152001-03-02 05:58:17 +00001958 self.url = 'http://%s:%d/' % (host, port)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001959 self.callback = callback
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001960 self.base.__init__(self, self.address, self.handler)
1961
1962 def serve_until_quit(self):
1963 import select
Guido van Rossum8ca162f2002-04-07 06:36:23 +00001964 self.quit = False
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001965 while not self.quit:
1966 rd, wr, ex = select.select([self.socket.fileno()], [], [], 1)
1967 if rd: self.handle_request()
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001968
1969 def server_activate(self):
1970 self.base.server_activate(self)
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001971 if self.callback: self.callback(self)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001972
1973 DocServer.base = BaseHTTPServer.HTTPServer
1974 DocServer.handler = DocHandler
1975 DocHandler.MessageClass = Message
1976 try:
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001977 try:
1978 DocServer(port, callback).serve_until_quit()
1979 except (KeyboardInterrupt, select.error):
1980 pass
1981 finally:
Ka-Ping Yee66246962001-04-12 11:59:50 +00001982 if completer: completer()
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001983
1984# ----------------------------------------------------- graphical interface
1985
1986def gui():
1987 """Graphical interface (starts web server and pops up a control window)."""
1988 class GUI:
1989 def __init__(self, window, port=7464):
1990 self.window = window
1991 self.server = None
1992 self.scanner = None
1993
1994 import Tkinter
1995 self.server_frm = Tkinter.Frame(window)
1996 self.title_lbl = Tkinter.Label(self.server_frm,
1997 text='Starting server...\n ')
1998 self.open_btn = Tkinter.Button(self.server_frm,
1999 text='open browser', command=self.open, state='disabled')
2000 self.quit_btn = Tkinter.Button(self.server_frm,
2001 text='quit serving', command=self.quit, state='disabled')
2002
2003 self.search_frm = Tkinter.Frame(window)
2004 self.search_lbl = Tkinter.Label(self.search_frm, text='Search for')
2005 self.search_ent = Tkinter.Entry(self.search_frm)
2006 self.search_ent.bind('<Return>', self.search)
2007 self.stop_btn = Tkinter.Button(self.search_frm,
2008 text='stop', pady=0, command=self.stop, state='disabled')
2009 if sys.platform == 'win32':
Ka-Ping Yeec92cdf72001-03-02 05:54:35 +00002010 # Trying to hide and show this button crashes under Windows.
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002011 self.stop_btn.pack(side='right')
2012
2013 self.window.title('pydoc')
2014 self.window.protocol('WM_DELETE_WINDOW', self.quit)
2015 self.title_lbl.pack(side='top', fill='x')
2016 self.open_btn.pack(side='left', fill='x', expand=1)
2017 self.quit_btn.pack(side='right', fill='x', expand=1)
2018 self.server_frm.pack(side='top', fill='x')
2019
2020 self.search_lbl.pack(side='left')
2021 self.search_ent.pack(side='right', fill='x', expand=1)
2022 self.search_frm.pack(side='top', fill='x')
2023 self.search_ent.focus_set()
2024
Ka-Ping Yeec92cdf72001-03-02 05:54:35 +00002025 font = ('helvetica', sys.platform == 'win32' and 8 or 10)
Ka-Ping Yee239432a2001-03-02 02:45:08 +00002026 self.result_lst = Tkinter.Listbox(window, font=font, height=6)
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002027 self.result_lst.bind('<Button-1>', self.select)
2028 self.result_lst.bind('<Double-Button-1>', self.goto)
2029 self.result_scr = Tkinter.Scrollbar(window,
2030 orient='vertical', command=self.result_lst.yview)
2031 self.result_lst.config(yscrollcommand=self.result_scr.set)
2032
2033 self.result_frm = Tkinter.Frame(window)
2034 self.goto_btn = Tkinter.Button(self.result_frm,
2035 text='go to selected', command=self.goto)
2036 self.hide_btn = Tkinter.Button(self.result_frm,
2037 text='hide results', command=self.hide)
2038 self.goto_btn.pack(side='left', fill='x', expand=1)
2039 self.hide_btn.pack(side='right', fill='x', expand=1)
2040
2041 self.window.update()
2042 self.minwidth = self.window.winfo_width()
2043 self.minheight = self.window.winfo_height()
2044 self.bigminheight = (self.server_frm.winfo_reqheight() +
2045 self.search_frm.winfo_reqheight() +
2046 self.result_lst.winfo_reqheight() +
2047 self.result_frm.winfo_reqheight())
2048 self.bigwidth, self.bigheight = self.minwidth, self.bigminheight
2049 self.expanded = 0
2050 self.window.wm_geometry('%dx%d' % (self.minwidth, self.minheight))
2051 self.window.wm_minsize(self.minwidth, self.minheight)
Martin v. Löwis5b26abb2002-12-28 09:23:09 +00002052 self.window.tk.willdispatch()
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002053
2054 import threading
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00002055 threading.Thread(
2056 target=serve, args=(port, self.ready, self.quit)).start()
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002057
2058 def ready(self, server):
2059 self.server = server
2060 self.title_lbl.config(
2061 text='Python documentation server at\n' + server.url)
2062 self.open_btn.config(state='normal')
2063 self.quit_btn.config(state='normal')
2064
Ka-Ping Yee239432a2001-03-02 02:45:08 +00002065 def open(self, event=None, url=None):
2066 url = url or self.server.url
2067 try:
2068 import webbrowser
2069 webbrowser.open(url)
2070 except ImportError: # pre-webbrowser.py compatibility
Ka-Ping Yeec92cdf72001-03-02 05:54:35 +00002071 if sys.platform == 'win32':
Ka-Ping Yee239432a2001-03-02 02:45:08 +00002072 os.system('start "%s"' % url)
2073 elif sys.platform == 'mac':
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +00002074 try: import ic
Ka-Ping Yee239432a2001-03-02 02:45:08 +00002075 except ImportError: pass
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +00002076 else: ic.launchurl(url)
Ka-Ping Yee239432a2001-03-02 02:45:08 +00002077 else:
2078 rc = os.system('netscape -remote "openURL(%s)" &' % url)
2079 if rc: os.system('netscape "%s" &' % url)
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002080
2081 def quit(self, event=None):
2082 if self.server:
2083 self.server.quit = 1
2084 self.window.quit()
2085
2086 def search(self, event=None):
2087 key = self.search_ent.get()
2088 self.stop_btn.pack(side='right')
2089 self.stop_btn.config(state='normal')
2090 self.search_lbl.config(text='Searching for "%s"...' % key)
2091 self.search_ent.forget()
2092 self.search_lbl.pack(side='left')
2093 self.result_lst.delete(0, 'end')
2094 self.goto_btn.config(state='disabled')
2095 self.expand()
2096
2097 import threading
2098 if self.scanner:
2099 self.scanner.quit = 1
2100 self.scanner = ModuleScanner()
2101 threading.Thread(target=self.scanner.run,
Ka-Ping Yee6dcfa382001-04-12 20:27:31 +00002102 args=(self.update, key, self.done)).start()
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002103
2104 def update(self, path, modname, desc):
2105 if modname[-9:] == '.__init__':
2106 modname = modname[:-9] + ' (package)'
2107 self.result_lst.insert('end',
2108 modname + ' - ' + (desc or '(no description)'))
2109
2110 def stop(self, event=None):
2111 if self.scanner:
2112 self.scanner.quit = 1
2113 self.scanner = None
2114
2115 def done(self):
2116 self.scanner = None
2117 self.search_lbl.config(text='Search for')
2118 self.search_lbl.pack(side='left')
2119 self.search_ent.pack(side='right', fill='x', expand=1)
2120 if sys.platform != 'win32': self.stop_btn.forget()
2121 self.stop_btn.config(state='disabled')
2122
2123 def select(self, event=None):
2124 self.goto_btn.config(state='normal')
2125
2126 def goto(self, event=None):
2127 selection = self.result_lst.curselection()
2128 if selection:
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002129 modname = split(self.result_lst.get(selection[0]))[0]
Ka-Ping Yee239432a2001-03-02 02:45:08 +00002130 self.open(url=self.server.url + modname + '.html')
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002131
2132 def collapse(self):
2133 if not self.expanded: return
2134 self.result_frm.forget()
2135 self.result_scr.forget()
2136 self.result_lst.forget()
2137 self.bigwidth = self.window.winfo_width()
2138 self.bigheight = self.window.winfo_height()
2139 self.window.wm_geometry('%dx%d' % (self.minwidth, self.minheight))
2140 self.window.wm_minsize(self.minwidth, self.minheight)
2141 self.expanded = 0
2142
2143 def expand(self):
2144 if self.expanded: return
2145 self.result_frm.pack(side='bottom', fill='x')
2146 self.result_scr.pack(side='right', fill='y')
2147 self.result_lst.pack(side='top', fill='both', expand=1)
2148 self.window.wm_geometry('%dx%d' % (self.bigwidth, self.bigheight))
2149 self.window.wm_minsize(self.minwidth, self.bigminheight)
2150 self.expanded = 1
2151
2152 def hide(self, event=None):
2153 self.stop()
2154 self.collapse()
2155
2156 import Tkinter
2157 try:
Martin v. Löwise09bd932004-08-22 16:13:26 +00002158 root = Tkinter.Tk()
2159 # Tk will crash if pythonw.exe has an XP .manifest
2160 # file and the root has is not destroyed explicitly.
2161 # If the problem is ever fixed in Tk, the explicit
2162 # destroy can go.
2163 try:
2164 gui = GUI(root)
2165 root.mainloop()
2166 finally:
2167 root.destroy()
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002168 except KeyboardInterrupt:
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002169 pass
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002170
2171# -------------------------------------------------- command-line interface
2172
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00002173def ispath(x):
Ka-Ping Yee45daeb02002-08-11 15:11:33 +00002174 return isinstance(x, str) and find(x, os.sep) >= 0
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00002175
Ka-Ping Yee1d384632001-03-01 00:24:32 +00002176def cli():
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002177 """Command-line interface (looks at sys.argv to decide what to do)."""
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002178 import getopt
2179 class BadUsage: pass
2180
Ka-Ping Yee3bda8792001-03-23 13:17:50 +00002181 # Scripts don't get the current directory in their path by default.
Ka-Ping Yeef78a81b2001-03-27 08:13:42 +00002182 scriptdir = os.path.dirname(sys.argv[0])
2183 if scriptdir in sys.path:
2184 sys.path.remove(scriptdir)
Ka-Ping Yee3bda8792001-03-23 13:17:50 +00002185 sys.path.insert(0, '.')
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002186
Ka-Ping Yee3bda8792001-03-23 13:17:50 +00002187 try:
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002188 opts, args = getopt.getopt(sys.argv[1:], 'gk:p:w')
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002189 writing = 0
2190
2191 for opt, val in opts:
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002192 if opt == '-g':
2193 gui()
2194 return
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002195 if opt == '-k':
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002196 apropos(val)
2197 return
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002198 if opt == '-p':
2199 try:
2200 port = int(val)
2201 except ValueError:
2202 raise BadUsage
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002203 def ready(server):
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00002204 print 'pydoc server ready at %s' % server.url
2205 def stopped():
2206 print 'pydoc server stopped'
2207 serve(port, ready, stopped)
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002208 return
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002209 if opt == '-w':
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002210 writing = 1
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002211
2212 if not args: raise BadUsage
2213 for arg in args:
Ka-Ping Yee45daeb02002-08-11 15:11:33 +00002214 if ispath(arg) and not os.path.exists(arg):
2215 print 'file %r does not exist' % arg
2216 break
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002217 try:
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00002218 if ispath(arg) and os.path.isfile(arg):
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002219 arg = importfile(arg)
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00002220 if writing:
2221 if ispath(arg) and os.path.isdir(arg):
2222 writedocs(arg)
2223 else:
2224 writedoc(arg)
2225 else:
Martin v. Löwisb8c084e2003-06-14 09:03:46 +00002226 help.help(arg)
Ka-Ping Yee3bda8792001-03-23 13:17:50 +00002227 except ErrorDuringImport, value:
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00002228 print value
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002229
2230 except (getopt.error, BadUsage):
Martin v. Löwisf9b08b82003-10-31 13:05:21 +00002231 cmd = os.path.basename(sys.argv[0])
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002232 print """pydoc - the Python documentation tool
2233
2234%s <name> ...
2235 Show text documentation on something. <name> may be the name of a
Martin v. Löwisb8c084e2003-06-14 09:03:46 +00002236 Python keyword, topic, function, module, or package, or a dotted
2237 reference to a class or function within a module or module in a
2238 package. If <name> contains a '%s', it is used as the path to a
2239 Python source file to document. If name is 'keywords', 'topics',
2240 or 'modules', a listing of these things is displayed.
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002241
2242%s -k <keyword>
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002243 Search for a keyword in the synopsis lines of all available modules.
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002244
2245%s -p <port>
2246 Start an HTTP server on the given port on the local machine.
2247
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002248%s -g
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +00002249 Pop up a graphical interface for finding and serving documentation.
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002250
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002251%s -w <name> ...
2252 Write out the HTML documentation for a module to a file in the current
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +00002253 directory. If <name> contains a '%s', it is treated as a filename; if
2254 it names a directory, documentation is written for all the contents.
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002255""" % (cmd, os.sep, cmd, cmd, cmd, cmd, os.sep)
Ka-Ping Yee1d384632001-03-01 00:24:32 +00002256
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002257if __name__ == '__main__': cli()