blob: b82e0e7d7183d0519529590ae2448550752f0726 [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)."""
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000257 try:
Ka-Ping Yee9a2dcf82005-11-05 05:04:41 +0000258 # If forceload is 1 and the module has been previously loaded from
259 # disk, we always have to reload the module. Checking the file's
260 # mtime isn't good enough (e.g. the module could contain a class
261 # that inherits from another module that has changed).
262 if forceload and path in sys.modules:
263 if path not in sys.builtin_module_names:
264 # Avoid simply calling reload() because it leaves names in
265 # the currently loaded module lying around if they're not
266 # defined in the new source file. Instead, remove the
267 # module from sys.modules and re-import. Also remove any
268 # submodules because they won't appear in the newly loaded
269 # module's namespace if they're already in sys.modules.
270 subs = [m for m in sys.modules if m.startswith(path + '.')]
271 for key in [path] + subs:
272 # Prevent garbage collection.
273 cache[key] = sys.modules[key]
274 del sys.modules[key]
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000275 module = __import__(path)
276 except:
277 # Did the error occur before or after the module was found?
278 (exc, value, tb) = info = sys.exc_info()
Raymond Hettinger54f02222002-06-01 14:18:47 +0000279 if path in sys.modules:
Fred Drakedb390c12005-10-28 14:39:47 +0000280 # An error occurred while executing the imported module.
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000281 raise ErrorDuringImport(sys.modules[path].__file__, info)
282 elif exc is SyntaxError:
283 # A SyntaxError occurred before we could execute the module.
284 raise ErrorDuringImport(value.filename, info)
285 elif exc is ImportError and \
286 split(lower(str(value)))[:2] == ['no', 'module']:
287 # The module was not found.
288 return None
289 else:
290 # Some other error occurred during the importing process.
291 raise ErrorDuringImport(path, sys.exc_info())
292 for part in split(path, '.')[1:]:
293 try: module = getattr(module, part)
294 except AttributeError: return None
295 return module
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000296
297# ---------------------------------------------------- formatter base class
298
299class Doc:
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000300 def document(self, object, name=None, *args):
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000301 """Generate documentation for an object."""
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000302 args = (object, name) + args
Brett Cannon28a4f0f2003-06-11 23:38:55 +0000303 # 'try' clause is to attempt to handle the possibility that inspect
304 # identifies something in a way that pydoc itself has issues handling;
305 # think 'super' and how it is a descriptor (which raises the exception
306 # by lacking a __name__ attribute) and an instance.
307 try:
308 if inspect.ismodule(object): return self.docmodule(*args)
309 if inspect.isclass(object): return self.docclass(*args)
310 if inspect.isroutine(object): return self.docroutine(*args)
311 except AttributeError:
312 pass
Johannes Gijsbers8de645a2004-11-07 19:16:05 +0000313 if isinstance(object, property): return self.docproperty(*args)
Guido van Rossum68468eb2003-02-27 20:14:51 +0000314 return self.docother(*args)
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000315
316 def fail(self, object, name=None, *args):
317 """Raise an exception for unimplemented types."""
318 message = "don't know how to document object%s of type %s" % (
319 name and ' ' + repr(name), type(object).__name__)
320 raise TypeError, message
321
322 docmodule = docclass = docroutine = docother = fail
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000323
Skip Montanaro4997a692003-09-10 16:47:51 +0000324 def getdocloc(self, object):
325 """Return the location of module docs or None"""
326
327 try:
328 file = inspect.getabsfile(object)
329 except TypeError:
330 file = '(built-in)'
331
332 docloc = os.environ.get("PYTHONDOCS",
333 "http://www.python.org/doc/current/lib")
334 basedir = os.path.join(sys.exec_prefix, "lib",
335 "python"+sys.version[0:3])
336 if (isinstance(object, type(os)) and
337 (object.__name__ in ('errno', 'exceptions', 'gc', 'imp',
338 'marshal', 'posix', 'signal', 'sys',
339 'thread', 'zipimport') or
340 (file.startswith(basedir) and
341 not file.startswith(os.path.join(basedir, 'site-packages'))))):
Skip Montanarof2134842004-06-07 02:40:05 +0000342 htmlfile = "module-%s.html" % object.__name__
Skip Montanaro4997a692003-09-10 16:47:51 +0000343 if docloc.startswith("http://"):
Skip Montanarof2134842004-06-07 02:40:05 +0000344 docloc = "%s/%s" % (docloc.rstrip("/"), htmlfile)
Skip Montanaro4997a692003-09-10 16:47:51 +0000345 else:
Skip Montanarof2134842004-06-07 02:40:05 +0000346 docloc = os.path.join(docloc, htmlfile)
Skip Montanaro4997a692003-09-10 16:47:51 +0000347 else:
348 docloc = None
349 return docloc
350
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000351# -------------------------------------------- HTML documentation generator
352
353class HTMLRepr(Repr):
354 """Class for safely making an HTML representation of a Python object."""
355 def __init__(self):
356 Repr.__init__(self)
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000357 self.maxlist = self.maxtuple = 20
358 self.maxdict = 10
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000359 self.maxstring = self.maxother = 100
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000360
361 def escape(self, text):
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +0000362 return replace(text, '&', '&amp;', '<', '&lt;', '>', '&gt;')
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000363
364 def repr(self, object):
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000365 return Repr.repr(self, object)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000366
367 def repr1(self, x, level):
Skip Montanaro0fe8fce2003-06-27 15:45:41 +0000368 if hasattr(type(x), '__name__'):
369 methodname = 'repr_' + join(split(type(x).__name__), '_')
370 if hasattr(self, methodname):
371 return getattr(self, methodname)(x, level)
372 return self.escape(cram(stripid(repr(x)), self.maxother))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000373
374 def repr_string(self, x, level):
Ka-Ping Yeea2fe1032001-03-02 01:19:14 +0000375 test = cram(x, self.maxstring)
376 testrepr = repr(test)
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +0000377 if '\\' in test and '\\' not in replace(testrepr, r'\\', ''):
Ka-Ping Yeea2fe1032001-03-02 01:19:14 +0000378 # Backslashes are only literal in the string and are never
379 # needed to make any special characters, so show a raw string.
380 return 'r' + testrepr[0] + self.escape(test) + testrepr[0]
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +0000381 return re.sub(r'((\\[\\abfnrtv\'"]|\\[0-9]..|\\x..|\\u....)+)',
Ka-Ping Yeea2fe1032001-03-02 01:19:14 +0000382 r'<font color="#c040c0">\1</font>',
383 self.escape(testrepr))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000384
Skip Montanarodf708782002-03-07 22:58:02 +0000385 repr_str = repr_string
386
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000387 def repr_instance(self, x, level):
388 try:
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000389 return self.escape(cram(stripid(repr(x)), self.maxstring))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000390 except:
391 return self.escape('<%s instance>' % x.__class__.__name__)
392
393 repr_unicode = repr_string
394
395class HTMLDoc(Doc):
396 """Formatter class for HTML documentation."""
397
398 # ------------------------------------------- HTML formatting utilities
399
400 _repr_instance = HTMLRepr()
401 repr = _repr_instance.repr
402 escape = _repr_instance.escape
403
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000404 def page(self, title, contents):
405 """Format an HTML page."""
406 return '''
Tim Peters59ed4482001-10-31 04:20:26 +0000407<!doctype html PUBLIC "-//W3C//DTD HTML 4.0 Transitional//EN">
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000408<html><head><title>Python: %s</title>
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000409</head><body bgcolor="#f0f0f8">
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000410%s
411</body></html>''' % (title, contents)
412
413 def heading(self, title, fgcol, bgcol, extras=''):
414 """Format a page heading."""
415 return '''
Tim Peters59ed4482001-10-31 04:20:26 +0000416<table width="100%%" cellspacing=0 cellpadding=2 border=0 summary="heading">
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000417<tr bgcolor="%s">
Tim Peters2306d242001-09-25 03:18:32 +0000418<td valign=bottom>&nbsp;<br>
419<font color="%s" face="helvetica, arial">&nbsp;<br>%s</font></td
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000420><td align=right valign=bottom
Ka-Ping Yee987ec902001-03-23 13:35:45 +0000421><font color="%s" face="helvetica, arial">%s</font></td></tr></table>
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000422 ''' % (bgcol, fgcol, title, fgcol, extras or '&nbsp;')
423
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000424 def section(self, title, fgcol, bgcol, contents, width=6,
425 prelude='', marginalia=None, gap='&nbsp;'):
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000426 """Format a section with a heading."""
427 if marginalia is None:
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +0000428 marginalia = '<tt>' + '&nbsp;' * width + '</tt>'
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000429 result = '''<p>
Tim Peters59ed4482001-10-31 04:20:26 +0000430<table width="100%%" cellspacing=0 cellpadding=2 border=0 summary="section">
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000431<tr bgcolor="%s">
Tim Peters2306d242001-09-25 03:18:32 +0000432<td colspan=3 valign=bottom>&nbsp;<br>
433<font color="%s" face="helvetica, arial">%s</font></td></tr>
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000434 ''' % (bgcol, fgcol, title)
435 if prelude:
436 result = result + '''
Ka-Ping Yee987ec902001-03-23 13:35:45 +0000437<tr bgcolor="%s"><td rowspan=2>%s</td>
438<td colspan=2>%s</td></tr>
439<tr><td>%s</td>''' % (bgcol, marginalia, prelude, gap)
440 else:
441 result = result + '''
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000442<tr><td bgcolor="%s">%s</td><td>%s</td>''' % (bgcol, marginalia, gap)
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000443
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +0000444 return result + '\n<td width="100%%">%s</td></tr></table>' % contents
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000445
446 def bigsection(self, title, *args):
447 """Format a section with a big heading."""
448 title = '<big><strong>%s</strong></big>' % title
Guido van Rossum68468eb2003-02-27 20:14:51 +0000449 return self.section(title, *args)
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000450
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000451 def preformat(self, text):
452 """Format literal preformatted text."""
453 text = self.escape(expandtabs(text))
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +0000454 return replace(text, '\n\n', '\n \n', '\n\n', '\n \n',
455 ' ', '&nbsp;', '\n', '<br>\n')
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000456
457 def multicolumn(self, list, format, cols=4):
458 """Format a list of items into a multi-column list."""
459 result = ''
460 rows = (len(list)+cols-1)/cols
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000461 for col in range(cols):
462 result = result + '<td width="%d%%" valign=top>' % (100/cols)
463 for i in range(rows*col, rows*col+rows):
464 if i < len(list):
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000465 result = result + format(list[i]) + '<br>\n'
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000466 result = result + '</td>'
Tim Peters59ed4482001-10-31 04:20:26 +0000467 return '<table width="100%%" summary="list"><tr>%s</tr></table>' % result
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000468
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000469 def grey(self, text): return '<font color="#909090">%s</font>' % text
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000470
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000471 def namelink(self, name, *dicts):
472 """Make a link for an identifier, given name-to-URL mappings."""
473 for dict in dicts:
Raymond Hettinger54f02222002-06-01 14:18:47 +0000474 if name in dict:
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000475 return '<a href="%s">%s</a>' % (dict[name], name)
476 return name
477
Ka-Ping Yeeb7a48302001-04-12 20:39:14 +0000478 def classlink(self, object, modname):
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000479 """Make a link for a class."""
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000480 name, module = object.__name__, sys.modules.get(object.__module__)
481 if hasattr(module, name) and getattr(module, name) is object:
Ka-Ping Yeeb7a48302001-04-12 20:39:14 +0000482 return '<a href="%s.html#%s">%s</a>' % (
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000483 module.__name__, name, classname(object, modname))
484 return classname(object, modname)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000485
486 def modulelink(self, object):
487 """Make a link for a module."""
488 return '<a href="%s.html">%s</a>' % (object.__name__, object.__name__)
489
490 def modpkglink(self, (name, path, ispackage, shadowed)):
491 """Make a link for a module or package to display in an index."""
492 if shadowed:
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000493 return self.grey(name)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000494 if path:
495 url = '%s.%s.html' % (path, name)
496 else:
497 url = '%s.html' % name
498 if ispackage:
499 text = '<strong>%s</strong>&nbsp;(package)' % name
500 else:
501 text = name
502 return '<a href="%s">%s</a>' % (url, text)
503
504 def markup(self, text, escape=None, funcs={}, classes={}, methods={}):
505 """Mark up some plain text, given a context of symbols to look for.
506 Each context dictionary maps object names to anchor names."""
507 escape = escape or self.escape
508 results = []
509 here = 0
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000510 pattern = re.compile(r'\b((http|ftp)://\S+[\w/]|'
511 r'RFC[- ]?(\d+)|'
Ka-Ping Yeef78a81b2001-03-27 08:13:42 +0000512 r'PEP[- ]?(\d+)|'
Neil Schemenauerd69711c2002-03-24 23:02:07 +0000513 r'(self\.)?(\w+))')
Guido van Rossum8ca162f2002-04-07 06:36:23 +0000514 while True:
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000515 match = pattern.search(text, here)
516 if not match: break
517 start, end = match.span()
518 results.append(escape(text[here:start]))
519
Ka-Ping Yeef78a81b2001-03-27 08:13:42 +0000520 all, scheme, rfc, pep, selfdot, name = match.groups()
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000521 if scheme:
Neil Schemenauercddc1a02002-03-24 23:11:21 +0000522 url = escape(all).replace('"', '&quot;')
523 results.append('<a href="%s">%s</a>' % (url, url))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000524 elif rfc:
Ka-Ping Yeef78a81b2001-03-27 08:13:42 +0000525 url = 'http://www.rfc-editor.org/rfc/rfc%d.txt' % int(rfc)
526 results.append('<a href="%s">%s</a>' % (url, escape(all)))
527 elif pep:
528 url = 'http://www.python.org/peps/pep-%04d.html' % int(pep)
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000529 results.append('<a href="%s">%s</a>' % (url, escape(all)))
530 elif text[end:end+1] == '(':
531 results.append(self.namelink(name, methods, funcs, classes))
532 elif selfdot:
533 results.append('self.<strong>%s</strong>' % name)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000534 else:
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000535 results.append(self.namelink(name, classes))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000536 here = end
537 results.append(escape(text[here:]))
538 return join(results, '')
539
540 # ---------------------------------------------- type-specific routines
541
Ka-Ping Yeeb7a48302001-04-12 20:39:14 +0000542 def formattree(self, tree, modname, parent=None):
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000543 """Produce HTML for a class tree as given by inspect.getclasstree()."""
544 result = ''
545 for entry in tree:
546 if type(entry) is type(()):
547 c, bases = entry
Tim Peters2306d242001-09-25 03:18:32 +0000548 result = result + '<dt><font face="helvetica, arial">'
Ka-Ping Yeeb7a48302001-04-12 20:39:14 +0000549 result = result + self.classlink(c, modname)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000550 if bases and bases != (parent,):
551 parents = []
552 for base in bases:
Ka-Ping Yeeb7a48302001-04-12 20:39:14 +0000553 parents.append(self.classlink(base, modname))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000554 result = result + '(' + join(parents, ', ') + ')'
Tim Peters2306d242001-09-25 03:18:32 +0000555 result = result + '\n</font></dt>'
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000556 elif type(entry) is type([]):
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000557 result = result + '<dd>\n%s</dd>\n' % self.formattree(
Ka-Ping Yeeb7a48302001-04-12 20:39:14 +0000558 entry, modname, c)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000559 return '<dl>\n%s</dl>\n' % result
560
Tim Peters8dd7ade2001-10-18 19:56:17 +0000561 def docmodule(self, object, name=None, mod=None, *ignored):
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000562 """Produce HTML documentation for a module object."""
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000563 name = object.__name__ # ignore the passed-in name
Skip Montanaroa5616d22004-06-11 04:46:12 +0000564 try:
565 all = object.__all__
566 except AttributeError:
567 all = None
Ka-Ping Yee66efbc72001-03-01 13:55:20 +0000568 parts = split(name, '.')
569 links = []
570 for i in range(len(parts)-1):
571 links.append(
572 '<a href="%s.html"><font color="#ffffff">%s</font></a>' %
573 (join(parts[:i+1], '.'), parts[i]))
574 linkedname = join(links + parts[-1:], '.')
575 head = '<big><big><strong>%s</strong></big></big>' % linkedname
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000576 try:
Ka-Ping Yee239432a2001-03-02 02:45:08 +0000577 path = inspect.getabsfile(object)
Ka-Ping Yee6191a232001-04-13 15:00:27 +0000578 url = path
579 if sys.platform == 'win32':
580 import nturl2path
581 url = nturl2path.pathname2url(path)
582 filelink = '<a href="file:%s">%s</a>' % (url, path)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000583 except TypeError:
584 filelink = '(built-in)'
Ka-Ping Yee6f3f9a42001-02-27 22:42:36 +0000585 info = []
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000586 if hasattr(object, '__version__'):
Ka-Ping Yee6f3f9a42001-02-27 22:42:36 +0000587 version = str(object.__version__)
Ka-Ping Yee40c49912001-02-27 22:46:01 +0000588 if version[:11] == '$' + 'Revision: ' and version[-1:] == '$':
589 version = strip(version[11:-1])
Ka-Ping Yee1d384632001-03-01 00:24:32 +0000590 info.append('version %s' % self.escape(version))
Ka-Ping Yee6f3f9a42001-02-27 22:42:36 +0000591 if hasattr(object, '__date__'):
592 info.append(self.escape(str(object.__date__)))
593 if info:
594 head = head + ' (%s)' % join(info, ', ')
Skip Montanaro4997a692003-09-10 16:47:51 +0000595 docloc = self.getdocloc(object)
596 if docloc is not None:
597 docloc = '<br><a href="%(docloc)s">Module Docs</a>' % locals()
598 else:
599 docloc = ''
Ka-Ping Yee66efbc72001-03-01 13:55:20 +0000600 result = self.heading(
Skip Montanaro4997a692003-09-10 16:47:51 +0000601 head, '#ffffff', '#7799ee',
602 '<a href=".">index</a><br>' + filelink + docloc)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000603
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000604 modules = inspect.getmembers(object, inspect.ismodule)
605
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000606 classes, cdict = [], {}
607 for key, value in inspect.getmembers(object, inspect.isclass):
Johannes Gijsbers4c11f602004-08-30 14:13:04 +0000608 # if __all__ exists, believe it. Otherwise use old heuristic.
609 if (all is not None or
610 (inspect.getmodule(value) or object) is object):
Skip Montanaroa5616d22004-06-11 04:46:12 +0000611 if visiblename(key, all):
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000612 classes.append((key, value))
613 cdict[key] = cdict[value] = '#' + key
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000614 for key, value in classes:
615 for base in value.__bases__:
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000616 key, modname = base.__name__, base.__module__
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000617 module = sys.modules.get(modname)
618 if modname != name and module and hasattr(module, key):
619 if getattr(module, key) is base:
Raymond Hettinger54f02222002-06-01 14:18:47 +0000620 if not key in cdict:
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000621 cdict[key] = cdict[base] = modname + '.html#' + key
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000622 funcs, fdict = [], {}
623 for key, value in inspect.getmembers(object, inspect.isroutine):
Johannes Gijsbers4c11f602004-08-30 14:13:04 +0000624 # if __all__ exists, believe it. Otherwise use old heuristic.
625 if (all is not None or
626 inspect.isbuiltin(value) or inspect.getmodule(value) is object):
Skip Montanaroa5616d22004-06-11 04:46:12 +0000627 if visiblename(key, all):
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000628 funcs.append((key, value))
629 fdict[key] = '#-' + key
630 if inspect.isfunction(value): fdict[value] = fdict[key]
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000631 data = []
632 for key, value in inspect.getmembers(object, isdata):
Skip Montanaroa5616d22004-06-11 04:46:12 +0000633 if visiblename(key, all):
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000634 data.append((key, value))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000635
636 doc = self.markup(getdoc(object), self.preformat, fdict, cdict)
637 doc = doc and '<tt>%s</tt>' % doc
Tim Peters2306d242001-09-25 03:18:32 +0000638 result = result + '<p>%s</p>\n' % doc
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000639
640 if hasattr(object, '__path__'):
641 modpkgs = []
642 modnames = []
643 for file in os.listdir(object.__path__[0]):
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +0000644 path = os.path.join(object.__path__[0], file)
645 modname = inspect.getmodulename(file)
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000646 if modname != '__init__':
647 if modname and modname not in modnames:
648 modpkgs.append((modname, name, 0, 0))
649 modnames.append(modname)
650 elif ispackage(path):
651 modpkgs.append((file, name, 1, 0))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000652 modpkgs.sort()
653 contents = self.multicolumn(modpkgs, self.modpkglink)
654 result = result + self.bigsection(
655 'Package Contents', '#ffffff', '#aa55cc', contents)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000656 elif modules:
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000657 contents = self.multicolumn(
658 modules, lambda (key, value), s=self: s.modulelink(value))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000659 result = result + self.bigsection(
660 'Modules', '#fffff', '#aa55cc', contents)
661
662 if classes:
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000663 classlist = map(lambda (key, value): value, classes)
Ka-Ping Yeeb7a48302001-04-12 20:39:14 +0000664 contents = [
665 self.formattree(inspect.getclasstree(classlist, 1), name)]
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000666 for key, value in classes:
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +0000667 contents.append(self.document(value, key, name, fdict, cdict))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000668 result = result + self.bigsection(
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000669 'Classes', '#ffffff', '#ee77aa', join(contents))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000670 if funcs:
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000671 contents = []
672 for key, value in funcs:
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +0000673 contents.append(self.document(value, key, name, fdict, cdict))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000674 result = result + self.bigsection(
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000675 'Functions', '#ffffff', '#eeaa77', join(contents))
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000676 if data:
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000677 contents = []
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000678 for key, value in data:
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000679 contents.append(self.document(value, key))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000680 result = result + self.bigsection(
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000681 'Data', '#ffffff', '#55aa55', join(contents, '<br>\n'))
Ka-Ping Yee6f3f9a42001-02-27 22:42:36 +0000682 if hasattr(object, '__author__'):
683 contents = self.markup(str(object.__author__), self.preformat)
684 result = result + self.bigsection(
685 'Author', '#ffffff', '#7799ee', contents)
Ka-Ping Yee6f3f9a42001-02-27 22:42:36 +0000686 if hasattr(object, '__credits__'):
687 contents = self.markup(str(object.__credits__), self.preformat)
688 result = result + self.bigsection(
689 'Credits', '#ffffff', '#7799ee', contents)
690
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000691 return result
692
Tim Peters8dd7ade2001-10-18 19:56:17 +0000693 def docclass(self, object, name=None, mod=None, funcs={}, classes={},
694 *ignored):
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000695 """Produce HTML documentation for a class object."""
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000696 realname = object.__name__
697 name = name or realname
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000698 bases = object.__bases__
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000699
Tim Petersb47879b2001-09-24 04:47:19 +0000700 contents = []
701 push = contents.append
702
Tim Petersfa26f7c2001-09-24 08:05:11 +0000703 # Cute little class to pump out a horizontal rule between sections.
704 class HorizontalRule:
705 def __init__(self):
706 self.needone = 0
707 def maybe(self):
708 if self.needone:
709 push('<hr>\n')
710 self.needone = 1
711 hr = HorizontalRule()
712
Tim Petersc86f6ca2001-09-26 21:31:51 +0000713 # List the mro, if non-trivial.
Raymond Hettinger756b3f32004-01-29 06:37:52 +0000714 mro = deque(inspect.getmro(object))
Tim Petersc86f6ca2001-09-26 21:31:51 +0000715 if len(mro) > 2:
716 hr.maybe()
717 push('<dl><dt>Method resolution order:</dt>\n')
718 for base in mro:
719 push('<dd>%s</dd>\n' % self.classlink(base,
720 object.__module__))
721 push('</dl>\n')
722
Tim Petersb47879b2001-09-24 04:47:19 +0000723 def spill(msg, attrs, predicate):
Tim Petersfa26f7c2001-09-24 08:05:11 +0000724 ok, attrs = _split_list(attrs, predicate)
Tim Petersb47879b2001-09-24 04:47:19 +0000725 if ok:
Tim Petersfa26f7c2001-09-24 08:05:11 +0000726 hr.maybe()
Tim Petersb47879b2001-09-24 04:47:19 +0000727 push(msg)
728 for name, kind, homecls, value in ok:
729 push(self.document(getattr(object, name), name, mod,
730 funcs, classes, mdict, object))
731 push('\n')
732 return attrs
733
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +0000734 def spilldescriptors(msg, attrs, predicate):
Tim Petersfa26f7c2001-09-24 08:05:11 +0000735 ok, attrs = _split_list(attrs, predicate)
Tim Petersb47879b2001-09-24 04:47:19 +0000736 if ok:
Tim Petersfa26f7c2001-09-24 08:05:11 +0000737 hr.maybe()
Tim Petersb47879b2001-09-24 04:47:19 +0000738 push(msg)
739 for name, kind, homecls, value in ok:
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +0000740 push(self._docdescriptor(name, value, mod))
Tim Petersb47879b2001-09-24 04:47:19 +0000741 return attrs
742
Tim Petersfa26f7c2001-09-24 08:05:11 +0000743 def spilldata(msg, attrs, predicate):
744 ok, attrs = _split_list(attrs, predicate)
Tim Petersb47879b2001-09-24 04:47:19 +0000745 if ok:
Tim Petersfa26f7c2001-09-24 08:05:11 +0000746 hr.maybe()
Tim Petersb47879b2001-09-24 04:47:19 +0000747 push(msg)
748 for name, kind, homecls, value in ok:
749 base = self.docother(getattr(object, name), name, mod)
Martin v. Löwise59e2ba2003-05-03 09:09:02 +0000750 if callable(value) or inspect.isdatadescriptor(value):
Guido van Rossum5e355b22002-05-21 20:56:15 +0000751 doc = getattr(value, "__doc__", None)
752 else:
753 doc = None
Tim Petersb47879b2001-09-24 04:47:19 +0000754 if doc is None:
755 push('<dl><dt>%s</dl>\n' % base)
756 else:
757 doc = self.markup(getdoc(value), self.preformat,
758 funcs, classes, mdict)
Tim Peters2306d242001-09-25 03:18:32 +0000759 doc = '<dd><tt>%s</tt>' % doc
Tim Petersb47879b2001-09-24 04:47:19 +0000760 push('<dl><dt>%s%s</dl>\n' % (base, doc))
761 push('\n')
762 return attrs
763
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000764 attrs = filter(lambda (name, kind, cls, value): visiblename(name),
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +0000765 classify_class_attrs(object))
Tim Petersb47879b2001-09-24 04:47:19 +0000766 mdict = {}
767 for key, kind, homecls, value in attrs:
768 mdict[key] = anchor = '#' + name + '-' + key
769 value = getattr(object, key)
770 try:
771 # The value may not be hashable (e.g., a data attr with
772 # a dict or list value).
773 mdict[value] = anchor
774 except TypeError:
775 pass
776
Tim Petersfa26f7c2001-09-24 08:05:11 +0000777 while attrs:
Tim Peters351e3622001-09-27 03:29:51 +0000778 if mro:
Raymond Hettinger756b3f32004-01-29 06:37:52 +0000779 thisclass = mro.popleft()
Tim Peters351e3622001-09-27 03:29:51 +0000780 else:
781 thisclass = attrs[0][2]
Tim Petersfa26f7c2001-09-24 08:05:11 +0000782 attrs, inherited = _split_list(attrs, lambda t: t[2] is thisclass)
783
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000784 if thisclass is __builtin__.object:
785 attrs = inherited
786 continue
787 elif thisclass is object:
788 tag = 'defined here'
Tim Petersb47879b2001-09-24 04:47:19 +0000789 else:
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000790 tag = 'inherited from %s' % self.classlink(thisclass,
791 object.__module__)
Tim Petersb47879b2001-09-24 04:47:19 +0000792 tag += ':<br>\n'
793
794 # Sort attrs by name.
Raymond Hettinger6b59f5f2003-10-16 05:53:16 +0000795 attrs.sort(key=lambda t: t[0])
Tim Petersb47879b2001-09-24 04:47:19 +0000796
797 # Pump out the attrs, segregated by kind.
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000798 attrs = spill('Methods %s' % tag, attrs,
Tim Petersb47879b2001-09-24 04:47:19 +0000799 lambda t: t[1] == 'method')
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000800 attrs = spill('Class methods %s' % tag, attrs,
Tim Petersb47879b2001-09-24 04:47:19 +0000801 lambda t: t[1] == 'class method')
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000802 attrs = spill('Static methods %s' % tag, attrs,
Tim Petersb47879b2001-09-24 04:47:19 +0000803 lambda t: t[1] == 'static method')
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +0000804 attrs = spilldescriptors('Data descriptors %s' % tag, attrs,
805 lambda t: t[1] == 'data descriptor')
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000806 attrs = spilldata('Data and other attributes %s' % tag, attrs,
Tim Petersfa26f7c2001-09-24 08:05:11 +0000807 lambda t: t[1] == 'data')
Tim Petersb47879b2001-09-24 04:47:19 +0000808 assert attrs == []
Tim Peters351e3622001-09-27 03:29:51 +0000809 attrs = inherited
Tim Petersb47879b2001-09-24 04:47:19 +0000810
811 contents = ''.join(contents)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000812
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000813 if name == realname:
814 title = '<a name="%s">class <strong>%s</strong></a>' % (
815 name, realname)
816 else:
817 title = '<strong>%s</strong> = <a name="%s">class %s</a>' % (
818 name, name, realname)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000819 if bases:
820 parents = []
821 for base in bases:
Ka-Ping Yeeb7a48302001-04-12 20:39:14 +0000822 parents.append(self.classlink(base, object.__module__))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000823 title = title + '(%s)' % join(parents, ', ')
Tim Peters2306d242001-09-25 03:18:32 +0000824 doc = self.markup(getdoc(object), self.preformat, funcs, classes, mdict)
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000825 doc = doc and '<tt>%s<br>&nbsp;</tt>' % doc
Tim Petersc86f6ca2001-09-26 21:31:51 +0000826
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000827 return self.section(title, '#000000', '#ffc8d8', contents, 3, doc)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000828
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000829 def formatvalue(self, object):
830 """Format an argument default value as text."""
Tim Peters2306d242001-09-25 03:18:32 +0000831 return self.grey('=' + self.repr(object))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000832
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +0000833 def docroutine(self, object, name=None, mod=None,
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000834 funcs={}, classes={}, methods={}, cl=None):
Ka-Ping Yee66efbc72001-03-01 13:55:20 +0000835 """Produce HTML documentation for a function or method object."""
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000836 realname = object.__name__
837 name = name or realname
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000838 anchor = (cl and cl.__name__ or '') + '-' + name
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000839 note = ''
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000840 skipdocs = 0
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000841 if inspect.ismethod(object):
Ka-Ping Yee6dcfa382001-04-12 20:27:31 +0000842 imclass = object.im_class
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000843 if cl:
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +0000844 if imclass is not cl:
Ka-Ping Yeeb7a48302001-04-12 20:39:14 +0000845 note = ' from ' + self.classlink(imclass, mod)
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000846 else:
Ka-Ping Yeeb7a48302001-04-12 20:39:14 +0000847 if object.im_self:
848 note = ' method of %s instance' % self.classlink(
849 object.im_self.__class__, mod)
850 else:
851 note = ' unbound %s method' % self.classlink(imclass,mod)
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000852 object = object.im_func
853
854 if name == realname:
855 title = '<a name="%s"><strong>%s</strong></a>' % (anchor, realname)
856 else:
Raymond Hettinger54f02222002-06-01 14:18:47 +0000857 if (cl and realname in cl.__dict__ and
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000858 cl.__dict__[realname] is object):
Ka-Ping Yeee280c062001-03-23 14:05:53 +0000859 reallink = '<a href="#%s">%s</a>' % (
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000860 cl.__name__ + '-' + realname, realname)
861 skipdocs = 1
862 else:
863 reallink = realname
864 title = '<a name="%s"><strong>%s</strong></a> = %s' % (
865 anchor, name, reallink)
Tim Peters4bcfa312001-09-20 06:08:24 +0000866 if inspect.isfunction(object):
Ka-Ping Yee66efbc72001-03-01 13:55:20 +0000867 args, varargs, varkw, defaults = inspect.getargspec(object)
868 argspec = inspect.formatargspec(
869 args, varargs, varkw, defaults, formatvalue=self.formatvalue)
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000870 if realname == '<lambda>':
Tim Peters59ed4482001-10-31 04:20:26 +0000871 title = '<strong>%s</strong> <em>lambda</em> ' % name
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000872 argspec = argspec[1:-1] # remove parentheses
Tim Peters4bcfa312001-09-20 06:08:24 +0000873 else:
874 argspec = '(...)'
Ka-Ping Yee66efbc72001-03-01 13:55:20 +0000875
Tim Peters2306d242001-09-25 03:18:32 +0000876 decl = title + argspec + (note and self.grey(
877 '<font face="helvetica, arial">%s</font>' % note))
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000878
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000879 if skipdocs:
Tim Peters2306d242001-09-25 03:18:32 +0000880 return '<dl><dt>%s</dt></dl>\n' % decl
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000881 else:
882 doc = self.markup(
883 getdoc(object), self.preformat, funcs, classes, methods)
Tim Peters2306d242001-09-25 03:18:32 +0000884 doc = doc and '<dd><tt>%s</tt></dd>' % doc
885 return '<dl><dt>%s</dt>%s</dl>\n' % (decl, doc)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000886
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +0000887 def _docdescriptor(self, name, value, mod):
Johannes Gijsbers8de645a2004-11-07 19:16:05 +0000888 results = []
889 push = results.append
890
891 if name:
892 push('<dl><dt><strong>%s</strong></dt>\n' % name)
893 if value.__doc__ is not None:
Ka-Ping Yeebba6acc2005-02-19 22:58:26 +0000894 doc = self.markup(getdoc(value), self.preformat)
Johannes Gijsbers8de645a2004-11-07 19:16:05 +0000895 push('<dd><tt>%s</tt></dd>\n' % doc)
Johannes Gijsbers8de645a2004-11-07 19:16:05 +0000896 push('</dl>\n')
897
898 return ''.join(results)
899
900 def docproperty(self, object, name=None, mod=None, cl=None):
901 """Produce html documentation for a property."""
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +0000902 return self._docdescriptor(name, object, mod)
Johannes Gijsbers8de645a2004-11-07 19:16:05 +0000903
Tim Peters8dd7ade2001-10-18 19:56:17 +0000904 def docother(self, object, name=None, mod=None, *ignored):
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000905 """Produce HTML documentation for a data object."""
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +0000906 lhs = name and '<strong>%s</strong> = ' % name or ''
907 return lhs + self.repr(object)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000908
909 def index(self, dir, shadowed=None):
910 """Generate an HTML index for a directory of modules."""
911 modpkgs = []
912 if shadowed is None: shadowed = {}
913 seen = {}
914 files = os.listdir(dir)
915
916 def found(name, ispackage,
917 modpkgs=modpkgs, shadowed=shadowed, seen=seen):
Raymond Hettinger4f759d82002-11-02 02:02:46 +0000918 if name not in seen:
919 modpkgs.append((name, '', ispackage, name in shadowed))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000920 seen[name] = 1
921 shadowed[name] = 1
922
923 # Package spam/__init__.py takes precedence over module spam.py.
924 for file in files:
925 path = os.path.join(dir, file)
926 if ispackage(path): found(file, 1)
927 for file in files:
928 path = os.path.join(dir, file)
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +0000929 if os.path.isfile(path):
930 modname = inspect.getmodulename(file)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000931 if modname: found(modname, 0)
932
933 modpkgs.sort()
934 contents = self.multicolumn(modpkgs, self.modpkglink)
935 return self.bigsection(dir, '#ffffff', '#ee77aa', contents)
936
937# -------------------------------------------- text documentation generator
938
939class TextRepr(Repr):
940 """Class for safely making a text representation of a Python object."""
941 def __init__(self):
942 Repr.__init__(self)
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000943 self.maxlist = self.maxtuple = 20
944 self.maxdict = 10
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000945 self.maxstring = self.maxother = 100
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000946
947 def repr1(self, x, level):
Skip Montanaro0fe8fce2003-06-27 15:45:41 +0000948 if hasattr(type(x), '__name__'):
949 methodname = 'repr_' + join(split(type(x).__name__), '_')
950 if hasattr(self, methodname):
951 return getattr(self, methodname)(x, level)
952 return cram(stripid(repr(x)), self.maxother)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000953
Ka-Ping Yeea2fe1032001-03-02 01:19:14 +0000954 def repr_string(self, x, level):
955 test = cram(x, self.maxstring)
956 testrepr = repr(test)
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +0000957 if '\\' in test and '\\' not in replace(testrepr, r'\\', ''):
Ka-Ping Yeea2fe1032001-03-02 01:19:14 +0000958 # Backslashes are only literal in the string and are never
959 # needed to make any special characters, so show a raw string.
960 return 'r' + testrepr[0] + test + testrepr[0]
961 return testrepr
962
Skip Montanarodf708782002-03-07 22:58:02 +0000963 repr_str = repr_string
964
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000965 def repr_instance(self, x, level):
966 try:
Ka-Ping Yee1d384632001-03-01 00:24:32 +0000967 return cram(stripid(repr(x)), self.maxstring)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000968 except:
969 return '<%s instance>' % x.__class__.__name__
970
971class TextDoc(Doc):
972 """Formatter class for text documentation."""
973
974 # ------------------------------------------- text formatting utilities
975
976 _repr_instance = TextRepr()
977 repr = _repr_instance.repr
978
979 def bold(self, text):
980 """Format a string in bold by overstriking."""
981 return join(map(lambda ch: ch + '\b' + ch, text), '')
982
983 def indent(self, text, prefix=' '):
984 """Indent text by prepending a given prefix to each line."""
985 if not text: return ''
986 lines = split(text, '\n')
987 lines = map(lambda line, prefix=prefix: prefix + line, lines)
988 if lines: lines[-1] = rstrip(lines[-1])
989 return join(lines, '\n')
990
991 def section(self, title, contents):
992 """Format a section with a given heading."""
993 return self.bold(title) + '\n' + rstrip(self.indent(contents)) + '\n\n'
994
995 # ---------------------------------------------- type-specific routines
996
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000997 def formattree(self, tree, modname, parent=None, prefix=''):
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000998 """Render in text a class tree as returned by inspect.getclasstree()."""
999 result = ''
1000 for entry in tree:
1001 if type(entry) is type(()):
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001002 c, bases = entry
1003 result = result + prefix + classname(c, modname)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001004 if bases and bases != (parent,):
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001005 parents = map(lambda c, m=modname: classname(c, m), bases)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001006 result = result + '(%s)' % join(parents, ', ')
1007 result = result + '\n'
1008 elif type(entry) is type([]):
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001009 result = result + self.formattree(
1010 entry, modname, c, prefix + ' ')
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001011 return result
1012
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001013 def docmodule(self, object, name=None, mod=None):
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001014 """Produce text documentation for a given module object."""
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001015 name = object.__name__ # ignore the passed-in name
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +00001016 synop, desc = splitdoc(getdoc(object))
1017 result = self.section('NAME', name + (synop and ' - ' + synop))
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001018
1019 try:
Skip Montanaroa5616d22004-06-11 04:46:12 +00001020 all = object.__all__
1021 except AttributeError:
1022 all = None
1023
1024 try:
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001025 file = inspect.getabsfile(object)
1026 except TypeError:
1027 file = '(built-in)'
Ka-Ping Yee239432a2001-03-02 02:45:08 +00001028 result = result + self.section('FILE', file)
Skip Montanaro4997a692003-09-10 16:47:51 +00001029
1030 docloc = self.getdocloc(object)
1031 if docloc is not None:
1032 result = result + self.section('MODULE DOCS', docloc)
1033
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +00001034 if desc:
1035 result = result + self.section('DESCRIPTION', desc)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001036
1037 classes = []
1038 for key, value in inspect.getmembers(object, inspect.isclass):
Johannes Gijsbers4c11f602004-08-30 14:13:04 +00001039 # if __all__ exists, believe it. Otherwise use old heuristic.
1040 if (all is not None
1041 or (inspect.getmodule(value) or object) is object):
Skip Montanaroa5616d22004-06-11 04:46:12 +00001042 if visiblename(key, all):
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +00001043 classes.append((key, value))
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001044 funcs = []
1045 for key, value in inspect.getmembers(object, inspect.isroutine):
Johannes Gijsbers4c11f602004-08-30 14:13:04 +00001046 # if __all__ exists, believe it. Otherwise use old heuristic.
1047 if (all is not None or
1048 inspect.isbuiltin(value) or inspect.getmodule(value) is object):
Skip Montanaroa5616d22004-06-11 04:46:12 +00001049 if visiblename(key, all):
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +00001050 funcs.append((key, value))
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001051 data = []
1052 for key, value in inspect.getmembers(object, isdata):
Skip Montanaroa5616d22004-06-11 04:46:12 +00001053 if visiblename(key, all):
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001054 data.append((key, value))
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001055
1056 if hasattr(object, '__path__'):
1057 modpkgs = []
1058 for file in os.listdir(object.__path__[0]):
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +00001059 path = os.path.join(object.__path__[0], file)
1060 modname = inspect.getmodulename(file)
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +00001061 if modname != '__init__':
1062 if modname and modname not in modpkgs:
1063 modpkgs.append(modname)
1064 elif ispackage(path):
1065 modpkgs.append(file + ' (package)')
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001066 modpkgs.sort()
1067 result = result + self.section(
1068 'PACKAGE CONTENTS', join(modpkgs, '\n'))
1069
1070 if classes:
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001071 classlist = map(lambda (key, value): value, classes)
1072 contents = [self.formattree(
1073 inspect.getclasstree(classlist, 1), name)]
1074 for key, value in classes:
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001075 contents.append(self.document(value, key, name))
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001076 result = result + self.section('CLASSES', join(contents, '\n'))
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001077
1078 if funcs:
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001079 contents = []
1080 for key, value in funcs:
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001081 contents.append(self.document(value, key, name))
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001082 result = result + self.section('FUNCTIONS', join(contents, '\n'))
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001083
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001084 if data:
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001085 contents = []
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001086 for key, value in data:
Georg Brandl8b813db2005-10-01 16:32:31 +00001087 contents.append(self.docother(value, key, name, maxlen=70))
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001088 result = result + self.section('DATA', join(contents, '\n'))
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001089
1090 if hasattr(object, '__version__'):
1091 version = str(object.__version__)
Ka-Ping Yee1d384632001-03-01 00:24:32 +00001092 if version[:11] == '$' + 'Revision: ' and version[-1:] == '$':
1093 version = strip(version[11:-1])
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001094 result = result + self.section('VERSION', version)
Ka-Ping Yee6f3f9a42001-02-27 22:42:36 +00001095 if hasattr(object, '__date__'):
1096 result = result + self.section('DATE', str(object.__date__))
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001097 if hasattr(object, '__author__'):
Ka-Ping Yee6f3f9a42001-02-27 22:42:36 +00001098 result = result + self.section('AUTHOR', str(object.__author__))
1099 if hasattr(object, '__credits__'):
1100 result = result + self.section('CREDITS', str(object.__credits__))
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001101 return result
1102
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001103 def docclass(self, object, name=None, mod=None):
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001104 """Produce text documentation for a given class object."""
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001105 realname = object.__name__
1106 name = name or realname
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001107 bases = object.__bases__
1108
Tim Petersc86f6ca2001-09-26 21:31:51 +00001109 def makename(c, m=object.__module__):
1110 return classname(c, m)
1111
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001112 if name == realname:
1113 title = 'class ' + self.bold(realname)
1114 else:
1115 title = self.bold(name) + ' = class ' + realname
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001116 if bases:
Ka-Ping Yee3bda8792001-03-23 13:17:50 +00001117 parents = map(makename, bases)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001118 title = title + '(%s)' % join(parents, ', ')
1119
1120 doc = getdoc(object)
Tim Peters28355492001-09-23 21:29:55 +00001121 contents = doc and [doc + '\n'] or []
1122 push = contents.append
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001123
Tim Petersc86f6ca2001-09-26 21:31:51 +00001124 # List the mro, if non-trivial.
Raymond Hettinger756b3f32004-01-29 06:37:52 +00001125 mro = deque(inspect.getmro(object))
Tim Petersc86f6ca2001-09-26 21:31:51 +00001126 if len(mro) > 2:
1127 push("Method resolution order:")
1128 for base in mro:
1129 push(' ' + makename(base))
1130 push('')
1131
Tim Petersf4aad8e2001-09-24 22:40:47 +00001132 # Cute little class to pump out a horizontal rule between sections.
1133 class HorizontalRule:
1134 def __init__(self):
1135 self.needone = 0
1136 def maybe(self):
1137 if self.needone:
1138 push('-' * 70)
1139 self.needone = 1
1140 hr = HorizontalRule()
1141
Tim Peters28355492001-09-23 21:29:55 +00001142 def spill(msg, attrs, predicate):
Tim Petersfa26f7c2001-09-24 08:05:11 +00001143 ok, attrs = _split_list(attrs, predicate)
Tim Peters28355492001-09-23 21:29:55 +00001144 if ok:
Tim Petersf4aad8e2001-09-24 22:40:47 +00001145 hr.maybe()
Tim Peters28355492001-09-23 21:29:55 +00001146 push(msg)
1147 for name, kind, homecls, value in ok:
1148 push(self.document(getattr(object, name),
1149 name, mod, object))
1150 return attrs
1151
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +00001152 def spilldescriptors(msg, attrs, predicate):
Tim Petersfa26f7c2001-09-24 08:05:11 +00001153 ok, attrs = _split_list(attrs, predicate)
Tim Peters28355492001-09-23 21:29:55 +00001154 if ok:
Tim Petersf4aad8e2001-09-24 22:40:47 +00001155 hr.maybe()
Tim Peters28355492001-09-23 21:29:55 +00001156 push(msg)
1157 for name, kind, homecls, value in ok:
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +00001158 push(self._docdescriptor(name, value, mod))
Tim Peters28355492001-09-23 21:29:55 +00001159 return attrs
Tim Petersb47879b2001-09-24 04:47:19 +00001160
Tim Petersfa26f7c2001-09-24 08:05:11 +00001161 def spilldata(msg, attrs, predicate):
1162 ok, attrs = _split_list(attrs, predicate)
Tim Peters28355492001-09-23 21:29:55 +00001163 if ok:
Tim Petersf4aad8e2001-09-24 22:40:47 +00001164 hr.maybe()
Tim Peters28355492001-09-23 21:29:55 +00001165 push(msg)
1166 for name, kind, homecls, value in ok:
Martin v. Löwise59e2ba2003-05-03 09:09:02 +00001167 if callable(value) or inspect.isdatadescriptor(value):
Ka-Ping Yeebba6acc2005-02-19 22:58:26 +00001168 doc = getdoc(value)
Guido van Rossum5e355b22002-05-21 20:56:15 +00001169 else:
1170 doc = None
Tim Peters28355492001-09-23 21:29:55 +00001171 push(self.docother(getattr(object, name),
Georg Brandl8b813db2005-10-01 16:32:31 +00001172 name, mod, maxlen=70, doc=doc) + '\n')
Tim Peters28355492001-09-23 21:29:55 +00001173 return attrs
1174
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +00001175 attrs = filter(lambda (name, kind, cls, value): visiblename(name),
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +00001176 classify_class_attrs(object))
Tim Petersfa26f7c2001-09-24 08:05:11 +00001177 while attrs:
Tim Peters351e3622001-09-27 03:29:51 +00001178 if mro:
Raymond Hettinger756b3f32004-01-29 06:37:52 +00001179 thisclass = mro.popleft()
Tim Peters351e3622001-09-27 03:29:51 +00001180 else:
1181 thisclass = attrs[0][2]
Tim Petersfa26f7c2001-09-24 08:05:11 +00001182 attrs, inherited = _split_list(attrs, lambda t: t[2] is thisclass)
1183
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +00001184 if thisclass is __builtin__.object:
1185 attrs = inherited
1186 continue
1187 elif thisclass is object:
Tim Peters28355492001-09-23 21:29:55 +00001188 tag = "defined here"
1189 else:
Tim Petersfa26f7c2001-09-24 08:05:11 +00001190 tag = "inherited from %s" % classname(thisclass,
1191 object.__module__)
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +00001192 filter(lambda t: not t[0].startswith('_'), attrs)
Tim Peters28355492001-09-23 21:29:55 +00001193
1194 # Sort attrs by name.
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +00001195 attrs.sort()
Tim Peters28355492001-09-23 21:29:55 +00001196
1197 # Pump out the attrs, segregated by kind.
Tim Petersf4aad8e2001-09-24 22:40:47 +00001198 attrs = spill("Methods %s:\n" % tag, attrs,
Tim Peters28355492001-09-23 21:29:55 +00001199 lambda t: t[1] == 'method')
Tim Petersf4aad8e2001-09-24 22:40:47 +00001200 attrs = spill("Class methods %s:\n" % tag, attrs,
Tim Peters28355492001-09-23 21:29:55 +00001201 lambda t: t[1] == 'class method')
Tim Petersf4aad8e2001-09-24 22:40:47 +00001202 attrs = spill("Static methods %s:\n" % tag, attrs,
Tim Peters28355492001-09-23 21:29:55 +00001203 lambda t: t[1] == 'static method')
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +00001204 attrs = spilldescriptors("Data descriptors %s:\n" % tag, attrs,
1205 lambda t: t[1] == 'data descriptor')
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +00001206 attrs = spilldata("Data and other attributes %s:\n" % tag, attrs,
1207 lambda t: t[1] == 'data')
Tim Peters28355492001-09-23 21:29:55 +00001208 assert attrs == []
Tim Peters351e3622001-09-27 03:29:51 +00001209 attrs = inherited
Tim Peters28355492001-09-23 21:29:55 +00001210
1211 contents = '\n'.join(contents)
1212 if not contents:
1213 return title + '\n'
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001214 return title + '\n' + self.indent(rstrip(contents), ' | ') + '\n'
1215
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001216 def formatvalue(self, object):
1217 """Format an argument default value as text."""
1218 return '=' + self.repr(object)
1219
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001220 def docroutine(self, object, name=None, mod=None, cl=None):
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001221 """Produce text documentation for a function or method object."""
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001222 realname = object.__name__
1223 name = name or realname
1224 note = ''
Ka-Ping Yee3bda8792001-03-23 13:17:50 +00001225 skipdocs = 0
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001226 if inspect.ismethod(object):
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001227 imclass = object.im_class
Ka-Ping Yee3bda8792001-03-23 13:17:50 +00001228 if cl:
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001229 if imclass is not cl:
1230 note = ' from ' + classname(imclass, mod)
Ka-Ping Yee3bda8792001-03-23 13:17:50 +00001231 else:
Ka-Ping Yeeb7a48302001-04-12 20:39:14 +00001232 if object.im_self:
1233 note = ' method of %s instance' % classname(
1234 object.im_self.__class__, mod)
1235 else:
1236 note = ' unbound %s method' % classname(imclass,mod)
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001237 object = object.im_func
1238
1239 if name == realname:
1240 title = self.bold(realname)
1241 else:
Raymond Hettinger54f02222002-06-01 14:18:47 +00001242 if (cl and realname in cl.__dict__ and
Ka-Ping Yee3bda8792001-03-23 13:17:50 +00001243 cl.__dict__[realname] is object):
1244 skipdocs = 1
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001245 title = self.bold(name) + ' = ' + realname
Tim Peters4bcfa312001-09-20 06:08:24 +00001246 if inspect.isfunction(object):
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001247 args, varargs, varkw, defaults = inspect.getargspec(object)
1248 argspec = inspect.formatargspec(
1249 args, varargs, varkw, defaults, formatvalue=self.formatvalue)
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001250 if realname == '<lambda>':
1251 title = 'lambda'
1252 argspec = argspec[1:-1] # remove parentheses
Tim Peters4bcfa312001-09-20 06:08:24 +00001253 else:
1254 argspec = '(...)'
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001255 decl = title + argspec + note
1256
Ka-Ping Yee3bda8792001-03-23 13:17:50 +00001257 if skipdocs:
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001258 return decl + '\n'
Ka-Ping Yee3bda8792001-03-23 13:17:50 +00001259 else:
1260 doc = getdoc(object) or ''
1261 return decl + '\n' + (doc and rstrip(self.indent(doc)) + '\n')
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001262
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +00001263 def _docdescriptor(self, name, value, mod):
Johannes Gijsbers8de645a2004-11-07 19:16:05 +00001264 results = []
1265 push = results.append
1266
1267 if name:
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +00001268 push(self.bold(name))
1269 push('\n')
Johannes Gijsbers8de645a2004-11-07 19:16:05 +00001270 doc = getdoc(value) or ''
1271 if doc:
1272 push(self.indent(doc))
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +00001273 push('\n')
1274 return ''.join(results)
Johannes Gijsbers8de645a2004-11-07 19:16:05 +00001275
1276 def docproperty(self, object, name=None, mod=None, cl=None):
1277 """Produce text documentation for a property."""
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +00001278 return self._docdescriptor(name, object, mod)
Johannes Gijsbers8de645a2004-11-07 19:16:05 +00001279
Georg Brandl8b813db2005-10-01 16:32:31 +00001280 def docother(self, object, name=None, mod=None, parent=None, maxlen=None, doc=None):
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001281 """Produce text documentation for a data object."""
1282 repr = self.repr(object)
1283 if maxlen:
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001284 line = (name and name + ' = ' or '') + repr
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001285 chop = maxlen - len(line)
1286 if chop < 0: repr = repr[:chop] + '...'
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001287 line = (name and self.bold(name) + ' = ' or '') + repr
Tim Peters28355492001-09-23 21:29:55 +00001288 if doc is not None:
1289 line += '\n' + self.indent(str(doc))
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001290 return line
1291
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001292# --------------------------------------------------------- user interfaces
1293
1294def pager(text):
1295 """The first time this is called, determine what kind of pager to use."""
1296 global pager
1297 pager = getpager()
1298 pager(text)
1299
1300def getpager():
1301 """Decide what method to use for paging through text."""
1302 if type(sys.stdout) is not types.FileType:
1303 return plainpager
1304 if not sys.stdin.isatty() or not sys.stdout.isatty():
1305 return plainpager
Raymond Hettinger54f02222002-06-01 14:18:47 +00001306 if 'PAGER' in os.environ:
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +00001307 if sys.platform == 'win32': # pipes completely broken in Windows
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001308 return lambda text: tempfilepager(plain(text), os.environ['PAGER'])
Raymond Hettingerdbecd932005-02-06 06:57:08 +00001309 elif os.environ.get('TERM') in ('dumb', 'emacs'):
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001310 return lambda text: pipepager(plain(text), os.environ['PAGER'])
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +00001311 else:
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001312 return lambda text: pipepager(text, os.environ['PAGER'])
Ka-Ping Yeea487e4e2005-11-05 04:49:18 +00001313 if os.environ.get('TERM') in ('dumb', 'emacs'):
1314 return plainpager
Andrew MacIntyre54e0eab2002-03-03 03:12:30 +00001315 if sys.platform == 'win32' or sys.platform.startswith('os2'):
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001316 return lambda text: tempfilepager(plain(text), 'more <')
Skip Montanarod404bee2002-09-26 21:44:57 +00001317 if hasattr(os, 'system') and os.system('(less) 2>/dev/null') == 0:
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001318 return lambda text: pipepager(text, 'less')
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001319
1320 import tempfile
Guido van Rossum3b0a3292002-08-09 16:38:32 +00001321 (fd, filename) = tempfile.mkstemp()
1322 os.close(fd)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001323 try:
1324 if hasattr(os, 'system') and os.system('more %s' % filename) == 0:
1325 return lambda text: pipepager(text, 'more')
1326 else:
1327 return ttypager
1328 finally:
1329 os.unlink(filename)
1330
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001331def plain(text):
1332 """Remove boldface formatting from text."""
1333 return re.sub('.\b', '', text)
1334
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001335def pipepager(text, cmd):
1336 """Page through text by feeding it to another program."""
1337 pipe = os.popen(cmd, 'w')
1338 try:
1339 pipe.write(text)
1340 pipe.close()
1341 except IOError:
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001342 pass # Ignore broken pipes caused by quitting the pager program.
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001343
1344def tempfilepager(text, cmd):
1345 """Page through text by invoking a program on a temporary file."""
1346 import tempfile
Tim Peters550e4e52003-02-07 01:53:46 +00001347 filename = tempfile.mktemp()
1348 file = open(filename, 'w')
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001349 file.write(text)
1350 file.close()
1351 try:
Ka-Ping Yeec92cdf72001-03-02 05:54:35 +00001352 os.system(cmd + ' ' + filename)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001353 finally:
1354 os.unlink(filename)
1355
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001356def ttypager(text):
1357 """Page through text on a text terminal."""
1358 lines = split(plain(text), '\n')
1359 try:
1360 import tty
1361 fd = sys.stdin.fileno()
1362 old = tty.tcgetattr(fd)
1363 tty.setcbreak(fd)
1364 getchar = lambda: sys.stdin.read(1)
Ka-Ping Yee457aab22001-02-27 23:36:29 +00001365 except (ImportError, AttributeError):
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001366 tty = None
1367 getchar = lambda: sys.stdin.readline()[:-1][:1]
1368
1369 try:
1370 r = inc = os.environ.get('LINES', 25) - 1
1371 sys.stdout.write(join(lines[:inc], '\n') + '\n')
1372 while lines[r:]:
1373 sys.stdout.write('-- more --')
1374 sys.stdout.flush()
1375 c = getchar()
1376
Raymond Hettingerdbecd932005-02-06 06:57:08 +00001377 if c in ('q', 'Q'):
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001378 sys.stdout.write('\r \r')
1379 break
Raymond Hettingerdbecd932005-02-06 06:57:08 +00001380 elif c in ('\r', '\n'):
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001381 sys.stdout.write('\r \r' + lines[r] + '\n')
1382 r = r + 1
1383 continue
Raymond Hettingerdbecd932005-02-06 06:57:08 +00001384 if c in ('b', 'B', '\x1b'):
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001385 r = r - inc - inc
1386 if r < 0: r = 0
1387 sys.stdout.write('\n' + join(lines[r:r+inc], '\n') + '\n')
1388 r = r + inc
1389
1390 finally:
1391 if tty:
1392 tty.tcsetattr(fd, tty.TCSAFLUSH, old)
1393
1394def plainpager(text):
1395 """Simply print unformatted text. This is the ultimate fallback."""
1396 sys.stdout.write(plain(text))
1397
1398def describe(thing):
Ka-Ping Yee3bda8792001-03-23 13:17:50 +00001399 """Produce a short description of the given thing."""
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001400 if inspect.ismodule(thing):
1401 if thing.__name__ in sys.builtin_module_names:
1402 return 'built-in module ' + thing.__name__
1403 if hasattr(thing, '__path__'):
1404 return 'package ' + thing.__name__
1405 else:
1406 return 'module ' + thing.__name__
1407 if inspect.isbuiltin(thing):
1408 return 'built-in function ' + thing.__name__
1409 if inspect.isclass(thing):
1410 return 'class ' + thing.__name__
1411 if inspect.isfunction(thing):
1412 return 'function ' + thing.__name__
1413 if inspect.ismethod(thing):
1414 return 'method ' + thing.__name__
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001415 if type(thing) is types.InstanceType:
1416 return 'instance of ' + thing.__class__.__name__
1417 return type(thing).__name__
1418
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001419def locate(path, forceload=0):
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001420 """Locate an object by name or dotted path, importing as necessary."""
Guido van Rossum97dede02003-02-16 01:12:32 +00001421 parts = [part for part in split(path, '.') if part]
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001422 module, n = None, 0
1423 while n < len(parts):
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001424 nextmodule = safeimport(join(parts[:n+1], '.'), forceload)
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001425 if nextmodule: module, n = nextmodule, n + 1
1426 else: break
1427 if module:
1428 object = module
1429 for part in parts[n:]:
1430 try: object = getattr(object, part)
1431 except AttributeError: return None
1432 return object
1433 else:
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001434 if hasattr(__builtin__, path):
1435 return getattr(__builtin__, path)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001436
1437# --------------------------------------- interactive interpreter interface
1438
1439text = TextDoc()
1440html = HTMLDoc()
1441
Ka-Ping Yee45daeb02002-08-11 15:11:33 +00001442def resolve(thing, forceload=0):
1443 """Given an object or a path to an object, get the object and its name."""
1444 if isinstance(thing, str):
1445 object = locate(thing, forceload)
1446 if not object:
1447 raise ImportError, 'no Python documentation found for %r' % thing
1448 return object, thing
1449 else:
1450 return thing, getattr(thing, '__name__', None)
1451
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001452def doc(thing, title='Python Library Documentation: %s', forceload=0):
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001453 """Display text documentation, given an object or a path to an object."""
Ka-Ping Yee45daeb02002-08-11 15:11:33 +00001454 try:
1455 object, name = resolve(thing, forceload)
1456 desc = describe(object)
1457 module = inspect.getmodule(object)
1458 if name and '.' in name:
1459 desc += ' in ' + name[:name.rfind('.')]
1460 elif module and module is not object:
1461 desc += ' in module ' + module.__name__
Ka-Ping Yeebba6acc2005-02-19 22:58:26 +00001462 if not (inspect.ismodule(object) or
1463 inspect.isclass(object) or
1464 inspect.isroutine(object) or
1465 isinstance(object, property)):
1466 # If the passed object is a piece of data or an instance,
1467 # document its available methods instead of its value.
1468 object = type(object)
1469 desc += ' object'
Ka-Ping Yee45daeb02002-08-11 15:11:33 +00001470 pager(title % desc + '\n\n' + text.document(object, name))
1471 except (ImportError, ErrorDuringImport), value:
1472 print value
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001473
Ka-Ping Yee45daeb02002-08-11 15:11:33 +00001474def writedoc(thing, forceload=0):
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001475 """Write HTML documentation to a file in the current directory."""
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001476 try:
Ka-Ping Yee45daeb02002-08-11 15:11:33 +00001477 object, name = resolve(thing, forceload)
1478 page = html.page(describe(object), html.document(object, name))
1479 file = open(name + '.html', 'w')
1480 file.write(page)
1481 file.close()
1482 print 'wrote', name + '.html'
1483 except (ImportError, ErrorDuringImport), value:
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001484 print value
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001485
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001486def writedocs(dir, pkgpath='', done=None):
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001487 """Write out HTML documentation for all modules in a directory tree."""
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001488 if done is None: done = {}
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001489 for file in os.listdir(dir):
1490 path = os.path.join(dir, file)
1491 if ispackage(path):
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001492 writedocs(path, pkgpath + file + '.', done)
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001493 elif os.path.isfile(path):
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +00001494 modname = inspect.getmodulename(path)
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001495 if modname:
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +00001496 if modname == '__init__':
1497 modname = pkgpath[:-1] # remove trailing period
1498 else:
1499 modname = pkgpath + modname
1500 if modname not in done:
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001501 done[modname] = 1
1502 writedoc(modname)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001503
1504class Helper:
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001505 keywords = {
1506 'and': 'BOOLEAN',
Ka-Ping Yee79c009d2001-04-13 10:53:25 +00001507 'assert': ('ref/assert', ''),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001508 'break': ('ref/break', 'while for'),
1509 'class': ('ref/class', 'CLASSES SPECIALMETHODS'),
1510 'continue': ('ref/continue', 'while for'),
1511 'def': ('ref/function', ''),
1512 'del': ('ref/del', 'BASICMETHODS'),
1513 'elif': 'if',
1514 'else': ('ref/if', 'while for'),
1515 'except': 'try',
1516 'exec': ('ref/exec', ''),
1517 'finally': 'try',
1518 'for': ('ref/for', 'break continue while'),
1519 'from': 'import',
1520 'global': ('ref/global', 'NAMESPACES'),
1521 'if': ('ref/if', 'TRUTHVALUE'),
1522 'import': ('ref/import', 'MODULES'),
1523 'in': ('ref/comparisons', 'SEQUENCEMETHODS2'),
1524 'is': 'COMPARISON',
Neal Norwitz742dde42003-03-30 20:31:34 +00001525 'lambda': ('ref/lambdas', 'FUNCTIONS'),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001526 'not': 'BOOLEAN',
1527 'or': 'BOOLEAN',
Neal Norwitz742dde42003-03-30 20:31:34 +00001528 'pass': ('ref/pass', ''),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001529 'print': ('ref/print', ''),
1530 'raise': ('ref/raise', 'EXCEPTIONS'),
Ka-Ping Yee79c009d2001-04-13 10:53:25 +00001531 'return': ('ref/return', 'FUNCTIONS'),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001532 'try': ('ref/try', 'EXCEPTIONS'),
1533 'while': ('ref/while', 'break continue if TRUTHVALUE'),
Tim Petersfb05c4e2002-10-30 05:21:00 +00001534 'yield': ('ref/yield', ''),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001535 }
1536
1537 topics = {
1538 'TYPES': ('ref/types', 'STRINGS UNICODE NUMBERS SEQUENCES MAPPINGS FUNCTIONS CLASSES MODULES FILES inspect'),
Ka-Ping Yee79c009d2001-04-13 10:53:25 +00001539 'STRINGS': ('ref/strings', 'str UNICODE SEQUENCES STRINGMETHODS FORMATTING TYPES'),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001540 'STRINGMETHODS': ('lib/string-methods', 'STRINGS FORMATTING'),
1541 'FORMATTING': ('lib/typesseq-strings', 'OPERATORS'),
Neal Norwitz54f871e2003-05-26 13:49:54 +00001542 'UNICODE': ('ref/strings', 'encodings unicode SEQUENCES STRINGMETHODS FORMATTING TYPES'),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001543 'NUMBERS': ('ref/numbers', 'INTEGER FLOAT COMPLEX TYPES'),
1544 'INTEGER': ('ref/integers', 'int range'),
1545 'FLOAT': ('ref/floating', 'float math'),
1546 'COMPLEX': ('ref/imaginary', 'complex cmath'),
Ka-Ping Yee79c009d2001-04-13 10:53:25 +00001547 'SEQUENCES': ('lib/typesseq', 'STRINGMETHODS FORMATTING xrange LISTS'),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001548 'MAPPINGS': 'DICTIONARIES',
1549 'FUNCTIONS': ('lib/typesfunctions', 'def TYPES'),
1550 'METHODS': ('lib/typesmethods', 'class def CLASSES TYPES'),
1551 'CODEOBJECTS': ('lib/bltin-code-objects', 'compile FUNCTIONS TYPES'),
Ka-Ping Yee79c009d2001-04-13 10:53:25 +00001552 'TYPEOBJECTS': ('lib/bltin-type-objects', 'types TYPES'),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001553 'FRAMEOBJECTS': 'TYPES',
1554 'TRACEBACKS': 'TYPES',
1555 'NONE': ('lib/bltin-null-object', ''),
1556 'ELLIPSIS': ('lib/bltin-ellipsis-object', 'SLICINGS'),
1557 'FILES': ('lib/bltin-file-objects', ''),
1558 'SPECIALATTRIBUTES': ('lib/specialattrs', ''),
1559 'CLASSES': ('ref/types', 'class SPECIALMETHODS PRIVATENAMES'),
1560 'MODULES': ('lib/typesmodules', 'import'),
1561 'PACKAGES': 'import',
1562 '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'),
1563 'OPERATORS': 'EXPRESSIONS',
1564 'PRECEDENCE': 'EXPRESSIONS',
1565 'OBJECTS': ('ref/objects', 'TYPES'),
1566 'SPECIALMETHODS': ('ref/specialnames', 'BASICMETHODS ATTRIBUTEMETHODS CALLABLEMETHODS SEQUENCEMETHODS1 MAPPINGMETHODS SEQUENCEMETHODS2 NUMBERMETHODS CLASSES'),
Ka-Ping Yee79c009d2001-04-13 10:53:25 +00001567 'BASICMETHODS': ('ref/customization', 'cmp hash repr str SPECIALMETHODS'),
1568 'ATTRIBUTEMETHODS': ('ref/attribute-access', 'ATTRIBUTES SPECIALMETHODS'),
1569 'CALLABLEMETHODS': ('ref/callable-types', 'CALLS SPECIALMETHODS'),
1570 'SEQUENCEMETHODS1': ('ref/sequence-types', 'SEQUENCES SEQUENCEMETHODS2 SPECIALMETHODS'),
1571 'SEQUENCEMETHODS2': ('ref/sequence-methods', 'SEQUENCES SEQUENCEMETHODS1 SPECIALMETHODS'),
1572 'MAPPINGMETHODS': ('ref/sequence-types', 'MAPPINGS SPECIALMETHODS'),
1573 'NUMBERMETHODS': ('ref/numeric-types', 'NUMBERS AUGMENTEDASSIGNMENT SPECIALMETHODS'),
Neal Norwitz54f871e2003-05-26 13:49:54 +00001574 'EXECUTION': ('ref/execmodel', 'NAMESPACES DYNAMICFEATURES EXCEPTIONS'),
1575 'NAMESPACES': ('ref/naming', 'global ASSIGNMENT DELETION DYNAMICFEATURES'),
1576 'DYNAMICFEATURES': ('ref/dynamic-features', ''),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001577 'SCOPING': 'NAMESPACES',
1578 'FRAMES': 'NAMESPACES',
1579 'EXCEPTIONS': ('ref/exceptions', 'try except finally raise'),
Neal Norwitz54f871e2003-05-26 13:49:54 +00001580 'COERCIONS': ('ref/coercion-rules','CONVERSIONS'),
1581 'CONVERSIONS': ('ref/conversions', 'COERCIONS'),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001582 'IDENTIFIERS': ('ref/identifiers', 'keywords SPECIALIDENTIFIERS'),
1583 'SPECIALIDENTIFIERS': ('ref/id-classes', ''),
Ka-Ping Yee79c009d2001-04-13 10:53:25 +00001584 'PRIVATENAMES': ('ref/atom-identifiers', ''),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001585 'LITERALS': ('ref/atom-literals', 'STRINGS BACKQUOTES NUMBERS TUPLELITERALS LISTLITERALS DICTIONARYLITERALS'),
1586 'TUPLES': 'SEQUENCES',
Ka-Ping Yee79c009d2001-04-13 10:53:25 +00001587 'TUPLELITERALS': ('ref/exprlists', 'TUPLES LITERALS'),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001588 'LISTS': ('lib/typesseq-mutable', 'LISTLITERALS'),
Ka-Ping Yee79c009d2001-04-13 10:53:25 +00001589 'LISTLITERALS': ('ref/lists', 'LISTS LITERALS'),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001590 'DICTIONARIES': ('lib/typesmapping', 'DICTIONARYLITERALS'),
Ka-Ping Yee79c009d2001-04-13 10:53:25 +00001591 'DICTIONARYLITERALS': ('ref/dict', 'DICTIONARIES LITERALS'),
1592 'BACKQUOTES': ('ref/string-conversions', 'repr str STRINGS LITERALS'),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001593 'ATTRIBUTES': ('ref/attribute-references', 'getattr hasattr setattr ATTRIBUTEMETHODS'),
1594 'SUBSCRIPTS': ('ref/subscriptions', 'SEQUENCEMETHODS1'),
1595 'SLICINGS': ('ref/slicings', 'SEQUENCEMETHODS2'),
1596 'CALLS': ('ref/calls', 'EXPRESSIONS'),
1597 'POWER': ('ref/power', 'EXPRESSIONS'),
1598 'UNARY': ('ref/unary', 'EXPRESSIONS'),
1599 'BINARY': ('ref/binary', 'EXPRESSIONS'),
1600 'SHIFTING': ('ref/shifting', 'EXPRESSIONS'),
1601 'BITWISE': ('ref/bitwise', 'EXPRESSIONS'),
1602 'COMPARISON': ('ref/comparisons', 'EXPRESSIONS BASICMETHODS'),
Neal Norwitzf98159c2003-02-07 20:49:40 +00001603 'BOOLEAN': ('ref/Booleans', 'EXPRESSIONS TRUTHVALUE'),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001604 'ASSERTION': 'assert',
1605 'ASSIGNMENT': ('ref/assignment', 'AUGMENTEDASSIGNMENT'),
Ka-Ping Yee79c009d2001-04-13 10:53:25 +00001606 'AUGMENTEDASSIGNMENT': ('ref/augassign', 'NUMBERMETHODS'),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001607 'DELETION': 'del',
1608 'PRINTING': 'print',
1609 'RETURNING': 'return',
1610 'IMPORTING': 'import',
1611 'CONDITIONAL': 'if',
1612 'LOOPING': ('ref/compound', 'for while break continue'),
1613 'TRUTHVALUE': ('lib/truth', 'if while and or not BASICMETHODS'),
Ka-Ping Yee79c009d2001-04-13 10:53:25 +00001614 'DEBUGGING': ('lib/module-pdb', 'pdb'),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001615 }
1616
1617 def __init__(self, input, output):
1618 self.input = input
1619 self.output = output
1620 self.docdir = None
1621 execdir = os.path.dirname(sys.executable)
1622 homedir = os.environ.get('PYTHONHOME')
1623 for dir in [os.environ.get('PYTHONDOCS'),
1624 homedir and os.path.join(homedir, 'doc'),
1625 os.path.join(execdir, 'doc'),
1626 '/usr/doc/python-docs-' + split(sys.version)[0],
1627 '/usr/doc/python-' + split(sys.version)[0],
1628 '/usr/doc/python-docs-' + sys.version[:3],
Jack Jansenb2628b02002-08-23 08:40:42 +00001629 '/usr/doc/python-' + sys.version[:3],
1630 os.path.join(sys.prefix, 'Resources/English.lproj/Documentation')]:
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001631 if dir and os.path.isdir(os.path.join(dir, 'lib')):
1632 self.docdir = dir
1633
Ka-Ping Yee79c009d2001-04-13 10:53:25 +00001634 def __repr__(self):
Ka-Ping Yee9bc576b2001-04-13 13:57:31 +00001635 if inspect.stack()[1][3] == '?':
Ka-Ping Yee79c009d2001-04-13 10:53:25 +00001636 self()
1637 return ''
Ka-Ping Yee9bc576b2001-04-13 13:57:31 +00001638 return '<pydoc.Helper instance>'
Ka-Ping Yee79c009d2001-04-13 10:53:25 +00001639
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001640 def __call__(self, request=None):
1641 if request is not None:
1642 self.help(request)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001643 else:
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001644 self.intro()
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001645 self.interact()
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001646 self.output.write('''
Fred Drakee61967f2001-05-10 18:41:02 +00001647You are now leaving help and returning to the Python interpreter.
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001648If you want to ask for help on a particular object directly from the
1649interpreter, you can type "help(object)". Executing "help('string')"
1650has the same effect as typing a particular string at the help> prompt.
1651''')
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001652
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001653 def interact(self):
1654 self.output.write('\n')
Guido van Rossum8ca162f2002-04-07 06:36:23 +00001655 while True:
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001656 try:
Johannes Gijsberse7691d32004-08-17 13:21:53 +00001657 request = self.getline('help> ')
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001658 if not request: break
Johannes Gijsberse7691d32004-08-17 13:21:53 +00001659 except (KeyboardInterrupt, EOFError):
1660 break
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001661 request = strip(replace(request, '"', '', "'", ''))
Raymond Hettingerdbecd932005-02-06 06:57:08 +00001662 if lower(request) in ('q', 'quit'): break
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001663 self.help(request)
1664
Johannes Gijsberse7691d32004-08-17 13:21:53 +00001665 def getline(self, prompt):
1666 """Read one line, using raw_input when available."""
1667 if self.input is sys.stdin:
1668 return raw_input(prompt)
1669 else:
1670 self.output.write(prompt)
1671 self.output.flush()
1672 return self.input.readline()
1673
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001674 def help(self, request):
1675 if type(request) is type(''):
1676 if request == 'help': self.intro()
1677 elif request == 'keywords': self.listkeywords()
1678 elif request == 'topics': self.listtopics()
1679 elif request == 'modules': self.listmodules()
1680 elif request[:8] == 'modules ':
1681 self.listmodules(split(request)[1])
Raymond Hettinger54f02222002-06-01 14:18:47 +00001682 elif request in self.keywords: self.showtopic(request)
1683 elif request in self.topics: self.showtopic(request)
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001684 elif request: doc(request, 'Help on %s:')
1685 elif isinstance(request, Helper): self()
1686 else: doc(request, 'Help on %s:')
1687 self.output.write('\n')
1688
1689 def intro(self):
1690 self.output.write('''
1691Welcome to Python %s! This is the online help utility.
1692
1693If this is your first time using Python, you should definitely check out
1694the tutorial on the Internet at http://www.python.org/doc/tut/.
1695
1696Enter the name of any module, keyword, or topic to get help on writing
1697Python programs and using Python modules. To quit this help utility and
1698return to the interpreter, just type "quit".
1699
1700To get a list of available modules, keywords, or topics, type "modules",
1701"keywords", or "topics". Each module also comes with a one-line summary
1702of what it does; to list the modules whose summaries contain a given word
1703such as "spam", type "modules spam".
1704''' % sys.version[:3])
1705
1706 def list(self, items, columns=4, width=80):
1707 items = items[:]
1708 items.sort()
1709 colw = width / columns
1710 rows = (len(items) + columns - 1) / columns
1711 for row in range(rows):
1712 for col in range(columns):
1713 i = col * rows + row
1714 if i < len(items):
1715 self.output.write(items[i])
1716 if col < columns - 1:
1717 self.output.write(' ' + ' ' * (colw-1 - len(items[i])))
1718 self.output.write('\n')
1719
1720 def listkeywords(self):
1721 self.output.write('''
1722Here is a list of the Python keywords. Enter any keyword to get more help.
1723
1724''')
1725 self.list(self.keywords.keys())
1726
1727 def listtopics(self):
1728 self.output.write('''
1729Here is a list of available topics. Enter any topic name to get more help.
1730
1731''')
1732 self.list(self.topics.keys())
1733
1734 def showtopic(self, topic):
1735 if not self.docdir:
1736 self.output.write('''
1737Sorry, topic and keyword documentation is not available because the Python
1738HTML documentation files could not be found. If you have installed them,
1739please set the environment variable PYTHONDOCS to indicate their location.
1740''')
1741 return
1742 target = self.topics.get(topic, self.keywords.get(topic))
1743 if not target:
1744 self.output.write('no documentation found for %s\n' % repr(topic))
1745 return
1746 if type(target) is type(''):
1747 return self.showtopic(target)
1748
1749 filename, xrefs = target
1750 filename = self.docdir + '/' + filename + '.html'
1751 try:
1752 file = open(filename)
1753 except:
1754 self.output.write('could not read docs from %s\n' % filename)
1755 return
1756
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001757 divpat = re.compile('<div[^>]*navigat.*?</div.*?>', re.I | re.S)
1758 addrpat = re.compile('<address.*?>.*?</address.*?>', re.I | re.S)
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001759 document = re.sub(addrpat, '', re.sub(divpat, '', file.read()))
1760 file.close()
1761
1762 import htmllib, formatter, StringIO
1763 buffer = StringIO.StringIO()
1764 parser = htmllib.HTMLParser(
1765 formatter.AbstractFormatter(formatter.DumbWriter(buffer)))
1766 parser.start_table = parser.do_p
1767 parser.end_table = lambda parser=parser: parser.do_p({})
1768 parser.start_tr = parser.do_br
1769 parser.start_td = parser.start_th = lambda a, b=buffer: b.write('\t')
1770 parser.feed(document)
1771 buffer = replace(buffer.getvalue(), '\xa0', ' ', '\n', '\n ')
1772 pager(' ' + strip(buffer) + '\n')
Ka-Ping Yeeda793892001-04-13 11:02:51 +00001773 if xrefs:
1774 buffer = StringIO.StringIO()
1775 formatter.DumbWriter(buffer).send_flowing_data(
1776 'Related help topics: ' + join(split(xrefs), ', ') + '\n')
1777 self.output.write('\n%s\n' % buffer.getvalue())
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001778
1779 def listmodules(self, key=''):
1780 if key:
1781 self.output.write('''
1782Here is a list of matching modules. Enter any module name to get more help.
1783
1784''')
1785 apropos(key)
1786 else:
1787 self.output.write('''
1788Please wait a moment while I gather a list of all available modules...
1789
1790''')
1791 modules = {}
1792 def callback(path, modname, desc, modules=modules):
1793 if modname and modname[-9:] == '.__init__':
1794 modname = modname[:-9] + ' (package)'
1795 if find(modname, '.') < 0:
1796 modules[modname] = 1
1797 ModuleScanner().run(callback)
1798 self.list(modules.keys())
1799 self.output.write('''
1800Enter any module name to get more help. Or, type "modules spam" to search
1801for modules whose descriptions contain the word "spam".
1802''')
1803
1804help = Helper(sys.stdin, sys.stdout)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001805
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001806class Scanner:
1807 """A generic tree iterator."""
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001808 def __init__(self, roots, children, descendp):
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001809 self.roots = roots[:]
1810 self.state = []
1811 self.children = children
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001812 self.descendp = descendp
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001813
1814 def next(self):
1815 if not self.state:
1816 if not self.roots:
1817 return None
1818 root = self.roots.pop(0)
1819 self.state = [(root, self.children(root))]
1820 node, children = self.state[-1]
1821 if not children:
1822 self.state.pop()
1823 return self.next()
1824 child = children.pop(0)
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001825 if self.descendp(child):
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001826 self.state.append((child, self.children(child)))
1827 return child
1828
1829class ModuleScanner(Scanner):
1830 """An interruptible scanner that searches module synopses."""
1831 def __init__(self):
1832 roots = map(lambda dir: (dir, ''), pathdirs())
Ka-Ping Yeeeca15c12001-04-13 13:53:07 +00001833 Scanner.__init__(self, roots, self.submodules, self.isnewpackage)
Raymond Hettinger32200ae2002-06-01 19:51:15 +00001834 self.inodes = map(lambda (dir, pkg): os.stat(dir).st_ino, roots)
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001835
1836 def submodules(self, (dir, package)):
1837 children = []
1838 for file in os.listdir(dir):
1839 path = os.path.join(dir, file)
Tim Peters30edd232001-03-16 08:29:48 +00001840 if ispackage(path):
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001841 children.append((path, package + (package and '.') + file))
1842 else:
1843 children.append((path, package))
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001844 children.sort() # so that spam.py comes before spam.pyc or spam.pyo
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001845 return children
1846
Ka-Ping Yeeeca15c12001-04-13 13:53:07 +00001847 def isnewpackage(self, (dir, package)):
Raymond Hettinger32200ae2002-06-01 19:51:15 +00001848 inode = os.path.exists(dir) and os.stat(dir).st_ino
Ka-Ping Yeeeca15c12001-04-13 13:53:07 +00001849 if not (os.path.islink(dir) and inode in self.inodes):
Ka-Ping Yee6191a232001-04-13 15:00:27 +00001850 self.inodes.append(inode) # detect circular symbolic links
Ka-Ping Yeeeca15c12001-04-13 13:53:07 +00001851 return ispackage(dir)
Guido van Rossum8ca162f2002-04-07 06:36:23 +00001852 return False
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001853
Ka-Ping Yee66246962001-04-12 11:59:50 +00001854 def run(self, callback, key=None, completer=None):
1855 if key: key = lower(key)
Guido van Rossum8ca162f2002-04-07 06:36:23 +00001856 self.quit = False
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001857 seen = {}
1858
1859 for modname in sys.builtin_module_names:
Ka-Ping Yee239432a2001-03-02 02:45:08 +00001860 if modname != '__main__':
1861 seen[modname] = 1
Ka-Ping Yee66246962001-04-12 11:59:50 +00001862 if key is None:
1863 callback(None, modname, '')
1864 else:
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001865 desc = split(__import__(modname).__doc__ or '', '\n')[0]
Ka-Ping Yee66246962001-04-12 11:59:50 +00001866 if find(lower(modname + ' - ' + desc), key) >= 0:
1867 callback(None, modname, desc)
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001868
1869 while not self.quit:
1870 node = self.next()
1871 if not node: break
1872 path, package = node
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +00001873 modname = inspect.getmodulename(path)
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001874 if os.path.isfile(path) and modname:
1875 modname = package + (package and '.') + modname
Raymond Hettinger54f02222002-06-01 14:18:47 +00001876 if not modname in seen:
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001877 seen[modname] = 1 # if we see spam.py, skip spam.pyc
Ka-Ping Yee66246962001-04-12 11:59:50 +00001878 if key is None:
1879 callback(path, modname, '')
1880 else:
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001881 desc = synopsis(path) or ''
1882 if find(lower(modname + ' - ' + desc), key) >= 0:
1883 callback(path, modname, desc)
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001884 if completer: completer()
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001885
1886def apropos(key):
1887 """Print all the one-line module summaries that contain a substring."""
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001888 def callback(path, modname, desc):
1889 if modname[-9:] == '.__init__':
1890 modname = modname[:-9] + ' (package)'
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +00001891 print modname, desc and '- ' + desc
1892 try: import warnings
1893 except ImportError: pass
1894 else: warnings.filterwarnings('ignore') # ignore problems during import
Ka-Ping Yee66246962001-04-12 11:59:50 +00001895 ModuleScanner().run(callback, key)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001896
1897# --------------------------------------------------- web browser interface
1898
Ka-Ping Yee66246962001-04-12 11:59:50 +00001899def serve(port, callback=None, completer=None):
Ka-Ping Yeefd540692001-04-12 12:54:36 +00001900 import BaseHTTPServer, mimetools, select
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001901
1902 # Patch up mimetools.Message so it doesn't break if rfc822 is reloaded.
1903 class Message(mimetools.Message):
1904 def __init__(self, fp, seekable=1):
1905 Message = self.__class__
1906 Message.__bases__[0].__bases__[0].__init__(self, fp, seekable)
1907 self.encodingheader = self.getheader('content-transfer-encoding')
1908 self.typeheader = self.getheader('content-type')
1909 self.parsetype()
1910 self.parseplist()
1911
1912 class DocHandler(BaseHTTPServer.BaseHTTPRequestHandler):
1913 def send_document(self, title, contents):
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001914 try:
1915 self.send_response(200)
1916 self.send_header('Content-Type', 'text/html')
1917 self.end_headers()
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001918 self.wfile.write(html.page(title, contents))
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001919 except IOError: pass
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001920
1921 def do_GET(self):
1922 path = self.path
1923 if path[-5:] == '.html': path = path[:-5]
1924 if path[:1] == '/': path = path[1:]
1925 if path and path != '.':
1926 try:
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001927 obj = locate(path, forceload=1)
Ka-Ping Yee3bda8792001-03-23 13:17:50 +00001928 except ErrorDuringImport, value:
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001929 self.send_document(path, html.escape(str(value)))
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001930 return
Ka-Ping Yee3bda8792001-03-23 13:17:50 +00001931 if obj:
1932 self.send_document(describe(obj), html.document(obj, path))
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001933 else:
1934 self.send_document(path,
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001935'no Python documentation found for %s' % repr(path))
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001936 else:
1937 heading = html.heading(
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001938'<big><big><strong>Python: Index of Modules</strong></big></big>',
1939'#ffffff', '#7799ee')
Ka-Ping Yee239432a2001-03-02 02:45:08 +00001940 def bltinlink(name):
1941 return '<a href="%s.html">%s</a>' % (name, name)
Ka-Ping Yee3bda8792001-03-23 13:17:50 +00001942 names = filter(lambda x: x != '__main__',
1943 sys.builtin_module_names)
Ka-Ping Yee239432a2001-03-02 02:45:08 +00001944 contents = html.multicolumn(names, bltinlink)
1945 indices = ['<p>' + html.bigsection(
1946 'Built-in Modules', '#ffffff', '#ee77aa', contents)]
1947
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001948 seen = {}
1949 for dir in pathdirs():
1950 indices.append(html.index(dir, seen))
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001951 contents = heading + join(indices) + '''<p align=right>
Tim Peters2306d242001-09-25 03:18:32 +00001952<font color="#909090" face="helvetica, arial"><strong>
1953pydoc</strong> by Ka-Ping Yee &lt;ping@lfw.org&gt;</font>'''
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001954 self.send_document('Index of Modules', contents)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001955
1956 def log_message(self, *args): pass
1957
Ka-Ping Yeefd540692001-04-12 12:54:36 +00001958 class DocServer(BaseHTTPServer.HTTPServer):
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001959 def __init__(self, port, callback):
Ka-Ping Yeedb8ed152001-03-02 05:58:17 +00001960 host = (sys.platform == 'mac') and '127.0.0.1' or 'localhost'
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001961 self.address = ('', port)
Ka-Ping Yeedb8ed152001-03-02 05:58:17 +00001962 self.url = 'http://%s:%d/' % (host, port)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001963 self.callback = callback
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001964 self.base.__init__(self, self.address, self.handler)
1965
1966 def serve_until_quit(self):
1967 import select
Guido van Rossum8ca162f2002-04-07 06:36:23 +00001968 self.quit = False
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001969 while not self.quit:
1970 rd, wr, ex = select.select([self.socket.fileno()], [], [], 1)
1971 if rd: self.handle_request()
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001972
1973 def server_activate(self):
1974 self.base.server_activate(self)
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001975 if self.callback: self.callback(self)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001976
1977 DocServer.base = BaseHTTPServer.HTTPServer
1978 DocServer.handler = DocHandler
1979 DocHandler.MessageClass = Message
1980 try:
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001981 try:
1982 DocServer(port, callback).serve_until_quit()
1983 except (KeyboardInterrupt, select.error):
1984 pass
1985 finally:
Ka-Ping Yee66246962001-04-12 11:59:50 +00001986 if completer: completer()
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001987
1988# ----------------------------------------------------- graphical interface
1989
1990def gui():
1991 """Graphical interface (starts web server and pops up a control window)."""
1992 class GUI:
1993 def __init__(self, window, port=7464):
1994 self.window = window
1995 self.server = None
1996 self.scanner = None
1997
1998 import Tkinter
1999 self.server_frm = Tkinter.Frame(window)
2000 self.title_lbl = Tkinter.Label(self.server_frm,
2001 text='Starting server...\n ')
2002 self.open_btn = Tkinter.Button(self.server_frm,
2003 text='open browser', command=self.open, state='disabled')
2004 self.quit_btn = Tkinter.Button(self.server_frm,
2005 text='quit serving', command=self.quit, state='disabled')
2006
2007 self.search_frm = Tkinter.Frame(window)
2008 self.search_lbl = Tkinter.Label(self.search_frm, text='Search for')
2009 self.search_ent = Tkinter.Entry(self.search_frm)
2010 self.search_ent.bind('<Return>', self.search)
2011 self.stop_btn = Tkinter.Button(self.search_frm,
2012 text='stop', pady=0, command=self.stop, state='disabled')
2013 if sys.platform == 'win32':
Ka-Ping Yeec92cdf72001-03-02 05:54:35 +00002014 # Trying to hide and show this button crashes under Windows.
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002015 self.stop_btn.pack(side='right')
2016
2017 self.window.title('pydoc')
2018 self.window.protocol('WM_DELETE_WINDOW', self.quit)
2019 self.title_lbl.pack(side='top', fill='x')
2020 self.open_btn.pack(side='left', fill='x', expand=1)
2021 self.quit_btn.pack(side='right', fill='x', expand=1)
2022 self.server_frm.pack(side='top', fill='x')
2023
2024 self.search_lbl.pack(side='left')
2025 self.search_ent.pack(side='right', fill='x', expand=1)
2026 self.search_frm.pack(side='top', fill='x')
2027 self.search_ent.focus_set()
2028
Ka-Ping Yeec92cdf72001-03-02 05:54:35 +00002029 font = ('helvetica', sys.platform == 'win32' and 8 or 10)
Ka-Ping Yee239432a2001-03-02 02:45:08 +00002030 self.result_lst = Tkinter.Listbox(window, font=font, height=6)
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002031 self.result_lst.bind('<Button-1>', self.select)
2032 self.result_lst.bind('<Double-Button-1>', self.goto)
2033 self.result_scr = Tkinter.Scrollbar(window,
2034 orient='vertical', command=self.result_lst.yview)
2035 self.result_lst.config(yscrollcommand=self.result_scr.set)
2036
2037 self.result_frm = Tkinter.Frame(window)
2038 self.goto_btn = Tkinter.Button(self.result_frm,
2039 text='go to selected', command=self.goto)
2040 self.hide_btn = Tkinter.Button(self.result_frm,
2041 text='hide results', command=self.hide)
2042 self.goto_btn.pack(side='left', fill='x', expand=1)
2043 self.hide_btn.pack(side='right', fill='x', expand=1)
2044
2045 self.window.update()
2046 self.minwidth = self.window.winfo_width()
2047 self.minheight = self.window.winfo_height()
2048 self.bigminheight = (self.server_frm.winfo_reqheight() +
2049 self.search_frm.winfo_reqheight() +
2050 self.result_lst.winfo_reqheight() +
2051 self.result_frm.winfo_reqheight())
2052 self.bigwidth, self.bigheight = self.minwidth, self.bigminheight
2053 self.expanded = 0
2054 self.window.wm_geometry('%dx%d' % (self.minwidth, self.minheight))
2055 self.window.wm_minsize(self.minwidth, self.minheight)
Martin v. Löwis5b26abb2002-12-28 09:23:09 +00002056 self.window.tk.willdispatch()
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002057
2058 import threading
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00002059 threading.Thread(
2060 target=serve, args=(port, self.ready, self.quit)).start()
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002061
2062 def ready(self, server):
2063 self.server = server
2064 self.title_lbl.config(
2065 text='Python documentation server at\n' + server.url)
2066 self.open_btn.config(state='normal')
2067 self.quit_btn.config(state='normal')
2068
Ka-Ping Yee239432a2001-03-02 02:45:08 +00002069 def open(self, event=None, url=None):
2070 url = url or self.server.url
2071 try:
2072 import webbrowser
2073 webbrowser.open(url)
2074 except ImportError: # pre-webbrowser.py compatibility
Ka-Ping Yeec92cdf72001-03-02 05:54:35 +00002075 if sys.platform == 'win32':
Ka-Ping Yee239432a2001-03-02 02:45:08 +00002076 os.system('start "%s"' % url)
2077 elif sys.platform == 'mac':
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +00002078 try: import ic
Ka-Ping Yee239432a2001-03-02 02:45:08 +00002079 except ImportError: pass
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +00002080 else: ic.launchurl(url)
Ka-Ping Yee239432a2001-03-02 02:45:08 +00002081 else:
2082 rc = os.system('netscape -remote "openURL(%s)" &' % url)
2083 if rc: os.system('netscape "%s" &' % url)
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002084
2085 def quit(self, event=None):
2086 if self.server:
2087 self.server.quit = 1
2088 self.window.quit()
2089
2090 def search(self, event=None):
2091 key = self.search_ent.get()
2092 self.stop_btn.pack(side='right')
2093 self.stop_btn.config(state='normal')
2094 self.search_lbl.config(text='Searching for "%s"...' % key)
2095 self.search_ent.forget()
2096 self.search_lbl.pack(side='left')
2097 self.result_lst.delete(0, 'end')
2098 self.goto_btn.config(state='disabled')
2099 self.expand()
2100
2101 import threading
2102 if self.scanner:
2103 self.scanner.quit = 1
2104 self.scanner = ModuleScanner()
2105 threading.Thread(target=self.scanner.run,
Ka-Ping Yee6dcfa382001-04-12 20:27:31 +00002106 args=(self.update, key, self.done)).start()
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002107
2108 def update(self, path, modname, desc):
2109 if modname[-9:] == '.__init__':
2110 modname = modname[:-9] + ' (package)'
2111 self.result_lst.insert('end',
2112 modname + ' - ' + (desc or '(no description)'))
2113
2114 def stop(self, event=None):
2115 if self.scanner:
2116 self.scanner.quit = 1
2117 self.scanner = None
2118
2119 def done(self):
2120 self.scanner = None
2121 self.search_lbl.config(text='Search for')
2122 self.search_lbl.pack(side='left')
2123 self.search_ent.pack(side='right', fill='x', expand=1)
2124 if sys.platform != 'win32': self.stop_btn.forget()
2125 self.stop_btn.config(state='disabled')
2126
2127 def select(self, event=None):
2128 self.goto_btn.config(state='normal')
2129
2130 def goto(self, event=None):
2131 selection = self.result_lst.curselection()
2132 if selection:
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002133 modname = split(self.result_lst.get(selection[0]))[0]
Ka-Ping Yee239432a2001-03-02 02:45:08 +00002134 self.open(url=self.server.url + modname + '.html')
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002135
2136 def collapse(self):
2137 if not self.expanded: return
2138 self.result_frm.forget()
2139 self.result_scr.forget()
2140 self.result_lst.forget()
2141 self.bigwidth = self.window.winfo_width()
2142 self.bigheight = self.window.winfo_height()
2143 self.window.wm_geometry('%dx%d' % (self.minwidth, self.minheight))
2144 self.window.wm_minsize(self.minwidth, self.minheight)
2145 self.expanded = 0
2146
2147 def expand(self):
2148 if self.expanded: return
2149 self.result_frm.pack(side='bottom', fill='x')
2150 self.result_scr.pack(side='right', fill='y')
2151 self.result_lst.pack(side='top', fill='both', expand=1)
2152 self.window.wm_geometry('%dx%d' % (self.bigwidth, self.bigheight))
2153 self.window.wm_minsize(self.minwidth, self.bigminheight)
2154 self.expanded = 1
2155
2156 def hide(self, event=None):
2157 self.stop()
2158 self.collapse()
2159
2160 import Tkinter
2161 try:
Martin v. Löwise09bd932004-08-22 16:13:26 +00002162 root = Tkinter.Tk()
2163 # Tk will crash if pythonw.exe has an XP .manifest
2164 # file and the root has is not destroyed explicitly.
2165 # If the problem is ever fixed in Tk, the explicit
2166 # destroy can go.
2167 try:
2168 gui = GUI(root)
2169 root.mainloop()
2170 finally:
2171 root.destroy()
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002172 except KeyboardInterrupt:
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002173 pass
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002174
2175# -------------------------------------------------- command-line interface
2176
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00002177def ispath(x):
Ka-Ping Yee45daeb02002-08-11 15:11:33 +00002178 return isinstance(x, str) and find(x, os.sep) >= 0
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00002179
Ka-Ping Yee1d384632001-03-01 00:24:32 +00002180def cli():
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002181 """Command-line interface (looks at sys.argv to decide what to do)."""
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002182 import getopt
2183 class BadUsage: pass
2184
Ka-Ping Yee3bda8792001-03-23 13:17:50 +00002185 # Scripts don't get the current directory in their path by default.
Ka-Ping Yeef78a81b2001-03-27 08:13:42 +00002186 scriptdir = os.path.dirname(sys.argv[0])
2187 if scriptdir in sys.path:
2188 sys.path.remove(scriptdir)
Ka-Ping Yee3bda8792001-03-23 13:17:50 +00002189 sys.path.insert(0, '.')
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002190
Ka-Ping Yee3bda8792001-03-23 13:17:50 +00002191 try:
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002192 opts, args = getopt.getopt(sys.argv[1:], 'gk:p:w')
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002193 writing = 0
2194
2195 for opt, val in opts:
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002196 if opt == '-g':
2197 gui()
2198 return
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002199 if opt == '-k':
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002200 apropos(val)
2201 return
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002202 if opt == '-p':
2203 try:
2204 port = int(val)
2205 except ValueError:
2206 raise BadUsage
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002207 def ready(server):
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00002208 print 'pydoc server ready at %s' % server.url
2209 def stopped():
2210 print 'pydoc server stopped'
2211 serve(port, ready, stopped)
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002212 return
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002213 if opt == '-w':
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002214 writing = 1
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002215
2216 if not args: raise BadUsage
2217 for arg in args:
Ka-Ping Yee45daeb02002-08-11 15:11:33 +00002218 if ispath(arg) and not os.path.exists(arg):
2219 print 'file %r does not exist' % arg
2220 break
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002221 try:
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00002222 if ispath(arg) and os.path.isfile(arg):
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002223 arg = importfile(arg)
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00002224 if writing:
2225 if ispath(arg) and os.path.isdir(arg):
2226 writedocs(arg)
2227 else:
2228 writedoc(arg)
2229 else:
Martin v. Löwisb8c084e2003-06-14 09:03:46 +00002230 help.help(arg)
Ka-Ping Yee3bda8792001-03-23 13:17:50 +00002231 except ErrorDuringImport, value:
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00002232 print value
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002233
2234 except (getopt.error, BadUsage):
Martin v. Löwisf9b08b82003-10-31 13:05:21 +00002235 cmd = os.path.basename(sys.argv[0])
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002236 print """pydoc - the Python documentation tool
2237
2238%s <name> ...
2239 Show text documentation on something. <name> may be the name of a
Martin v. Löwisb8c084e2003-06-14 09:03:46 +00002240 Python keyword, topic, function, module, or package, or a dotted
2241 reference to a class or function within a module or module in a
2242 package. If <name> contains a '%s', it is used as the path to a
2243 Python source file to document. If name is 'keywords', 'topics',
2244 or 'modules', a listing of these things is displayed.
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002245
2246%s -k <keyword>
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002247 Search for a keyword in the synopsis lines of all available modules.
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002248
2249%s -p <port>
2250 Start an HTTP server on the given port on the local machine.
2251
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002252%s -g
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +00002253 Pop up a graphical interface for finding and serving documentation.
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002254
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002255%s -w <name> ...
2256 Write out the HTML documentation for a module to a file in the current
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +00002257 directory. If <name> contains a '%s', it is treated as a filename; if
2258 it names a directory, documentation is written for all the contents.
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002259""" % (cmd, os.sep, cmd, cmd, cmd, cmd, os.sep)
Ka-Ping Yee1d384632001-03-01 00:24:32 +00002260
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002261if __name__ == '__main__': cli()