blob: 987d2137d7fa9b9d90c695b77b553f50472f7c0d [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
Georg Brandl0751d1a2008-01-21 17:13:03 +000030 http://docs.python.org/library/
Skip Montanaro4997a692003-09-10 16:47:51 +000031
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
Phillip J. Ebyceb30872006-04-18 00:59:55 +000055import sys, imp, os, re, types, inspect, __builtin__, pkgutil
Alexandre Vassalotti50a1acb2008-05-16 06:58:49 +000056from reprlib import Repr
Ka-Ping Yee3bda8792001-03-23 13:17:50 +000057from string import expandtabs, find, join, lower, split, strip, rfind, rstrip
Phillip J. Ebyceb30872006-04-18 00:59:55 +000058try:
59 from collections import deque
60except ImportError:
61 # Python 2.3 compatibility
62 class deque(list):
63 def popleft(self):
64 return self.pop(0)
Ka-Ping Yeedd175342001-02-27 14:43:46 +000065
66# --------------------------------------------------------- common routines
67
Ka-Ping Yeedd175342001-02-27 14:43:46 +000068def pathdirs():
69 """Convert sys.path into a list of absolute, existing, unique paths."""
70 dirs = []
Ka-Ping Yee1d384632001-03-01 00:24:32 +000071 normdirs = []
Ka-Ping Yeedd175342001-02-27 14:43:46 +000072 for dir in sys.path:
73 dir = os.path.abspath(dir or '.')
Ka-Ping Yee1d384632001-03-01 00:24:32 +000074 normdir = os.path.normcase(dir)
75 if normdir not in normdirs and os.path.isdir(dir):
Ka-Ping Yeedd175342001-02-27 14:43:46 +000076 dirs.append(dir)
Ka-Ping Yee1d384632001-03-01 00:24:32 +000077 normdirs.append(normdir)
Ka-Ping Yeedd175342001-02-27 14:43:46 +000078 return dirs
79
80def getdoc(object):
81 """Get the doc string or comments for an object."""
Ka-Ping Yee3bda8792001-03-23 13:17:50 +000082 result = inspect.getdoc(object) or inspect.getcomments(object)
Ka-Ping Yee239432a2001-03-02 02:45:08 +000083 return result and re.sub('^ *\n', '', rstrip(result)) or ''
Ka-Ping Yeedd175342001-02-27 14:43:46 +000084
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +000085def splitdoc(doc):
86 """Split a doc string into a synopsis line (if any) and the rest."""
87 lines = split(strip(doc), '\n')
88 if len(lines) == 1:
89 return lines[0], ''
90 elif len(lines) >= 2 and not rstrip(lines[1]):
91 return lines[0], join(lines[2:], '\n')
92 return '', join(lines, '\n')
93
Ka-Ping Yeedd175342001-02-27 14:43:46 +000094def classname(object, modname):
95 """Get a class name and qualify it with a module name if necessary."""
96 name = object.__name__
97 if object.__module__ != modname:
98 name = object.__module__ + '.' + name
99 return name
100
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000101def isdata(object):
Georg Brandl94432422005-07-22 21:52:25 +0000102 """Check if an object is of a type that probably means it's data."""
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000103 return not (inspect.ismodule(object) or inspect.isclass(object) or
104 inspect.isroutine(object) or inspect.isframe(object) or
105 inspect.istraceback(object) or inspect.iscode(object))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000106
107def replace(text, *pairs):
108 """Do a series of global replacements on a string."""
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +0000109 while pairs:
110 text = join(split(text, pairs[0]), pairs[1])
111 pairs = pairs[2:]
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000112 return text
113
114def cram(text, maxlen):
115 """Omit part of a string if needed to make it fit in a maximum length."""
116 if len(text) > maxlen:
Raymond Hettingerfca3bb62002-10-21 04:44:11 +0000117 pre = max(0, (maxlen-3)//2)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000118 post = max(0, maxlen-3-pre)
119 return text[:pre] + '...' + text[len(text)-post:]
120 return text
121
Brett Cannon84601f12004-06-19 01:22:48 +0000122_re_stripid = re.compile(r' at 0x[0-9a-f]{6,16}(>+)$', re.IGNORECASE)
Ka-Ping Yee1d384632001-03-01 00:24:32 +0000123def stripid(text):
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000124 """Remove the hexadecimal id from a Python object representation."""
Brett Cannonc6c1f472004-06-19 01:02:51 +0000125 # The behaviour of %p is implementation-dependent in terms of case.
126 if _re_stripid.search(repr(Exception)):
127 return _re_stripid.sub(r'\1', text)
Ka-Ping Yee1d384632001-03-01 00:24:32 +0000128 return text
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000129
Brett Cannonc6c1f472004-06-19 01:02:51 +0000130def _is_some_method(obj):
131 return inspect.ismethod(obj) or inspect.ismethoddescriptor(obj)
Tim Peters536d2262001-09-20 05:13:38 +0000132
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000133def allmethods(cl):
134 methods = {}
Tim Peters536d2262001-09-20 05:13:38 +0000135 for key, value in inspect.getmembers(cl, _is_some_method):
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000136 methods[key] = 1
137 for base in cl.__bases__:
138 methods.update(allmethods(base)) # all your base are belong to us
139 for key in methods.keys():
140 methods[key] = getattr(cl, key)
141 return methods
142
Tim Petersfa26f7c2001-09-24 08:05:11 +0000143def _split_list(s, predicate):
144 """Split sequence s via predicate, and return pair ([true], [false]).
145
146 The return value is a 2-tuple of lists,
147 ([x for x in s if predicate(x)],
148 [x for x in s if not predicate(x)])
149 """
150
Tim Peters28355492001-09-23 21:29:55 +0000151 yes = []
152 no = []
Tim Petersfa26f7c2001-09-24 08:05:11 +0000153 for x in s:
154 if predicate(x):
155 yes.append(x)
Tim Peters28355492001-09-23 21:29:55 +0000156 else:
Tim Petersfa26f7c2001-09-24 08:05:11 +0000157 no.append(x)
Tim Peters28355492001-09-23 21:29:55 +0000158 return yes, no
159
Skip Montanaroa5616d22004-06-11 04:46:12 +0000160def visiblename(name, all=None):
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000161 """Decide whether to show documentation on a variable."""
162 # Certain special names are redundant.
Raymond Hettingerdbecd932005-02-06 06:57:08 +0000163 if name in ('__builtins__', '__doc__', '__file__', '__path__',
164 '__module__', '__name__', '__slots__'): return 0
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000165 # Private names are hidden, but special names are displayed.
166 if name.startswith('__') and name.endswith('__'): return 1
Skip Montanaroa5616d22004-06-11 04:46:12 +0000167 if all is not None:
168 # only document that which the programmer exported in __all__
169 return name in all
170 else:
171 return not name.startswith('_')
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000172
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +0000173def classify_class_attrs(object):
174 """Wrap inspect.classify_class_attrs, with fixup for data descriptors."""
175 def fixup((name, kind, cls, value)):
176 if inspect.isdatadescriptor(value):
177 kind = 'data descriptor'
178 return name, kind, cls, value
179 return map(fixup, inspect.classify_class_attrs(object))
180
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000181# ----------------------------------------------------- module manipulation
182
183def ispackage(path):
184 """Guess whether a path refers to a package directory."""
185 if os.path.isdir(path):
Raymond Hettingerdbecd932005-02-06 06:57:08 +0000186 for ext in ('.py', '.pyc', '.pyo'):
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000187 if os.path.isfile(os.path.join(path, '__init__' + ext)):
Tim Petersbc0e9102002-04-04 22:55:58 +0000188 return True
189 return False
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000190
Phillip J. Ebyceb30872006-04-18 00:59:55 +0000191def source_synopsis(file):
192 line = file.readline()
193 while line[:1] == '#' or not strip(line):
194 line = file.readline()
195 if not line: break
196 line = strip(line)
197 if line[:4] == 'r"""': line = line[1:]
198 if line[:3] == '"""':
199 line = line[3:]
200 if line[-1:] == '\\': line = line[:-1]
201 while not strip(line):
202 line = file.readline()
203 if not line: break
204 result = strip(split(line, '"""')[0])
205 else: result = None
206 return result
207
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000208def synopsis(filename, cache={}):
209 """Get the one-line summary out of a module file."""
Raymond Hettinger32200ae2002-06-01 19:51:15 +0000210 mtime = os.stat(filename).st_mtime
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000211 lastupdate, result = cache.get(filename, (0, None))
212 if lastupdate < mtime:
213 info = inspect.getmoduleinfo(filename)
Georg Brandl26fd2e12006-03-08 09:34:53 +0000214 try:
215 file = open(filename)
216 except IOError:
217 # module can't be opened, so skip it
218 return None
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000219 if info and 'b' in info[2]: # binary modules have to be imported
220 try: module = imp.load_module('__temp__', file, filename, info[1:])
221 except: return None
Phillip J. Ebyceb30872006-04-18 00:59:55 +0000222 result = (module.__doc__ or '').splitlines()[0]
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000223 del sys.modules['__temp__']
224 else: # text modules can be directly examined
Phillip J. Ebyceb30872006-04-18 00:59:55 +0000225 result = source_synopsis(file)
226 file.close()
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000227 cache[filename] = (mtime, result)
228 return result
229
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000230class ErrorDuringImport(Exception):
231 """Errors that occurred while trying to import something to document it."""
232 def __init__(self, filename, (exc, value, tb)):
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000233 self.filename = filename
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000234 self.exc = exc
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000235 self.value = value
236 self.tb = tb
237
238 def __str__(self):
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000239 exc = self.exc
240 if type(exc) is types.ClassType:
241 exc = exc.__name__
242 return 'problem in %s - %s: %s' % (self.filename, exc, self.value)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000243
244def importfile(path):
245 """Import a Python source file or compiled file given its path."""
246 magic = imp.get_magic()
247 file = open(path, 'r')
248 if file.read(len(magic)) == magic:
249 kind = imp.PY_COMPILED
250 else:
251 kind = imp.PY_SOURCE
252 file.close()
253 filename = os.path.basename(path)
254 name, ext = os.path.splitext(filename)
255 file = open(path, 'r')
256 try:
257 module = imp.load_module(name, file, path, (ext, 'r', kind))
258 except:
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000259 raise ErrorDuringImport(path, sys.exc_info())
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000260 file.close()
261 return module
262
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000263def safeimport(path, forceload=0, cache={}):
264 """Import a module; handle errors; return None if the module isn't found.
265
266 If the module *is* found but an exception occurs, it's wrapped in an
267 ErrorDuringImport exception and reraised. Unlike __import__, if a
268 package path is specified, the module at the end of the path is returned,
269 not the package at the beginning. If the optional 'forceload' argument
270 is 1, we reload the module from disk (unless it's a dynamic extension)."""
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000271 try:
Ka-Ping Yee9a2dcf82005-11-05 05:04:41 +0000272 # If forceload is 1 and the module has been previously loaded from
273 # disk, we always have to reload the module. Checking the file's
274 # mtime isn't good enough (e.g. the module could contain a class
275 # that inherits from another module that has changed).
276 if forceload and path in sys.modules:
277 if path not in sys.builtin_module_names:
278 # Avoid simply calling reload() because it leaves names in
279 # the currently loaded module lying around if they're not
280 # defined in the new source file. Instead, remove the
281 # module from sys.modules and re-import. Also remove any
282 # submodules because they won't appear in the newly loaded
283 # module's namespace if they're already in sys.modules.
284 subs = [m for m in sys.modules if m.startswith(path + '.')]
285 for key in [path] + subs:
286 # Prevent garbage collection.
287 cache[key] = sys.modules[key]
288 del sys.modules[key]
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000289 module = __import__(path)
290 except:
291 # Did the error occur before or after the module was found?
292 (exc, value, tb) = info = sys.exc_info()
Raymond Hettinger54f02222002-06-01 14:18:47 +0000293 if path in sys.modules:
Fred Drakedb390c12005-10-28 14:39:47 +0000294 # An error occurred while executing the imported module.
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000295 raise ErrorDuringImport(sys.modules[path].__file__, info)
296 elif exc is SyntaxError:
297 # A SyntaxError occurred before we could execute the module.
298 raise ErrorDuringImport(value.filename, info)
299 elif exc is ImportError and \
300 split(lower(str(value)))[:2] == ['no', 'module']:
301 # The module was not found.
302 return None
303 else:
304 # Some other error occurred during the importing process.
305 raise ErrorDuringImport(path, sys.exc_info())
306 for part in split(path, '.')[1:]:
307 try: module = getattr(module, part)
308 except AttributeError: return None
309 return module
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000310
311# ---------------------------------------------------- formatter base class
312
313class Doc:
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000314 def document(self, object, name=None, *args):
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000315 """Generate documentation for an object."""
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000316 args = (object, name) + args
Brett Cannon28a4f0f2003-06-11 23:38:55 +0000317 # 'try' clause is to attempt to handle the possibility that inspect
318 # identifies something in a way that pydoc itself has issues handling;
319 # think 'super' and how it is a descriptor (which raises the exception
320 # by lacking a __name__ attribute) and an instance.
Barry Warsaw00decd72006-07-27 23:43:15 +0000321 if inspect.isgetsetdescriptor(object): return self.docdata(*args)
322 if inspect.ismemberdescriptor(object): return self.docdata(*args)
Brett Cannon28a4f0f2003-06-11 23:38:55 +0000323 try:
324 if inspect.ismodule(object): return self.docmodule(*args)
325 if inspect.isclass(object): return self.docclass(*args)
326 if inspect.isroutine(object): return self.docroutine(*args)
327 except AttributeError:
328 pass
Johannes Gijsbers8de645a2004-11-07 19:16:05 +0000329 if isinstance(object, property): return self.docproperty(*args)
Guido van Rossum68468eb2003-02-27 20:14:51 +0000330 return self.docother(*args)
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000331
332 def fail(self, object, name=None, *args):
333 """Raise an exception for unimplemented types."""
334 message = "don't know how to document object%s of type %s" % (
335 name and ' ' + repr(name), type(object).__name__)
336 raise TypeError, message
337
Barry Warsaw00decd72006-07-27 23:43:15 +0000338 docmodule = docclass = docroutine = docother = docproperty = docdata = fail
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000339
Skip Montanaro4997a692003-09-10 16:47:51 +0000340 def getdocloc(self, object):
341 """Return the location of module docs or None"""
342
343 try:
344 file = inspect.getabsfile(object)
345 except TypeError:
346 file = '(built-in)'
347
348 docloc = os.environ.get("PYTHONDOCS",
Georg Brandl0751d1a2008-01-21 17:13:03 +0000349 "http://docs.python.org/library")
Skip Montanaro4997a692003-09-10 16:47:51 +0000350 basedir = os.path.join(sys.exec_prefix, "lib",
351 "python"+sys.version[0:3])
352 if (isinstance(object, type(os)) and
353 (object.__name__ in ('errno', 'exceptions', 'gc', 'imp',
354 'marshal', 'posix', 'signal', 'sys',
355 'thread', 'zipimport') or
356 (file.startswith(basedir) and
357 not file.startswith(os.path.join(basedir, 'site-packages'))))):
358 if docloc.startswith("http://"):
Georg Brandl0751d1a2008-01-21 17:13:03 +0000359 docloc = "%s/%s" % (docloc.rstrip("/"), object.__name__)
Skip Montanaro4997a692003-09-10 16:47:51 +0000360 else:
Georg Brandl0751d1a2008-01-21 17:13:03 +0000361 docloc = os.path.join(docloc, object.__name__ + ".html")
Skip Montanaro4997a692003-09-10 16:47:51 +0000362 else:
363 docloc = None
364 return docloc
365
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000366# -------------------------------------------- HTML documentation generator
367
368class HTMLRepr(Repr):
369 """Class for safely making an HTML representation of a Python object."""
370 def __init__(self):
371 Repr.__init__(self)
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000372 self.maxlist = self.maxtuple = 20
373 self.maxdict = 10
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000374 self.maxstring = self.maxother = 100
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000375
376 def escape(self, text):
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +0000377 return replace(text, '&', '&amp;', '<', '&lt;', '>', '&gt;')
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000378
379 def repr(self, object):
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000380 return Repr.repr(self, object)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000381
382 def repr1(self, x, level):
Skip Montanaro0fe8fce2003-06-27 15:45:41 +0000383 if hasattr(type(x), '__name__'):
384 methodname = 'repr_' + join(split(type(x).__name__), '_')
385 if hasattr(self, methodname):
386 return getattr(self, methodname)(x, level)
387 return self.escape(cram(stripid(repr(x)), self.maxother))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000388
389 def repr_string(self, x, level):
Ka-Ping Yeea2fe1032001-03-02 01:19:14 +0000390 test = cram(x, self.maxstring)
391 testrepr = repr(test)
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +0000392 if '\\' in test and '\\' not in replace(testrepr, r'\\', ''):
Ka-Ping Yeea2fe1032001-03-02 01:19:14 +0000393 # Backslashes are only literal in the string and are never
394 # needed to make any special characters, so show a raw string.
395 return 'r' + testrepr[0] + self.escape(test) + testrepr[0]
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +0000396 return re.sub(r'((\\[\\abfnrtv\'"]|\\[0-9]..|\\x..|\\u....)+)',
Ka-Ping Yeea2fe1032001-03-02 01:19:14 +0000397 r'<font color="#c040c0">\1</font>',
398 self.escape(testrepr))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000399
Skip Montanarodf708782002-03-07 22:58:02 +0000400 repr_str = repr_string
401
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000402 def repr_instance(self, x, level):
403 try:
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000404 return self.escape(cram(stripid(repr(x)), self.maxstring))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000405 except:
406 return self.escape('<%s instance>' % x.__class__.__name__)
407
408 repr_unicode = repr_string
409
410class HTMLDoc(Doc):
411 """Formatter class for HTML documentation."""
412
413 # ------------------------------------------- HTML formatting utilities
414
415 _repr_instance = HTMLRepr()
416 repr = _repr_instance.repr
417 escape = _repr_instance.escape
418
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000419 def page(self, title, contents):
420 """Format an HTML page."""
421 return '''
Tim Peters59ed4482001-10-31 04:20:26 +0000422<!doctype html PUBLIC "-//W3C//DTD HTML 4.0 Transitional//EN">
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000423<html><head><title>Python: %s</title>
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000424</head><body bgcolor="#f0f0f8">
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000425%s
426</body></html>''' % (title, contents)
427
428 def heading(self, title, fgcol, bgcol, extras=''):
429 """Format a page heading."""
430 return '''
Tim Peters59ed4482001-10-31 04:20:26 +0000431<table width="100%%" cellspacing=0 cellpadding=2 border=0 summary="heading">
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000432<tr bgcolor="%s">
Tim Peters2306d242001-09-25 03:18:32 +0000433<td valign=bottom>&nbsp;<br>
434<font color="%s" face="helvetica, arial">&nbsp;<br>%s</font></td
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000435><td align=right valign=bottom
Ka-Ping Yee987ec902001-03-23 13:35:45 +0000436><font color="%s" face="helvetica, arial">%s</font></td></tr></table>
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000437 ''' % (bgcol, fgcol, title, fgcol, extras or '&nbsp;')
438
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000439 def section(self, title, fgcol, bgcol, contents, width=6,
440 prelude='', marginalia=None, gap='&nbsp;'):
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000441 """Format a section with a heading."""
442 if marginalia is None:
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +0000443 marginalia = '<tt>' + '&nbsp;' * width + '</tt>'
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000444 result = '''<p>
Tim Peters59ed4482001-10-31 04:20:26 +0000445<table width="100%%" cellspacing=0 cellpadding=2 border=0 summary="section">
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000446<tr bgcolor="%s">
Tim Peters2306d242001-09-25 03:18:32 +0000447<td colspan=3 valign=bottom>&nbsp;<br>
448<font color="%s" face="helvetica, arial">%s</font></td></tr>
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000449 ''' % (bgcol, fgcol, title)
450 if prelude:
451 result = result + '''
Ka-Ping Yee987ec902001-03-23 13:35:45 +0000452<tr bgcolor="%s"><td rowspan=2>%s</td>
453<td colspan=2>%s</td></tr>
454<tr><td>%s</td>''' % (bgcol, marginalia, prelude, gap)
455 else:
456 result = result + '''
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000457<tr><td bgcolor="%s">%s</td><td>%s</td>''' % (bgcol, marginalia, gap)
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000458
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +0000459 return result + '\n<td width="100%%">%s</td></tr></table>' % contents
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000460
461 def bigsection(self, title, *args):
462 """Format a section with a big heading."""
463 title = '<big><strong>%s</strong></big>' % title
Guido van Rossum68468eb2003-02-27 20:14:51 +0000464 return self.section(title, *args)
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000465
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000466 def preformat(self, text):
467 """Format literal preformatted text."""
468 text = self.escape(expandtabs(text))
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +0000469 return replace(text, '\n\n', '\n \n', '\n\n', '\n \n',
470 ' ', '&nbsp;', '\n', '<br>\n')
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000471
472 def multicolumn(self, list, format, cols=4):
473 """Format a list of items into a multi-column list."""
474 result = ''
475 rows = (len(list)+cols-1)/cols
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000476 for col in range(cols):
477 result = result + '<td width="%d%%" valign=top>' % (100/cols)
478 for i in range(rows*col, rows*col+rows):
479 if i < len(list):
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000480 result = result + format(list[i]) + '<br>\n'
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000481 result = result + '</td>'
Tim Peters59ed4482001-10-31 04:20:26 +0000482 return '<table width="100%%" summary="list"><tr>%s</tr></table>' % result
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000483
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000484 def grey(self, text): return '<font color="#909090">%s</font>' % text
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000485
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000486 def namelink(self, name, *dicts):
487 """Make a link for an identifier, given name-to-URL mappings."""
488 for dict in dicts:
Raymond Hettinger54f02222002-06-01 14:18:47 +0000489 if name in dict:
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000490 return '<a href="%s">%s</a>' % (dict[name], name)
491 return name
492
Ka-Ping Yeeb7a48302001-04-12 20:39:14 +0000493 def classlink(self, object, modname):
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000494 """Make a link for a class."""
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000495 name, module = object.__name__, sys.modules.get(object.__module__)
496 if hasattr(module, name) and getattr(module, name) is object:
Ka-Ping Yeeb7a48302001-04-12 20:39:14 +0000497 return '<a href="%s.html#%s">%s</a>' % (
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000498 module.__name__, name, classname(object, modname))
499 return classname(object, modname)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000500
501 def modulelink(self, object):
502 """Make a link for a module."""
503 return '<a href="%s.html">%s</a>' % (object.__name__, object.__name__)
504
505 def modpkglink(self, (name, path, ispackage, shadowed)):
506 """Make a link for a module or package to display in an index."""
507 if shadowed:
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000508 return self.grey(name)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000509 if path:
510 url = '%s.%s.html' % (path, name)
511 else:
512 url = '%s.html' % name
513 if ispackage:
514 text = '<strong>%s</strong>&nbsp;(package)' % name
515 else:
516 text = name
517 return '<a href="%s">%s</a>' % (url, text)
518
519 def markup(self, text, escape=None, funcs={}, classes={}, methods={}):
520 """Mark up some plain text, given a context of symbols to look for.
521 Each context dictionary maps object names to anchor names."""
522 escape = escape or self.escape
523 results = []
524 here = 0
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000525 pattern = re.compile(r'\b((http|ftp)://\S+[\w/]|'
526 r'RFC[- ]?(\d+)|'
Ka-Ping Yeef78a81b2001-03-27 08:13:42 +0000527 r'PEP[- ]?(\d+)|'
Neil Schemenauerd69711c2002-03-24 23:02:07 +0000528 r'(self\.)?(\w+))')
Guido van Rossum8ca162f2002-04-07 06:36:23 +0000529 while True:
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000530 match = pattern.search(text, here)
531 if not match: break
532 start, end = match.span()
533 results.append(escape(text[here:start]))
534
Ka-Ping Yeef78a81b2001-03-27 08:13:42 +0000535 all, scheme, rfc, pep, selfdot, name = match.groups()
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000536 if scheme:
Neil Schemenauercddc1a02002-03-24 23:11:21 +0000537 url = escape(all).replace('"', '&quot;')
538 results.append('<a href="%s">%s</a>' % (url, url))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000539 elif rfc:
Ka-Ping Yeef78a81b2001-03-27 08:13:42 +0000540 url = 'http://www.rfc-editor.org/rfc/rfc%d.txt' % int(rfc)
541 results.append('<a href="%s">%s</a>' % (url, escape(all)))
542 elif pep:
Andrew M. Kuchling727a5902008-02-05 16:06:57 +0000543 url = 'http://www.python.org/dev/peps/pep-%04d/' % int(pep)
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000544 results.append('<a href="%s">%s</a>' % (url, escape(all)))
545 elif text[end:end+1] == '(':
546 results.append(self.namelink(name, methods, funcs, classes))
547 elif selfdot:
548 results.append('self.<strong>%s</strong>' % name)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000549 else:
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000550 results.append(self.namelink(name, classes))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000551 here = end
552 results.append(escape(text[here:]))
553 return join(results, '')
554
555 # ---------------------------------------------- type-specific routines
556
Ka-Ping Yeeb7a48302001-04-12 20:39:14 +0000557 def formattree(self, tree, modname, parent=None):
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000558 """Produce HTML for a class tree as given by inspect.getclasstree()."""
559 result = ''
560 for entry in tree:
561 if type(entry) is type(()):
562 c, bases = entry
Tim Peters2306d242001-09-25 03:18:32 +0000563 result = result + '<dt><font face="helvetica, arial">'
Ka-Ping Yeeb7a48302001-04-12 20:39:14 +0000564 result = result + self.classlink(c, modname)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000565 if bases and bases != (parent,):
566 parents = []
567 for base in bases:
Ka-Ping Yeeb7a48302001-04-12 20:39:14 +0000568 parents.append(self.classlink(base, modname))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000569 result = result + '(' + join(parents, ', ') + ')'
Tim Peters2306d242001-09-25 03:18:32 +0000570 result = result + '\n</font></dt>'
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000571 elif type(entry) is type([]):
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000572 result = result + '<dd>\n%s</dd>\n' % self.formattree(
Ka-Ping Yeeb7a48302001-04-12 20:39:14 +0000573 entry, modname, c)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000574 return '<dl>\n%s</dl>\n' % result
575
Tim Peters8dd7ade2001-10-18 19:56:17 +0000576 def docmodule(self, object, name=None, mod=None, *ignored):
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000577 """Produce HTML documentation for a module object."""
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000578 name = object.__name__ # ignore the passed-in name
Skip Montanaroa5616d22004-06-11 04:46:12 +0000579 try:
580 all = object.__all__
581 except AttributeError:
582 all = None
Ka-Ping Yee66efbc72001-03-01 13:55:20 +0000583 parts = split(name, '.')
584 links = []
585 for i in range(len(parts)-1):
586 links.append(
587 '<a href="%s.html"><font color="#ffffff">%s</font></a>' %
588 (join(parts[:i+1], '.'), parts[i]))
589 linkedname = join(links + parts[-1:], '.')
590 head = '<big><big><strong>%s</strong></big></big>' % linkedname
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000591 try:
Ka-Ping Yee239432a2001-03-02 02:45:08 +0000592 path = inspect.getabsfile(object)
Ka-Ping Yee6191a232001-04-13 15:00:27 +0000593 url = path
594 if sys.platform == 'win32':
595 import nturl2path
596 url = nturl2path.pathname2url(path)
597 filelink = '<a href="file:%s">%s</a>' % (url, path)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000598 except TypeError:
599 filelink = '(built-in)'
Ka-Ping Yee6f3f9a42001-02-27 22:42:36 +0000600 info = []
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000601 if hasattr(object, '__version__'):
Ka-Ping Yee6f3f9a42001-02-27 22:42:36 +0000602 version = str(object.__version__)
Ka-Ping Yee40c49912001-02-27 22:46:01 +0000603 if version[:11] == '$' + 'Revision: ' and version[-1:] == '$':
604 version = strip(version[11:-1])
Ka-Ping Yee1d384632001-03-01 00:24:32 +0000605 info.append('version %s' % self.escape(version))
Ka-Ping Yee6f3f9a42001-02-27 22:42:36 +0000606 if hasattr(object, '__date__'):
607 info.append(self.escape(str(object.__date__)))
608 if info:
609 head = head + ' (%s)' % join(info, ', ')
Skip Montanaro4997a692003-09-10 16:47:51 +0000610 docloc = self.getdocloc(object)
611 if docloc is not None:
612 docloc = '<br><a href="%(docloc)s">Module Docs</a>' % locals()
613 else:
614 docloc = ''
Ka-Ping Yee66efbc72001-03-01 13:55:20 +0000615 result = self.heading(
Skip Montanaro4997a692003-09-10 16:47:51 +0000616 head, '#ffffff', '#7799ee',
617 '<a href=".">index</a><br>' + filelink + docloc)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000618
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000619 modules = inspect.getmembers(object, inspect.ismodule)
620
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000621 classes, cdict = [], {}
622 for key, value in inspect.getmembers(object, inspect.isclass):
Johannes Gijsbers4c11f602004-08-30 14:13:04 +0000623 # if __all__ exists, believe it. Otherwise use old heuristic.
624 if (all is not None or
625 (inspect.getmodule(value) or object) is object):
Skip Montanaroa5616d22004-06-11 04:46:12 +0000626 if visiblename(key, all):
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000627 classes.append((key, value))
628 cdict[key] = cdict[value] = '#' + key
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000629 for key, value in classes:
630 for base in value.__bases__:
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000631 key, modname = base.__name__, base.__module__
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000632 module = sys.modules.get(modname)
633 if modname != name and module and hasattr(module, key):
634 if getattr(module, key) is base:
Raymond Hettinger54f02222002-06-01 14:18:47 +0000635 if not key in cdict:
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000636 cdict[key] = cdict[base] = modname + '.html#' + key
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000637 funcs, fdict = [], {}
638 for key, value in inspect.getmembers(object, inspect.isroutine):
Johannes Gijsbers4c11f602004-08-30 14:13:04 +0000639 # if __all__ exists, believe it. Otherwise use old heuristic.
640 if (all is not None or
641 inspect.isbuiltin(value) or inspect.getmodule(value) is object):
Skip Montanaroa5616d22004-06-11 04:46:12 +0000642 if visiblename(key, all):
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000643 funcs.append((key, value))
644 fdict[key] = '#-' + key
645 if inspect.isfunction(value): fdict[value] = fdict[key]
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000646 data = []
647 for key, value in inspect.getmembers(object, isdata):
Skip Montanaroa5616d22004-06-11 04:46:12 +0000648 if visiblename(key, all):
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000649 data.append((key, value))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000650
651 doc = self.markup(getdoc(object), self.preformat, fdict, cdict)
652 doc = doc and '<tt>%s</tt>' % doc
Tim Peters2306d242001-09-25 03:18:32 +0000653 result = result + '<p>%s</p>\n' % doc
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000654
655 if hasattr(object, '__path__'):
656 modpkgs = []
Phillip J. Ebyceb30872006-04-18 00:59:55 +0000657 for importer, modname, ispkg in pkgutil.iter_modules(object.__path__):
658 modpkgs.append((modname, name, ispkg, 0))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000659 modpkgs.sort()
660 contents = self.multicolumn(modpkgs, self.modpkglink)
661 result = result + self.bigsection(
662 'Package Contents', '#ffffff', '#aa55cc', contents)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000663 elif modules:
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000664 contents = self.multicolumn(
665 modules, lambda (key, value), s=self: s.modulelink(value))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000666 result = result + self.bigsection(
Georg Brandlb169eaa2008-02-17 21:18:55 +0000667 'Modules', '#ffffff', '#aa55cc', contents)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000668
669 if classes:
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000670 classlist = map(lambda (key, value): value, classes)
Ka-Ping Yeeb7a48302001-04-12 20:39:14 +0000671 contents = [
672 self.formattree(inspect.getclasstree(classlist, 1), name)]
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000673 for key, value in classes:
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +0000674 contents.append(self.document(value, key, name, fdict, cdict))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000675 result = result + self.bigsection(
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000676 'Classes', '#ffffff', '#ee77aa', join(contents))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000677 if funcs:
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000678 contents = []
679 for key, value in funcs:
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +0000680 contents.append(self.document(value, key, name, fdict, cdict))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000681 result = result + self.bigsection(
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000682 'Functions', '#ffffff', '#eeaa77', join(contents))
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000683 if data:
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000684 contents = []
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000685 for key, value in data:
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000686 contents.append(self.document(value, key))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000687 result = result + self.bigsection(
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000688 'Data', '#ffffff', '#55aa55', join(contents, '<br>\n'))
Ka-Ping Yee6f3f9a42001-02-27 22:42:36 +0000689 if hasattr(object, '__author__'):
690 contents = self.markup(str(object.__author__), self.preformat)
691 result = result + self.bigsection(
692 'Author', '#ffffff', '#7799ee', contents)
Ka-Ping Yee6f3f9a42001-02-27 22:42:36 +0000693 if hasattr(object, '__credits__'):
694 contents = self.markup(str(object.__credits__), self.preformat)
695 result = result + self.bigsection(
696 'Credits', '#ffffff', '#7799ee', contents)
697
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000698 return result
699
Tim Peters8dd7ade2001-10-18 19:56:17 +0000700 def docclass(self, object, name=None, mod=None, funcs={}, classes={},
701 *ignored):
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000702 """Produce HTML documentation for a class object."""
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000703 realname = object.__name__
704 name = name or realname
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000705 bases = object.__bases__
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000706
Tim Petersb47879b2001-09-24 04:47:19 +0000707 contents = []
708 push = contents.append
709
Tim Petersfa26f7c2001-09-24 08:05:11 +0000710 # Cute little class to pump out a horizontal rule between sections.
711 class HorizontalRule:
712 def __init__(self):
713 self.needone = 0
714 def maybe(self):
715 if self.needone:
716 push('<hr>\n')
717 self.needone = 1
718 hr = HorizontalRule()
719
Tim Petersc86f6ca2001-09-26 21:31:51 +0000720 # List the mro, if non-trivial.
Raymond Hettinger756b3f32004-01-29 06:37:52 +0000721 mro = deque(inspect.getmro(object))
Tim Petersc86f6ca2001-09-26 21:31:51 +0000722 if len(mro) > 2:
723 hr.maybe()
724 push('<dl><dt>Method resolution order:</dt>\n')
725 for base in mro:
726 push('<dd>%s</dd>\n' % self.classlink(base,
727 object.__module__))
728 push('</dl>\n')
729
Tim Petersb47879b2001-09-24 04:47:19 +0000730 def spill(msg, attrs, predicate):
Tim Petersfa26f7c2001-09-24 08:05:11 +0000731 ok, attrs = _split_list(attrs, predicate)
Tim Petersb47879b2001-09-24 04:47:19 +0000732 if ok:
Tim Petersfa26f7c2001-09-24 08:05:11 +0000733 hr.maybe()
Tim Petersb47879b2001-09-24 04:47:19 +0000734 push(msg)
735 for name, kind, homecls, value in ok:
736 push(self.document(getattr(object, name), name, mod,
737 funcs, classes, mdict, object))
738 push('\n')
739 return attrs
740
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +0000741 def spilldescriptors(msg, attrs, predicate):
Tim Petersfa26f7c2001-09-24 08:05:11 +0000742 ok, attrs = _split_list(attrs, predicate)
Tim Petersb47879b2001-09-24 04:47:19 +0000743 if ok:
Tim Petersfa26f7c2001-09-24 08:05:11 +0000744 hr.maybe()
Tim Petersb47879b2001-09-24 04:47:19 +0000745 push(msg)
746 for name, kind, homecls, value in ok:
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +0000747 push(self._docdescriptor(name, value, mod))
Tim Petersb47879b2001-09-24 04:47:19 +0000748 return attrs
749
Tim Petersfa26f7c2001-09-24 08:05:11 +0000750 def spilldata(msg, attrs, predicate):
751 ok, attrs = _split_list(attrs, predicate)
Tim Petersb47879b2001-09-24 04:47:19 +0000752 if ok:
Tim Petersfa26f7c2001-09-24 08:05:11 +0000753 hr.maybe()
Tim Petersb47879b2001-09-24 04:47:19 +0000754 push(msg)
755 for name, kind, homecls, value in ok:
756 base = self.docother(getattr(object, name), name, mod)
Martin v. Löwise59e2ba2003-05-03 09:09:02 +0000757 if callable(value) or inspect.isdatadescriptor(value):
Guido van Rossum5e355b22002-05-21 20:56:15 +0000758 doc = getattr(value, "__doc__", None)
759 else:
760 doc = None
Tim Petersb47879b2001-09-24 04:47:19 +0000761 if doc is None:
762 push('<dl><dt>%s</dl>\n' % base)
763 else:
764 doc = self.markup(getdoc(value), self.preformat,
765 funcs, classes, mdict)
Tim Peters2306d242001-09-25 03:18:32 +0000766 doc = '<dd><tt>%s</tt>' % doc
Tim Petersb47879b2001-09-24 04:47:19 +0000767 push('<dl><dt>%s%s</dl>\n' % (base, doc))
768 push('\n')
769 return attrs
770
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000771 attrs = filter(lambda (name, kind, cls, value): visiblename(name),
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +0000772 classify_class_attrs(object))
Tim Petersb47879b2001-09-24 04:47:19 +0000773 mdict = {}
774 for key, kind, homecls, value in attrs:
775 mdict[key] = anchor = '#' + name + '-' + key
776 value = getattr(object, key)
777 try:
778 # The value may not be hashable (e.g., a data attr with
779 # a dict or list value).
780 mdict[value] = anchor
781 except TypeError:
782 pass
783
Tim Petersfa26f7c2001-09-24 08:05:11 +0000784 while attrs:
Tim Peters351e3622001-09-27 03:29:51 +0000785 if mro:
Raymond Hettinger756b3f32004-01-29 06:37:52 +0000786 thisclass = mro.popleft()
Tim Peters351e3622001-09-27 03:29:51 +0000787 else:
788 thisclass = attrs[0][2]
Tim Petersfa26f7c2001-09-24 08:05:11 +0000789 attrs, inherited = _split_list(attrs, lambda t: t[2] is thisclass)
790
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000791 if thisclass is __builtin__.object:
792 attrs = inherited
793 continue
794 elif thisclass is object:
795 tag = 'defined here'
Tim Petersb47879b2001-09-24 04:47:19 +0000796 else:
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000797 tag = 'inherited from %s' % self.classlink(thisclass,
798 object.__module__)
Tim Petersb47879b2001-09-24 04:47:19 +0000799 tag += ':<br>\n'
800
801 # Sort attrs by name.
Phillip J. Ebyceb30872006-04-18 00:59:55 +0000802 try:
803 attrs.sort(key=lambda t: t[0])
804 except TypeError:
805 attrs.sort(lambda t1, t2: cmp(t1[0], t2[0])) # 2.3 compat
Tim Petersb47879b2001-09-24 04:47:19 +0000806
807 # Pump out the attrs, segregated by kind.
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000808 attrs = spill('Methods %s' % tag, attrs,
Tim Petersb47879b2001-09-24 04:47:19 +0000809 lambda t: t[1] == 'method')
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000810 attrs = spill('Class methods %s' % tag, attrs,
Tim Petersb47879b2001-09-24 04:47:19 +0000811 lambda t: t[1] == 'class method')
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000812 attrs = spill('Static methods %s' % tag, attrs,
Tim Petersb47879b2001-09-24 04:47:19 +0000813 lambda t: t[1] == 'static method')
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +0000814 attrs = spilldescriptors('Data descriptors %s' % tag, attrs,
815 lambda t: t[1] == 'data descriptor')
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000816 attrs = spilldata('Data and other attributes %s' % tag, attrs,
Tim Petersfa26f7c2001-09-24 08:05:11 +0000817 lambda t: t[1] == 'data')
Tim Petersb47879b2001-09-24 04:47:19 +0000818 assert attrs == []
Tim Peters351e3622001-09-27 03:29:51 +0000819 attrs = inherited
Tim Petersb47879b2001-09-24 04:47:19 +0000820
821 contents = ''.join(contents)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000822
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000823 if name == realname:
824 title = '<a name="%s">class <strong>%s</strong></a>' % (
825 name, realname)
826 else:
827 title = '<strong>%s</strong> = <a name="%s">class %s</a>' % (
828 name, name, realname)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000829 if bases:
830 parents = []
831 for base in bases:
Ka-Ping Yeeb7a48302001-04-12 20:39:14 +0000832 parents.append(self.classlink(base, object.__module__))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000833 title = title + '(%s)' % join(parents, ', ')
Tim Peters2306d242001-09-25 03:18:32 +0000834 doc = self.markup(getdoc(object), self.preformat, funcs, classes, mdict)
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000835 doc = doc and '<tt>%s<br>&nbsp;</tt>' % doc
Tim Petersc86f6ca2001-09-26 21:31:51 +0000836
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000837 return self.section(title, '#000000', '#ffc8d8', contents, 3, doc)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000838
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000839 def formatvalue(self, object):
840 """Format an argument default value as text."""
Tim Peters2306d242001-09-25 03:18:32 +0000841 return self.grey('=' + self.repr(object))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000842
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +0000843 def docroutine(self, object, name=None, mod=None,
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000844 funcs={}, classes={}, methods={}, cl=None):
Ka-Ping Yee66efbc72001-03-01 13:55:20 +0000845 """Produce HTML documentation for a function or method object."""
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000846 realname = object.__name__
847 name = name or realname
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000848 anchor = (cl and cl.__name__ or '') + '-' + name
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000849 note = ''
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000850 skipdocs = 0
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000851 if inspect.ismethod(object):
Ka-Ping Yee6dcfa382001-04-12 20:27:31 +0000852 imclass = object.im_class
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000853 if cl:
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +0000854 if imclass is not cl:
Ka-Ping Yeeb7a48302001-04-12 20:39:14 +0000855 note = ' from ' + self.classlink(imclass, mod)
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000856 else:
Georg Brandl7e2b6bb2007-03-13 22:16:30 +0000857 if object.im_self is not None:
Ka-Ping Yeeb7a48302001-04-12 20:39:14 +0000858 note = ' method of %s instance' % self.classlink(
859 object.im_self.__class__, mod)
860 else:
861 note = ' unbound %s method' % self.classlink(imclass,mod)
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000862 object = object.im_func
863
864 if name == realname:
865 title = '<a name="%s"><strong>%s</strong></a>' % (anchor, realname)
866 else:
Raymond Hettinger54f02222002-06-01 14:18:47 +0000867 if (cl and realname in cl.__dict__ and
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000868 cl.__dict__[realname] is object):
Ka-Ping Yeee280c062001-03-23 14:05:53 +0000869 reallink = '<a href="#%s">%s</a>' % (
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000870 cl.__name__ + '-' + realname, realname)
871 skipdocs = 1
872 else:
873 reallink = realname
874 title = '<a name="%s"><strong>%s</strong></a> = %s' % (
875 anchor, name, reallink)
Tim Peters4bcfa312001-09-20 06:08:24 +0000876 if inspect.isfunction(object):
Ka-Ping Yee66efbc72001-03-01 13:55:20 +0000877 args, varargs, varkw, defaults = inspect.getargspec(object)
878 argspec = inspect.formatargspec(
879 args, varargs, varkw, defaults, formatvalue=self.formatvalue)
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000880 if realname == '<lambda>':
Tim Peters59ed4482001-10-31 04:20:26 +0000881 title = '<strong>%s</strong> <em>lambda</em> ' % name
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000882 argspec = argspec[1:-1] # remove parentheses
Tim Peters4bcfa312001-09-20 06:08:24 +0000883 else:
884 argspec = '(...)'
Ka-Ping Yee66efbc72001-03-01 13:55:20 +0000885
Tim Peters2306d242001-09-25 03:18:32 +0000886 decl = title + argspec + (note and self.grey(
887 '<font face="helvetica, arial">%s</font>' % note))
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000888
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000889 if skipdocs:
Tim Peters2306d242001-09-25 03:18:32 +0000890 return '<dl><dt>%s</dt></dl>\n' % decl
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000891 else:
892 doc = self.markup(
893 getdoc(object), self.preformat, funcs, classes, methods)
Tim Peters2306d242001-09-25 03:18:32 +0000894 doc = doc and '<dd><tt>%s</tt></dd>' % doc
895 return '<dl><dt>%s</dt>%s</dl>\n' % (decl, doc)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000896
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +0000897 def _docdescriptor(self, name, value, mod):
Johannes Gijsbers8de645a2004-11-07 19:16:05 +0000898 results = []
899 push = results.append
900
901 if name:
902 push('<dl><dt><strong>%s</strong></dt>\n' % name)
903 if value.__doc__ is not None:
Ka-Ping Yeebba6acc2005-02-19 22:58:26 +0000904 doc = self.markup(getdoc(value), self.preformat)
Johannes Gijsbers8de645a2004-11-07 19:16:05 +0000905 push('<dd><tt>%s</tt></dd>\n' % doc)
Johannes Gijsbers8de645a2004-11-07 19:16:05 +0000906 push('</dl>\n')
907
908 return ''.join(results)
909
910 def docproperty(self, object, name=None, mod=None, cl=None):
911 """Produce html documentation for a property."""
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +0000912 return self._docdescriptor(name, object, mod)
Johannes Gijsbers8de645a2004-11-07 19:16:05 +0000913
Tim Peters8dd7ade2001-10-18 19:56:17 +0000914 def docother(self, object, name=None, mod=None, *ignored):
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000915 """Produce HTML documentation for a data object."""
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +0000916 lhs = name and '<strong>%s</strong> = ' % name or ''
917 return lhs + self.repr(object)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000918
Barry Warsaw00decd72006-07-27 23:43:15 +0000919 def docdata(self, object, name=None, mod=None, cl=None):
920 """Produce html documentation for a data descriptor."""
921 return self._docdescriptor(name, object, mod)
922
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000923 def index(self, dir, shadowed=None):
924 """Generate an HTML index for a directory of modules."""
925 modpkgs = []
926 if shadowed is None: shadowed = {}
Phillip J. Ebyceb30872006-04-18 00:59:55 +0000927 for importer, name, ispkg in pkgutil.iter_modules([dir]):
928 modpkgs.append((name, '', ispkg, name in shadowed))
929 shadowed[name] = 1
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000930
931 modpkgs.sort()
932 contents = self.multicolumn(modpkgs, self.modpkglink)
933 return self.bigsection(dir, '#ffffff', '#ee77aa', contents)
934
935# -------------------------------------------- text documentation generator
936
937class TextRepr(Repr):
938 """Class for safely making a text representation of a Python object."""
939 def __init__(self):
940 Repr.__init__(self)
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000941 self.maxlist = self.maxtuple = 20
942 self.maxdict = 10
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000943 self.maxstring = self.maxother = 100
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000944
945 def repr1(self, x, level):
Skip Montanaro0fe8fce2003-06-27 15:45:41 +0000946 if hasattr(type(x), '__name__'):
947 methodname = 'repr_' + join(split(type(x).__name__), '_')
948 if hasattr(self, methodname):
949 return getattr(self, methodname)(x, level)
950 return cram(stripid(repr(x)), self.maxother)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000951
Ka-Ping Yeea2fe1032001-03-02 01:19:14 +0000952 def repr_string(self, x, level):
953 test = cram(x, self.maxstring)
954 testrepr = repr(test)
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +0000955 if '\\' in test and '\\' not in replace(testrepr, r'\\', ''):
Ka-Ping Yeea2fe1032001-03-02 01:19:14 +0000956 # Backslashes are only literal in the string and are never
957 # needed to make any special characters, so show a raw string.
958 return 'r' + testrepr[0] + test + testrepr[0]
959 return testrepr
960
Skip Montanarodf708782002-03-07 22:58:02 +0000961 repr_str = repr_string
962
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000963 def repr_instance(self, x, level):
964 try:
Ka-Ping Yee1d384632001-03-01 00:24:32 +0000965 return cram(stripid(repr(x)), self.maxstring)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000966 except:
967 return '<%s instance>' % x.__class__.__name__
968
969class TextDoc(Doc):
970 """Formatter class for text documentation."""
971
972 # ------------------------------------------- text formatting utilities
973
974 _repr_instance = TextRepr()
975 repr = _repr_instance.repr
976
977 def bold(self, text):
978 """Format a string in bold by overstriking."""
979 return join(map(lambda ch: ch + '\b' + ch, text), '')
980
981 def indent(self, text, prefix=' '):
982 """Indent text by prepending a given prefix to each line."""
983 if not text: return ''
984 lines = split(text, '\n')
985 lines = map(lambda line, prefix=prefix: prefix + line, lines)
986 if lines: lines[-1] = rstrip(lines[-1])
987 return join(lines, '\n')
988
989 def section(self, title, contents):
990 """Format a section with a given heading."""
991 return self.bold(title) + '\n' + rstrip(self.indent(contents)) + '\n\n'
992
993 # ---------------------------------------------- type-specific routines
994
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000995 def formattree(self, tree, modname, parent=None, prefix=''):
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000996 """Render in text a class tree as returned by inspect.getclasstree()."""
997 result = ''
998 for entry in tree:
999 if type(entry) is type(()):
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001000 c, bases = entry
1001 result = result + prefix + classname(c, modname)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001002 if bases and bases != (parent,):
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001003 parents = map(lambda c, m=modname: classname(c, m), bases)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001004 result = result + '(%s)' % join(parents, ', ')
1005 result = result + '\n'
1006 elif type(entry) is type([]):
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001007 result = result + self.formattree(
1008 entry, modname, c, prefix + ' ')
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001009 return result
1010
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001011 def docmodule(self, object, name=None, mod=None):
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001012 """Produce text documentation for a given module object."""
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001013 name = object.__name__ # ignore the passed-in name
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +00001014 synop, desc = splitdoc(getdoc(object))
1015 result = self.section('NAME', name + (synop and ' - ' + synop))
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001016
1017 try:
Skip Montanaroa5616d22004-06-11 04:46:12 +00001018 all = object.__all__
1019 except AttributeError:
1020 all = None
1021
1022 try:
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001023 file = inspect.getabsfile(object)
1024 except TypeError:
1025 file = '(built-in)'
Ka-Ping Yee239432a2001-03-02 02:45:08 +00001026 result = result + self.section('FILE', file)
Skip Montanaro4997a692003-09-10 16:47:51 +00001027
1028 docloc = self.getdocloc(object)
1029 if docloc is not None:
1030 result = result + self.section('MODULE DOCS', docloc)
1031
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +00001032 if desc:
1033 result = result + self.section('DESCRIPTION', desc)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001034
1035 classes = []
1036 for key, value in inspect.getmembers(object, inspect.isclass):
Johannes Gijsbers4c11f602004-08-30 14:13:04 +00001037 # if __all__ exists, believe it. Otherwise use old heuristic.
1038 if (all is not None
1039 or (inspect.getmodule(value) or object) is object):
Skip Montanaroa5616d22004-06-11 04:46:12 +00001040 if visiblename(key, all):
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +00001041 classes.append((key, value))
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001042 funcs = []
1043 for key, value in inspect.getmembers(object, inspect.isroutine):
Johannes Gijsbers4c11f602004-08-30 14:13:04 +00001044 # if __all__ exists, believe it. Otherwise use old heuristic.
1045 if (all is not None or
1046 inspect.isbuiltin(value) or inspect.getmodule(value) is object):
Skip Montanaroa5616d22004-06-11 04:46:12 +00001047 if visiblename(key, all):
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +00001048 funcs.append((key, value))
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001049 data = []
1050 for key, value in inspect.getmembers(object, isdata):
Skip Montanaroa5616d22004-06-11 04:46:12 +00001051 if visiblename(key, all):
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001052 data.append((key, value))
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001053
Georg Brandlf2dae0e2008-01-21 21:05:49 +00001054 modpkgs = []
1055 modpkgs_names = set()
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001056 if hasattr(object, '__path__'):
Phillip J. Ebyceb30872006-04-18 00:59:55 +00001057 for importer, modname, ispkg in pkgutil.iter_modules(object.__path__):
Georg Brandlf2dae0e2008-01-21 21:05:49 +00001058 modpkgs_names.add(modname)
Phillip J. Ebyceb30872006-04-18 00:59:55 +00001059 if ispkg:
1060 modpkgs.append(modname + ' (package)')
1061 else:
1062 modpkgs.append(modname)
1063
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001064 modpkgs.sort()
1065 result = result + self.section(
1066 'PACKAGE CONTENTS', join(modpkgs, '\n'))
1067
Georg Brandlf2dae0e2008-01-21 21:05:49 +00001068 # Detect submodules as sometimes created by C extensions
1069 submodules = []
1070 for key, value in inspect.getmembers(object, inspect.ismodule):
1071 if value.__name__.startswith(name + '.') and key not in modpkgs_names:
1072 submodules.append(key)
1073 if submodules:
1074 submodules.sort()
1075 result = result + self.section(
1076 'SUBMODULES', join(submodules, '\n'))
1077
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001078 if classes:
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001079 classlist = map(lambda (key, value): value, classes)
1080 contents = [self.formattree(
1081 inspect.getclasstree(classlist, 1), name)]
1082 for key, value in classes:
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001083 contents.append(self.document(value, key, name))
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001084 result = result + self.section('CLASSES', join(contents, '\n'))
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001085
1086 if funcs:
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001087 contents = []
1088 for key, value in funcs:
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001089 contents.append(self.document(value, key, name))
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001090 result = result + self.section('FUNCTIONS', join(contents, '\n'))
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001091
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001092 if data:
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001093 contents = []
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001094 for key, value in data:
Georg Brandl8b813db2005-10-01 16:32:31 +00001095 contents.append(self.docother(value, key, name, maxlen=70))
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001096 result = result + self.section('DATA', join(contents, '\n'))
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001097
1098 if hasattr(object, '__version__'):
1099 version = str(object.__version__)
Ka-Ping Yee1d384632001-03-01 00:24:32 +00001100 if version[:11] == '$' + 'Revision: ' and version[-1:] == '$':
1101 version = strip(version[11:-1])
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001102 result = result + self.section('VERSION', version)
Ka-Ping Yee6f3f9a42001-02-27 22:42:36 +00001103 if hasattr(object, '__date__'):
1104 result = result + self.section('DATE', str(object.__date__))
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001105 if hasattr(object, '__author__'):
Ka-Ping Yee6f3f9a42001-02-27 22:42:36 +00001106 result = result + self.section('AUTHOR', str(object.__author__))
1107 if hasattr(object, '__credits__'):
1108 result = result + self.section('CREDITS', str(object.__credits__))
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001109 return result
1110
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001111 def docclass(self, object, name=None, mod=None):
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001112 """Produce text documentation for a given class object."""
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001113 realname = object.__name__
1114 name = name or realname
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001115 bases = object.__bases__
1116
Tim Petersc86f6ca2001-09-26 21:31:51 +00001117 def makename(c, m=object.__module__):
1118 return classname(c, m)
1119
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001120 if name == realname:
1121 title = 'class ' + self.bold(realname)
1122 else:
1123 title = self.bold(name) + ' = class ' + realname
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001124 if bases:
Ka-Ping Yee3bda8792001-03-23 13:17:50 +00001125 parents = map(makename, bases)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001126 title = title + '(%s)' % join(parents, ', ')
1127
1128 doc = getdoc(object)
Tim Peters28355492001-09-23 21:29:55 +00001129 contents = doc and [doc + '\n'] or []
1130 push = contents.append
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001131
Tim Petersc86f6ca2001-09-26 21:31:51 +00001132 # List the mro, if non-trivial.
Raymond Hettinger756b3f32004-01-29 06:37:52 +00001133 mro = deque(inspect.getmro(object))
Tim Petersc86f6ca2001-09-26 21:31:51 +00001134 if len(mro) > 2:
1135 push("Method resolution order:")
1136 for base in mro:
1137 push(' ' + makename(base))
1138 push('')
1139
Tim Petersf4aad8e2001-09-24 22:40:47 +00001140 # Cute little class to pump out a horizontal rule between sections.
1141 class HorizontalRule:
1142 def __init__(self):
1143 self.needone = 0
1144 def maybe(self):
1145 if self.needone:
1146 push('-' * 70)
1147 self.needone = 1
1148 hr = HorizontalRule()
1149
Tim Peters28355492001-09-23 21:29:55 +00001150 def spill(msg, attrs, predicate):
Tim Petersfa26f7c2001-09-24 08:05:11 +00001151 ok, attrs = _split_list(attrs, predicate)
Tim Peters28355492001-09-23 21:29:55 +00001152 if ok:
Tim Petersf4aad8e2001-09-24 22:40:47 +00001153 hr.maybe()
Tim Peters28355492001-09-23 21:29:55 +00001154 push(msg)
1155 for name, kind, homecls, value in ok:
1156 push(self.document(getattr(object, name),
1157 name, mod, object))
1158 return attrs
1159
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +00001160 def spilldescriptors(msg, attrs, predicate):
Tim Petersfa26f7c2001-09-24 08:05:11 +00001161 ok, attrs = _split_list(attrs, predicate)
Tim Peters28355492001-09-23 21:29:55 +00001162 if ok:
Tim Petersf4aad8e2001-09-24 22:40:47 +00001163 hr.maybe()
Tim Peters28355492001-09-23 21:29:55 +00001164 push(msg)
1165 for name, kind, homecls, value in ok:
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +00001166 push(self._docdescriptor(name, value, mod))
Tim Peters28355492001-09-23 21:29:55 +00001167 return attrs
Tim Petersb47879b2001-09-24 04:47:19 +00001168
Tim Petersfa26f7c2001-09-24 08:05:11 +00001169 def spilldata(msg, attrs, predicate):
1170 ok, attrs = _split_list(attrs, predicate)
Tim Peters28355492001-09-23 21:29:55 +00001171 if ok:
Tim Petersf4aad8e2001-09-24 22:40:47 +00001172 hr.maybe()
Tim Peters28355492001-09-23 21:29:55 +00001173 push(msg)
1174 for name, kind, homecls, value in ok:
Martin v. Löwise59e2ba2003-05-03 09:09:02 +00001175 if callable(value) or inspect.isdatadescriptor(value):
Ka-Ping Yeebba6acc2005-02-19 22:58:26 +00001176 doc = getdoc(value)
Guido van Rossum5e355b22002-05-21 20:56:15 +00001177 else:
1178 doc = None
Tim Peters28355492001-09-23 21:29:55 +00001179 push(self.docother(getattr(object, name),
Georg Brandl8b813db2005-10-01 16:32:31 +00001180 name, mod, maxlen=70, doc=doc) + '\n')
Tim Peters28355492001-09-23 21:29:55 +00001181 return attrs
1182
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +00001183 attrs = filter(lambda (name, kind, cls, value): visiblename(name),
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +00001184 classify_class_attrs(object))
Tim Petersfa26f7c2001-09-24 08:05:11 +00001185 while attrs:
Tim Peters351e3622001-09-27 03:29:51 +00001186 if mro:
Raymond Hettinger756b3f32004-01-29 06:37:52 +00001187 thisclass = mro.popleft()
Tim Peters351e3622001-09-27 03:29:51 +00001188 else:
1189 thisclass = attrs[0][2]
Tim Petersfa26f7c2001-09-24 08:05:11 +00001190 attrs, inherited = _split_list(attrs, lambda t: t[2] is thisclass)
1191
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +00001192 if thisclass is __builtin__.object:
1193 attrs = inherited
1194 continue
1195 elif thisclass is object:
Tim Peters28355492001-09-23 21:29:55 +00001196 tag = "defined here"
1197 else:
Tim Petersfa26f7c2001-09-24 08:05:11 +00001198 tag = "inherited from %s" % classname(thisclass,
1199 object.__module__)
Tim Peters28355492001-09-23 21:29:55 +00001200
1201 # Sort attrs by name.
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +00001202 attrs.sort()
Tim Peters28355492001-09-23 21:29:55 +00001203
1204 # Pump out the attrs, segregated by kind.
Tim Petersf4aad8e2001-09-24 22:40:47 +00001205 attrs = spill("Methods %s:\n" % tag, attrs,
Tim Peters28355492001-09-23 21:29:55 +00001206 lambda t: t[1] == 'method')
Tim Petersf4aad8e2001-09-24 22:40:47 +00001207 attrs = spill("Class methods %s:\n" % tag, attrs,
Tim Peters28355492001-09-23 21:29:55 +00001208 lambda t: t[1] == 'class method')
Tim Petersf4aad8e2001-09-24 22:40:47 +00001209 attrs = spill("Static methods %s:\n" % tag, attrs,
Tim Peters28355492001-09-23 21:29:55 +00001210 lambda t: t[1] == 'static method')
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +00001211 attrs = spilldescriptors("Data descriptors %s:\n" % tag, attrs,
1212 lambda t: t[1] == 'data descriptor')
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +00001213 attrs = spilldata("Data and other attributes %s:\n" % tag, attrs,
1214 lambda t: t[1] == 'data')
Tim Peters28355492001-09-23 21:29:55 +00001215 assert attrs == []
Tim Peters351e3622001-09-27 03:29:51 +00001216 attrs = inherited
Tim Peters28355492001-09-23 21:29:55 +00001217
1218 contents = '\n'.join(contents)
1219 if not contents:
1220 return title + '\n'
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001221 return title + '\n' + self.indent(rstrip(contents), ' | ') + '\n'
1222
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001223 def formatvalue(self, object):
1224 """Format an argument default value as text."""
1225 return '=' + self.repr(object)
1226
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001227 def docroutine(self, object, name=None, mod=None, cl=None):
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001228 """Produce text documentation for a function or method object."""
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001229 realname = object.__name__
1230 name = name or realname
1231 note = ''
Ka-Ping Yee3bda8792001-03-23 13:17:50 +00001232 skipdocs = 0
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001233 if inspect.ismethod(object):
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001234 imclass = object.im_class
Ka-Ping Yee3bda8792001-03-23 13:17:50 +00001235 if cl:
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001236 if imclass is not cl:
1237 note = ' from ' + classname(imclass, mod)
Ka-Ping Yee3bda8792001-03-23 13:17:50 +00001238 else:
Georg Brandl7e2b6bb2007-03-13 22:16:30 +00001239 if object.im_self is not None:
Ka-Ping Yeeb7a48302001-04-12 20:39:14 +00001240 note = ' method of %s instance' % classname(
1241 object.im_self.__class__, mod)
1242 else:
1243 note = ' unbound %s method' % classname(imclass,mod)
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001244 object = object.im_func
1245
1246 if name == realname:
1247 title = self.bold(realname)
1248 else:
Raymond Hettinger54f02222002-06-01 14:18:47 +00001249 if (cl and realname in cl.__dict__ and
Ka-Ping Yee3bda8792001-03-23 13:17:50 +00001250 cl.__dict__[realname] is object):
1251 skipdocs = 1
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001252 title = self.bold(name) + ' = ' + realname
Tim Peters4bcfa312001-09-20 06:08:24 +00001253 if inspect.isfunction(object):
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001254 args, varargs, varkw, defaults = inspect.getargspec(object)
1255 argspec = inspect.formatargspec(
1256 args, varargs, varkw, defaults, formatvalue=self.formatvalue)
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001257 if realname == '<lambda>':
Georg Brandl501dd0d2006-02-17 09:45:40 +00001258 title = self.bold(name) + ' lambda '
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001259 argspec = argspec[1:-1] # remove parentheses
Tim Peters4bcfa312001-09-20 06:08:24 +00001260 else:
1261 argspec = '(...)'
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001262 decl = title + argspec + note
1263
Ka-Ping Yee3bda8792001-03-23 13:17:50 +00001264 if skipdocs:
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001265 return decl + '\n'
Ka-Ping Yee3bda8792001-03-23 13:17:50 +00001266 else:
1267 doc = getdoc(object) or ''
1268 return decl + '\n' + (doc and rstrip(self.indent(doc)) + '\n')
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001269
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +00001270 def _docdescriptor(self, name, value, mod):
Johannes Gijsbers8de645a2004-11-07 19:16:05 +00001271 results = []
1272 push = results.append
1273
1274 if name:
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +00001275 push(self.bold(name))
1276 push('\n')
Johannes Gijsbers8de645a2004-11-07 19:16:05 +00001277 doc = getdoc(value) or ''
1278 if doc:
1279 push(self.indent(doc))
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +00001280 push('\n')
1281 return ''.join(results)
Johannes Gijsbers8de645a2004-11-07 19:16:05 +00001282
1283 def docproperty(self, object, name=None, mod=None, cl=None):
1284 """Produce text documentation for a property."""
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +00001285 return self._docdescriptor(name, object, mod)
Johannes Gijsbers8de645a2004-11-07 19:16:05 +00001286
Barry Warsaw00decd72006-07-27 23:43:15 +00001287 def docdata(self, object, name=None, mod=None, cl=None):
1288 """Produce text documentation for a data descriptor."""
1289 return self._docdescriptor(name, object, mod)
1290
Georg Brandl8b813db2005-10-01 16:32:31 +00001291 def docother(self, object, name=None, mod=None, parent=None, maxlen=None, doc=None):
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001292 """Produce text documentation for a data object."""
1293 repr = self.repr(object)
1294 if maxlen:
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001295 line = (name and name + ' = ' or '') + repr
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001296 chop = maxlen - len(line)
1297 if chop < 0: repr = repr[:chop] + '...'
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001298 line = (name and self.bold(name) + ' = ' or '') + repr
Tim Peters28355492001-09-23 21:29:55 +00001299 if doc is not None:
1300 line += '\n' + self.indent(str(doc))
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001301 return line
1302
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001303# --------------------------------------------------------- user interfaces
1304
1305def pager(text):
1306 """The first time this is called, determine what kind of pager to use."""
1307 global pager
1308 pager = getpager()
1309 pager(text)
1310
1311def getpager():
1312 """Decide what method to use for paging through text."""
1313 if type(sys.stdout) is not types.FileType:
1314 return plainpager
1315 if not sys.stdin.isatty() or not sys.stdout.isatty():
1316 return plainpager
Raymond Hettinger54f02222002-06-01 14:18:47 +00001317 if 'PAGER' in os.environ:
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +00001318 if sys.platform == 'win32': # pipes completely broken in Windows
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001319 return lambda text: tempfilepager(plain(text), os.environ['PAGER'])
Raymond Hettingerdbecd932005-02-06 06:57:08 +00001320 elif os.environ.get('TERM') in ('dumb', 'emacs'):
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001321 return lambda text: pipepager(plain(text), os.environ['PAGER'])
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +00001322 else:
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001323 return lambda text: pipepager(text, os.environ['PAGER'])
Ka-Ping Yeea487e4e2005-11-05 04:49:18 +00001324 if os.environ.get('TERM') in ('dumb', 'emacs'):
1325 return plainpager
Andrew MacIntyre54e0eab2002-03-03 03:12:30 +00001326 if sys.platform == 'win32' or sys.platform.startswith('os2'):
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001327 return lambda text: tempfilepager(plain(text), 'more <')
Skip Montanarod404bee2002-09-26 21:44:57 +00001328 if hasattr(os, 'system') and os.system('(less) 2>/dev/null') == 0:
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001329 return lambda text: pipepager(text, 'less')
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001330
1331 import tempfile
Guido van Rossum3b0a3292002-08-09 16:38:32 +00001332 (fd, filename) = tempfile.mkstemp()
1333 os.close(fd)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001334 try:
1335 if hasattr(os, 'system') and os.system('more %s' % filename) == 0:
1336 return lambda text: pipepager(text, 'more')
1337 else:
1338 return ttypager
1339 finally:
1340 os.unlink(filename)
1341
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001342def plain(text):
1343 """Remove boldface formatting from text."""
1344 return re.sub('.\b', '', text)
1345
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001346def pipepager(text, cmd):
1347 """Page through text by feeding it to another program."""
1348 pipe = os.popen(cmd, 'w')
1349 try:
1350 pipe.write(text)
1351 pipe.close()
1352 except IOError:
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001353 pass # Ignore broken pipes caused by quitting the pager program.
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001354
1355def tempfilepager(text, cmd):
1356 """Page through text by invoking a program on a temporary file."""
1357 import tempfile
Tim Peters550e4e52003-02-07 01:53:46 +00001358 filename = tempfile.mktemp()
1359 file = open(filename, 'w')
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001360 file.write(text)
1361 file.close()
1362 try:
Ka-Ping Yeec92cdf72001-03-02 05:54:35 +00001363 os.system(cmd + ' ' + filename)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001364 finally:
1365 os.unlink(filename)
1366
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001367def ttypager(text):
1368 """Page through text on a text terminal."""
1369 lines = split(plain(text), '\n')
1370 try:
1371 import tty
1372 fd = sys.stdin.fileno()
1373 old = tty.tcgetattr(fd)
1374 tty.setcbreak(fd)
1375 getchar = lambda: sys.stdin.read(1)
Ka-Ping Yee457aab22001-02-27 23:36:29 +00001376 except (ImportError, AttributeError):
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001377 tty = None
1378 getchar = lambda: sys.stdin.readline()[:-1][:1]
1379
1380 try:
1381 r = inc = os.environ.get('LINES', 25) - 1
1382 sys.stdout.write(join(lines[:inc], '\n') + '\n')
1383 while lines[r:]:
1384 sys.stdout.write('-- more --')
1385 sys.stdout.flush()
1386 c = getchar()
1387
Raymond Hettingerdbecd932005-02-06 06:57:08 +00001388 if c in ('q', 'Q'):
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001389 sys.stdout.write('\r \r')
1390 break
Raymond Hettingerdbecd932005-02-06 06:57:08 +00001391 elif c in ('\r', '\n'):
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001392 sys.stdout.write('\r \r' + lines[r] + '\n')
1393 r = r + 1
1394 continue
Raymond Hettingerdbecd932005-02-06 06:57:08 +00001395 if c in ('b', 'B', '\x1b'):
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001396 r = r - inc - inc
1397 if r < 0: r = 0
1398 sys.stdout.write('\n' + join(lines[r:r+inc], '\n') + '\n')
1399 r = r + inc
1400
1401 finally:
1402 if tty:
1403 tty.tcsetattr(fd, tty.TCSAFLUSH, old)
1404
1405def plainpager(text):
1406 """Simply print unformatted text. This is the ultimate fallback."""
1407 sys.stdout.write(plain(text))
1408
1409def describe(thing):
Ka-Ping Yee3bda8792001-03-23 13:17:50 +00001410 """Produce a short description of the given thing."""
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001411 if inspect.ismodule(thing):
1412 if thing.__name__ in sys.builtin_module_names:
1413 return 'built-in module ' + thing.__name__
1414 if hasattr(thing, '__path__'):
1415 return 'package ' + thing.__name__
1416 else:
1417 return 'module ' + thing.__name__
1418 if inspect.isbuiltin(thing):
1419 return 'built-in function ' + thing.__name__
Barry Warsaw00decd72006-07-27 23:43:15 +00001420 if inspect.isgetsetdescriptor(thing):
1421 return 'getset descriptor %s.%s.%s' % (
1422 thing.__objclass__.__module__, thing.__objclass__.__name__,
1423 thing.__name__)
1424 if inspect.ismemberdescriptor(thing):
1425 return 'member descriptor %s.%s.%s' % (
1426 thing.__objclass__.__module__, thing.__objclass__.__name__,
1427 thing.__name__)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001428 if inspect.isclass(thing):
1429 return 'class ' + thing.__name__
1430 if inspect.isfunction(thing):
1431 return 'function ' + thing.__name__
1432 if inspect.ismethod(thing):
1433 return 'method ' + thing.__name__
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001434 if type(thing) is types.InstanceType:
1435 return 'instance of ' + thing.__class__.__name__
1436 return type(thing).__name__
1437
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001438def locate(path, forceload=0):
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001439 """Locate an object by name or dotted path, importing as necessary."""
Guido van Rossum97dede02003-02-16 01:12:32 +00001440 parts = [part for part in split(path, '.') if part]
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001441 module, n = None, 0
1442 while n < len(parts):
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001443 nextmodule = safeimport(join(parts[:n+1], '.'), forceload)
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001444 if nextmodule: module, n = nextmodule, n + 1
1445 else: break
1446 if module:
1447 object = module
1448 for part in parts[n:]:
1449 try: object = getattr(object, part)
1450 except AttributeError: return None
1451 return object
1452 else:
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001453 if hasattr(__builtin__, path):
1454 return getattr(__builtin__, path)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001455
1456# --------------------------------------- interactive interpreter interface
1457
1458text = TextDoc()
1459html = HTMLDoc()
1460
Ka-Ping Yee8ef1cf32007-01-14 04:25:15 +00001461class _OldStyleClass: pass
1462_OLD_INSTANCE_TYPE = type(_OldStyleClass())
1463
Ka-Ping Yee45daeb02002-08-11 15:11:33 +00001464def resolve(thing, forceload=0):
1465 """Given an object or a path to an object, get the object and its name."""
1466 if isinstance(thing, str):
1467 object = locate(thing, forceload)
1468 if not object:
1469 raise ImportError, 'no Python documentation found for %r' % thing
1470 return object, thing
1471 else:
1472 return thing, getattr(thing, '__name__', None)
1473
Georg Brandl8441f152007-03-13 20:02:57 +00001474def render_doc(thing, title='Python Library Documentation: %s', forceload=0):
1475 """Render text documentation, given an object or a path to an object."""
1476 object, name = resolve(thing, forceload)
1477 desc = describe(object)
1478 module = inspect.getmodule(object)
1479 if name and '.' in name:
1480 desc += ' in ' + name[:name.rfind('.')]
1481 elif module and module is not object:
1482 desc += ' in module ' + module.__name__
1483 if type(object) is _OLD_INSTANCE_TYPE:
1484 # If the passed object is an instance of an old-style class,
1485 # document its available methods instead of its value.
1486 object = object.__class__
1487 elif not (inspect.ismodule(object) or
1488 inspect.isclass(object) or
1489 inspect.isroutine(object) or
1490 inspect.isgetsetdescriptor(object) or
1491 inspect.ismemberdescriptor(object) or
1492 isinstance(object, property)):
1493 # If the passed object is a piece of data or an instance,
1494 # document its available methods instead of its value.
1495 object = type(object)
1496 desc += ' object'
1497 return title % desc + '\n\n' + text.document(object, name)
1498
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001499def doc(thing, title='Python Library Documentation: %s', forceload=0):
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001500 """Display text documentation, given an object or a path to an object."""
Ka-Ping Yee45daeb02002-08-11 15:11:33 +00001501 try:
Georg Brandl8441f152007-03-13 20:02:57 +00001502 pager(render_doc(thing, title, forceload))
Ka-Ping Yee45daeb02002-08-11 15:11:33 +00001503 except (ImportError, ErrorDuringImport), value:
1504 print value
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001505
Ka-Ping Yee45daeb02002-08-11 15:11:33 +00001506def writedoc(thing, forceload=0):
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001507 """Write HTML documentation to a file in the current directory."""
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001508 try:
Ka-Ping Yee45daeb02002-08-11 15:11:33 +00001509 object, name = resolve(thing, forceload)
1510 page = html.page(describe(object), html.document(object, name))
1511 file = open(name + '.html', 'w')
1512 file.write(page)
1513 file.close()
1514 print 'wrote', name + '.html'
1515 except (ImportError, ErrorDuringImport), value:
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001516 print value
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001517
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001518def writedocs(dir, pkgpath='', done=None):
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001519 """Write out HTML documentation for all modules in a directory tree."""
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001520 if done is None: done = {}
Phillip J. Ebyceb30872006-04-18 00:59:55 +00001521 for importer, modname, ispkg in pkgutil.walk_packages([dir], pkgpath):
1522 writedoc(modname)
1523 return
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001524
1525class Helper:
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001526 keywords = {
1527 'and': 'BOOLEAN',
Georg Brandlb6a87542007-03-13 10:06:48 +00001528 'as': 'with',
Ka-Ping Yee79c009d2001-04-13 10:53:25 +00001529 'assert': ('ref/assert', ''),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001530 'break': ('ref/break', 'while for'),
1531 'class': ('ref/class', 'CLASSES SPECIALMETHODS'),
1532 'continue': ('ref/continue', 'while for'),
1533 'def': ('ref/function', ''),
1534 'del': ('ref/del', 'BASICMETHODS'),
1535 'elif': 'if',
1536 'else': ('ref/if', 'while for'),
1537 'except': 'try',
1538 'exec': ('ref/exec', ''),
1539 'finally': 'try',
1540 'for': ('ref/for', 'break continue while'),
1541 'from': 'import',
1542 'global': ('ref/global', 'NAMESPACES'),
1543 'if': ('ref/if', 'TRUTHVALUE'),
1544 'import': ('ref/import', 'MODULES'),
1545 'in': ('ref/comparisons', 'SEQUENCEMETHODS2'),
1546 'is': 'COMPARISON',
Neal Norwitz742dde42003-03-30 20:31:34 +00001547 'lambda': ('ref/lambdas', 'FUNCTIONS'),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001548 'not': 'BOOLEAN',
1549 'or': 'BOOLEAN',
Neal Norwitz742dde42003-03-30 20:31:34 +00001550 'pass': ('ref/pass', ''),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001551 'print': ('ref/print', ''),
1552 'raise': ('ref/raise', 'EXCEPTIONS'),
Ka-Ping Yee79c009d2001-04-13 10:53:25 +00001553 'return': ('ref/return', 'FUNCTIONS'),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001554 'try': ('ref/try', 'EXCEPTIONS'),
1555 'while': ('ref/while', 'break continue if TRUTHVALUE'),
Georg Brandlb6a87542007-03-13 10:06:48 +00001556 'with': ('ref/with', 'CONTEXTMANAGERS EXCEPTIONS yield'),
Tim Petersfb05c4e2002-10-30 05:21:00 +00001557 'yield': ('ref/yield', ''),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001558 }
1559
1560 topics = {
1561 'TYPES': ('ref/types', 'STRINGS UNICODE NUMBERS SEQUENCES MAPPINGS FUNCTIONS CLASSES MODULES FILES inspect'),
Ka-Ping Yee79c009d2001-04-13 10:53:25 +00001562 'STRINGS': ('ref/strings', 'str UNICODE SEQUENCES STRINGMETHODS FORMATTING TYPES'),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001563 'STRINGMETHODS': ('lib/string-methods', 'STRINGS FORMATTING'),
1564 'FORMATTING': ('lib/typesseq-strings', 'OPERATORS'),
Neal Norwitz54f871e2003-05-26 13:49:54 +00001565 'UNICODE': ('ref/strings', 'encodings unicode SEQUENCES STRINGMETHODS FORMATTING TYPES'),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001566 'NUMBERS': ('ref/numbers', 'INTEGER FLOAT COMPLEX TYPES'),
1567 'INTEGER': ('ref/integers', 'int range'),
1568 'FLOAT': ('ref/floating', 'float math'),
1569 'COMPLEX': ('ref/imaginary', 'complex cmath'),
Ka-Ping Yee79c009d2001-04-13 10:53:25 +00001570 'SEQUENCES': ('lib/typesseq', 'STRINGMETHODS FORMATTING xrange LISTS'),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001571 'MAPPINGS': 'DICTIONARIES',
1572 'FUNCTIONS': ('lib/typesfunctions', 'def TYPES'),
1573 'METHODS': ('lib/typesmethods', 'class def CLASSES TYPES'),
1574 'CODEOBJECTS': ('lib/bltin-code-objects', 'compile FUNCTIONS TYPES'),
Ka-Ping Yee79c009d2001-04-13 10:53:25 +00001575 'TYPEOBJECTS': ('lib/bltin-type-objects', 'types TYPES'),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001576 'FRAMEOBJECTS': 'TYPES',
1577 'TRACEBACKS': 'TYPES',
1578 'NONE': ('lib/bltin-null-object', ''),
1579 'ELLIPSIS': ('lib/bltin-ellipsis-object', 'SLICINGS'),
1580 'FILES': ('lib/bltin-file-objects', ''),
1581 'SPECIALATTRIBUTES': ('lib/specialattrs', ''),
1582 'CLASSES': ('ref/types', 'class SPECIALMETHODS PRIVATENAMES'),
1583 'MODULES': ('lib/typesmodules', 'import'),
1584 'PACKAGES': 'import',
1585 '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'),
1586 'OPERATORS': 'EXPRESSIONS',
1587 'PRECEDENCE': 'EXPRESSIONS',
1588 'OBJECTS': ('ref/objects', 'TYPES'),
1589 'SPECIALMETHODS': ('ref/specialnames', 'BASICMETHODS ATTRIBUTEMETHODS CALLABLEMETHODS SEQUENCEMETHODS1 MAPPINGMETHODS SEQUENCEMETHODS2 NUMBERMETHODS CLASSES'),
Ka-Ping Yee79c009d2001-04-13 10:53:25 +00001590 'BASICMETHODS': ('ref/customization', 'cmp hash repr str SPECIALMETHODS'),
1591 'ATTRIBUTEMETHODS': ('ref/attribute-access', 'ATTRIBUTES SPECIALMETHODS'),
1592 'CALLABLEMETHODS': ('ref/callable-types', 'CALLS SPECIALMETHODS'),
1593 'SEQUENCEMETHODS1': ('ref/sequence-types', 'SEQUENCES SEQUENCEMETHODS2 SPECIALMETHODS'),
1594 'SEQUENCEMETHODS2': ('ref/sequence-methods', 'SEQUENCES SEQUENCEMETHODS1 SPECIALMETHODS'),
1595 'MAPPINGMETHODS': ('ref/sequence-types', 'MAPPINGS SPECIALMETHODS'),
1596 'NUMBERMETHODS': ('ref/numeric-types', 'NUMBERS AUGMENTEDASSIGNMENT SPECIALMETHODS'),
Neal Norwitz54f871e2003-05-26 13:49:54 +00001597 'EXECUTION': ('ref/execmodel', 'NAMESPACES DYNAMICFEATURES EXCEPTIONS'),
1598 'NAMESPACES': ('ref/naming', 'global ASSIGNMENT DELETION DYNAMICFEATURES'),
1599 'DYNAMICFEATURES': ('ref/dynamic-features', ''),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001600 'SCOPING': 'NAMESPACES',
1601 'FRAMES': 'NAMESPACES',
1602 'EXCEPTIONS': ('ref/exceptions', 'try except finally raise'),
Neal Norwitz54f871e2003-05-26 13:49:54 +00001603 'COERCIONS': ('ref/coercion-rules','CONVERSIONS'),
1604 'CONVERSIONS': ('ref/conversions', 'COERCIONS'),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001605 'IDENTIFIERS': ('ref/identifiers', 'keywords SPECIALIDENTIFIERS'),
1606 'SPECIALIDENTIFIERS': ('ref/id-classes', ''),
Ka-Ping Yee79c009d2001-04-13 10:53:25 +00001607 'PRIVATENAMES': ('ref/atom-identifiers', ''),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001608 'LITERALS': ('ref/atom-literals', 'STRINGS BACKQUOTES NUMBERS TUPLELITERALS LISTLITERALS DICTIONARYLITERALS'),
1609 'TUPLES': 'SEQUENCES',
Ka-Ping Yee79c009d2001-04-13 10:53:25 +00001610 'TUPLELITERALS': ('ref/exprlists', 'TUPLES LITERALS'),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001611 'LISTS': ('lib/typesseq-mutable', 'LISTLITERALS'),
Ka-Ping Yee79c009d2001-04-13 10:53:25 +00001612 'LISTLITERALS': ('ref/lists', 'LISTS LITERALS'),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001613 'DICTIONARIES': ('lib/typesmapping', 'DICTIONARYLITERALS'),
Ka-Ping Yee79c009d2001-04-13 10:53:25 +00001614 'DICTIONARYLITERALS': ('ref/dict', 'DICTIONARIES LITERALS'),
1615 'BACKQUOTES': ('ref/string-conversions', 'repr str STRINGS LITERALS'),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001616 'ATTRIBUTES': ('ref/attribute-references', 'getattr hasattr setattr ATTRIBUTEMETHODS'),
1617 'SUBSCRIPTS': ('ref/subscriptions', 'SEQUENCEMETHODS1'),
1618 'SLICINGS': ('ref/slicings', 'SEQUENCEMETHODS2'),
1619 'CALLS': ('ref/calls', 'EXPRESSIONS'),
1620 'POWER': ('ref/power', 'EXPRESSIONS'),
1621 'UNARY': ('ref/unary', 'EXPRESSIONS'),
1622 'BINARY': ('ref/binary', 'EXPRESSIONS'),
1623 'SHIFTING': ('ref/shifting', 'EXPRESSIONS'),
1624 'BITWISE': ('ref/bitwise', 'EXPRESSIONS'),
1625 'COMPARISON': ('ref/comparisons', 'EXPRESSIONS BASICMETHODS'),
Neal Norwitzf98159c2003-02-07 20:49:40 +00001626 'BOOLEAN': ('ref/Booleans', 'EXPRESSIONS TRUTHVALUE'),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001627 'ASSERTION': 'assert',
1628 'ASSIGNMENT': ('ref/assignment', 'AUGMENTEDASSIGNMENT'),
Ka-Ping Yee79c009d2001-04-13 10:53:25 +00001629 'AUGMENTEDASSIGNMENT': ('ref/augassign', 'NUMBERMETHODS'),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001630 'DELETION': 'del',
1631 'PRINTING': 'print',
1632 'RETURNING': 'return',
1633 'IMPORTING': 'import',
1634 'CONDITIONAL': 'if',
1635 'LOOPING': ('ref/compound', 'for while break continue'),
1636 'TRUTHVALUE': ('lib/truth', 'if while and or not BASICMETHODS'),
Ka-Ping Yee79c009d2001-04-13 10:53:25 +00001637 'DEBUGGING': ('lib/module-pdb', 'pdb'),
Georg Brandlb6a87542007-03-13 10:06:48 +00001638 'CONTEXTMANAGERS': ('ref/context-managers', 'with'),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001639 }
1640
1641 def __init__(self, input, output):
1642 self.input = input
1643 self.output = output
1644 self.docdir = None
1645 execdir = os.path.dirname(sys.executable)
1646 homedir = os.environ.get('PYTHONHOME')
Georg Brandl117a05e2007-03-02 14:37:12 +00001647 join = os.path.join
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001648 for dir in [os.environ.get('PYTHONDOCS'),
1649 homedir and os.path.join(homedir, 'doc'),
Georg Brandl117a05e2007-03-02 14:37:12 +00001650 join(execdir, 'doc'), # for Windows
1651 join(sys.prefix, 'doc/python-docs-' + split(sys.version)[0]),
1652 join(sys.prefix, 'doc/python-' + split(sys.version)[0]),
1653 join(sys.prefix, 'doc/python-docs-' + sys.version[:3]),
1654 join(sys.prefix, 'doc/python-' + sys.version[:3]),
1655 join(sys.prefix, 'Resources/English.lproj/Documentation')]:
1656 if dir and os.path.isdir(join(dir, 'lib')):
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001657 self.docdir = dir
Georg Brandl117a05e2007-03-02 14:37:12 +00001658 break
1659 if dir and os.path.isdir(join(dir, 'html', 'lib')):
1660 self.docdir = join(dir, 'html')
1661 break
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001662
Ka-Ping Yee79c009d2001-04-13 10:53:25 +00001663 def __repr__(self):
Ka-Ping Yee9bc576b2001-04-13 13:57:31 +00001664 if inspect.stack()[1][3] == '?':
Ka-Ping Yee79c009d2001-04-13 10:53:25 +00001665 self()
1666 return ''
Ka-Ping Yee9bc576b2001-04-13 13:57:31 +00001667 return '<pydoc.Helper instance>'
Ka-Ping Yee79c009d2001-04-13 10:53:25 +00001668
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001669 def __call__(self, request=None):
1670 if request is not None:
1671 self.help(request)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001672 else:
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001673 self.intro()
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001674 self.interact()
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001675 self.output.write('''
Fred Drakee61967f2001-05-10 18:41:02 +00001676You are now leaving help and returning to the Python interpreter.
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001677If you want to ask for help on a particular object directly from the
1678interpreter, you can type "help(object)". Executing "help('string')"
1679has the same effect as typing a particular string at the help> prompt.
1680''')
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001681
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001682 def interact(self):
1683 self.output.write('\n')
Guido van Rossum8ca162f2002-04-07 06:36:23 +00001684 while True:
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001685 try:
Johannes Gijsberse7691d32004-08-17 13:21:53 +00001686 request = self.getline('help> ')
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001687 if not request: break
Johannes Gijsberse7691d32004-08-17 13:21:53 +00001688 except (KeyboardInterrupt, EOFError):
1689 break
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001690 request = strip(replace(request, '"', '', "'", ''))
Raymond Hettingerdbecd932005-02-06 06:57:08 +00001691 if lower(request) in ('q', 'quit'): break
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001692 self.help(request)
1693
Johannes Gijsberse7691d32004-08-17 13:21:53 +00001694 def getline(self, prompt):
1695 """Read one line, using raw_input when available."""
1696 if self.input is sys.stdin:
1697 return raw_input(prompt)
1698 else:
1699 self.output.write(prompt)
1700 self.output.flush()
1701 return self.input.readline()
1702
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001703 def help(self, request):
1704 if type(request) is type(''):
1705 if request == 'help': self.intro()
1706 elif request == 'keywords': self.listkeywords()
1707 elif request == 'topics': self.listtopics()
1708 elif request == 'modules': self.listmodules()
1709 elif request[:8] == 'modules ':
1710 self.listmodules(split(request)[1])
Raymond Hettinger54f02222002-06-01 14:18:47 +00001711 elif request in self.keywords: self.showtopic(request)
1712 elif request in self.topics: self.showtopic(request)
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001713 elif request: doc(request, 'Help on %s:')
1714 elif isinstance(request, Helper): self()
1715 else: doc(request, 'Help on %s:')
1716 self.output.write('\n')
1717
1718 def intro(self):
1719 self.output.write('''
1720Welcome to Python %s! This is the online help utility.
1721
1722If this is your first time using Python, you should definitely check out
Georg Brandl0751d1a2008-01-21 17:13:03 +00001723the tutorial on the Internet at http://docs.python.org/tutorial/.
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001724
1725Enter the name of any module, keyword, or topic to get help on writing
1726Python programs and using Python modules. To quit this help utility and
1727return to the interpreter, just type "quit".
1728
1729To get a list of available modules, keywords, or topics, type "modules",
1730"keywords", or "topics". Each module also comes with a one-line summary
1731of what it does; to list the modules whose summaries contain a given word
1732such as "spam", type "modules spam".
1733''' % sys.version[:3])
1734
1735 def list(self, items, columns=4, width=80):
1736 items = items[:]
1737 items.sort()
1738 colw = width / columns
1739 rows = (len(items) + columns - 1) / columns
1740 for row in range(rows):
1741 for col in range(columns):
1742 i = col * rows + row
1743 if i < len(items):
1744 self.output.write(items[i])
1745 if col < columns - 1:
1746 self.output.write(' ' + ' ' * (colw-1 - len(items[i])))
1747 self.output.write('\n')
1748
1749 def listkeywords(self):
1750 self.output.write('''
1751Here is a list of the Python keywords. Enter any keyword to get more help.
1752
1753''')
1754 self.list(self.keywords.keys())
1755
1756 def listtopics(self):
1757 self.output.write('''
1758Here is a list of available topics. Enter any topic name to get more help.
1759
1760''')
1761 self.list(self.topics.keys())
1762
1763 def showtopic(self, topic):
1764 if not self.docdir:
1765 self.output.write('''
1766Sorry, topic and keyword documentation is not available because the Python
1767HTML documentation files could not be found. If you have installed them,
1768please set the environment variable PYTHONDOCS to indicate their location.
Andrew M. Kuchlingc911e912006-12-19 15:18:12 +00001769
1770On the Microsoft Windows operating system, the files can be built by
1771running "hh -decompile . PythonNN.chm" in the C:\PythonNN\Doc> directory.
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001772''')
1773 return
1774 target = self.topics.get(topic, self.keywords.get(topic))
1775 if not target:
1776 self.output.write('no documentation found for %s\n' % repr(topic))
1777 return
1778 if type(target) is type(''):
1779 return self.showtopic(target)
1780
1781 filename, xrefs = target
1782 filename = self.docdir + '/' + filename + '.html'
1783 try:
1784 file = open(filename)
1785 except:
1786 self.output.write('could not read docs from %s\n' % filename)
1787 return
1788
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001789 divpat = re.compile('<div[^>]*navigat.*?</div.*?>', re.I | re.S)
1790 addrpat = re.compile('<address.*?>.*?</address.*?>', re.I | re.S)
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001791 document = re.sub(addrpat, '', re.sub(divpat, '', file.read()))
1792 file.close()
1793
1794 import htmllib, formatter, StringIO
1795 buffer = StringIO.StringIO()
1796 parser = htmllib.HTMLParser(
1797 formatter.AbstractFormatter(formatter.DumbWriter(buffer)))
1798 parser.start_table = parser.do_p
1799 parser.end_table = lambda parser=parser: parser.do_p({})
1800 parser.start_tr = parser.do_br
1801 parser.start_td = parser.start_th = lambda a, b=buffer: b.write('\t')
1802 parser.feed(document)
1803 buffer = replace(buffer.getvalue(), '\xa0', ' ', '\n', '\n ')
1804 pager(' ' + strip(buffer) + '\n')
Ka-Ping Yeeda793892001-04-13 11:02:51 +00001805 if xrefs:
1806 buffer = StringIO.StringIO()
1807 formatter.DumbWriter(buffer).send_flowing_data(
1808 'Related help topics: ' + join(split(xrefs), ', ') + '\n')
1809 self.output.write('\n%s\n' % buffer.getvalue())
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001810
1811 def listmodules(self, key=''):
1812 if key:
1813 self.output.write('''
1814Here is a list of matching modules. Enter any module name to get more help.
1815
1816''')
1817 apropos(key)
1818 else:
1819 self.output.write('''
1820Please wait a moment while I gather a list of all available modules...
1821
1822''')
1823 modules = {}
1824 def callback(path, modname, desc, modules=modules):
1825 if modname and modname[-9:] == '.__init__':
1826 modname = modname[:-9] + ' (package)'
1827 if find(modname, '.') < 0:
1828 modules[modname] = 1
Ka-Ping Yee9e0f1162008-01-13 11:25:13 +00001829 def onerror(modname):
1830 callback(None, modname, None)
1831 ModuleScanner().run(callback, onerror=onerror)
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001832 self.list(modules.keys())
1833 self.output.write('''
1834Enter any module name to get more help. Or, type "modules spam" to search
1835for modules whose descriptions contain the word "spam".
1836''')
1837
1838help = Helper(sys.stdin, sys.stdout)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001839
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001840class Scanner:
1841 """A generic tree iterator."""
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001842 def __init__(self, roots, children, descendp):
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001843 self.roots = roots[:]
1844 self.state = []
1845 self.children = children
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001846 self.descendp = descendp
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001847
1848 def next(self):
1849 if not self.state:
1850 if not self.roots:
1851 return None
1852 root = self.roots.pop(0)
1853 self.state = [(root, self.children(root))]
1854 node, children = self.state[-1]
1855 if not children:
1856 self.state.pop()
1857 return self.next()
1858 child = children.pop(0)
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001859 if self.descendp(child):
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001860 self.state.append((child, self.children(child)))
1861 return child
1862
Phillip J. Ebyceb30872006-04-18 00:59:55 +00001863
1864class ModuleScanner:
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001865 """An interruptible scanner that searches module synopses."""
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001866
Ka-Ping Yee9e0f1162008-01-13 11:25:13 +00001867 def run(self, callback, key=None, completer=None, onerror=None):
Ka-Ping Yee66246962001-04-12 11:59:50 +00001868 if key: key = lower(key)
Guido van Rossum8ca162f2002-04-07 06:36:23 +00001869 self.quit = False
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001870 seen = {}
1871
1872 for modname in sys.builtin_module_names:
Ka-Ping Yee239432a2001-03-02 02:45:08 +00001873 if modname != '__main__':
1874 seen[modname] = 1
Ka-Ping Yee66246962001-04-12 11:59:50 +00001875 if key is None:
1876 callback(None, modname, '')
1877 else:
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001878 desc = split(__import__(modname).__doc__ or '', '\n')[0]
Ka-Ping Yee66246962001-04-12 11:59:50 +00001879 if find(lower(modname + ' - ' + desc), key) >= 0:
1880 callback(None, modname, desc)
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001881
Ka-Ping Yee9e0f1162008-01-13 11:25:13 +00001882 for importer, modname, ispkg in pkgutil.walk_packages(onerror=onerror):
Phillip J. Ebyceb30872006-04-18 00:59:55 +00001883 if self.quit:
1884 break
1885 if key is None:
1886 callback(None, modname, '')
1887 else:
1888 loader = importer.find_module(modname)
1889 if hasattr(loader,'get_source'):
1890 import StringIO
1891 desc = source_synopsis(
1892 StringIO.StringIO(loader.get_source(modname))
1893 ) or ''
1894 if hasattr(loader,'get_filename'):
1895 path = loader.get_filename(modname)
Ka-Ping Yee66246962001-04-12 11:59:50 +00001896 else:
Phillip J. Ebyceb30872006-04-18 00:59:55 +00001897 path = None
1898 else:
1899 module = loader.load_module(modname)
1900 desc = (module.__doc__ or '').splitlines()[0]
1901 path = getattr(module,'__file__',None)
1902 if find(lower(modname + ' - ' + desc), key) >= 0:
1903 callback(path, modname, desc)
1904
1905 if completer:
1906 completer()
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001907
1908def apropos(key):
1909 """Print all the one-line module summaries that contain a substring."""
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001910 def callback(path, modname, desc):
1911 if modname[-9:] == '.__init__':
1912 modname = modname[:-9] + ' (package)'
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +00001913 print modname, desc and '- ' + desc
1914 try: import warnings
1915 except ImportError: pass
1916 else: warnings.filterwarnings('ignore') # ignore problems during import
Ka-Ping Yee66246962001-04-12 11:59:50 +00001917 ModuleScanner().run(callback, key)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001918
1919# --------------------------------------------------- web browser interface
1920
Ka-Ping Yee66246962001-04-12 11:59:50 +00001921def serve(port, callback=None, completer=None):
Ka-Ping Yeefd540692001-04-12 12:54:36 +00001922 import BaseHTTPServer, mimetools, select
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001923
1924 # Patch up mimetools.Message so it doesn't break if rfc822 is reloaded.
1925 class Message(mimetools.Message):
1926 def __init__(self, fp, seekable=1):
1927 Message = self.__class__
1928 Message.__bases__[0].__bases__[0].__init__(self, fp, seekable)
1929 self.encodingheader = self.getheader('content-transfer-encoding')
1930 self.typeheader = self.getheader('content-type')
1931 self.parsetype()
1932 self.parseplist()
1933
1934 class DocHandler(BaseHTTPServer.BaseHTTPRequestHandler):
1935 def send_document(self, title, contents):
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001936 try:
1937 self.send_response(200)
1938 self.send_header('Content-Type', 'text/html')
1939 self.end_headers()
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001940 self.wfile.write(html.page(title, contents))
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001941 except IOError: pass
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001942
1943 def do_GET(self):
1944 path = self.path
1945 if path[-5:] == '.html': path = path[:-5]
1946 if path[:1] == '/': path = path[1:]
1947 if path and path != '.':
1948 try:
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001949 obj = locate(path, forceload=1)
Ka-Ping Yee3bda8792001-03-23 13:17:50 +00001950 except ErrorDuringImport, value:
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001951 self.send_document(path, html.escape(str(value)))
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001952 return
Ka-Ping Yee3bda8792001-03-23 13:17:50 +00001953 if obj:
1954 self.send_document(describe(obj), html.document(obj, path))
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001955 else:
1956 self.send_document(path,
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001957'no Python documentation found for %s' % repr(path))
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001958 else:
1959 heading = html.heading(
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001960'<big><big><strong>Python: Index of Modules</strong></big></big>',
1961'#ffffff', '#7799ee')
Ka-Ping Yee239432a2001-03-02 02:45:08 +00001962 def bltinlink(name):
1963 return '<a href="%s.html">%s</a>' % (name, name)
Ka-Ping Yee3bda8792001-03-23 13:17:50 +00001964 names = filter(lambda x: x != '__main__',
1965 sys.builtin_module_names)
Ka-Ping Yee239432a2001-03-02 02:45:08 +00001966 contents = html.multicolumn(names, bltinlink)
1967 indices = ['<p>' + html.bigsection(
1968 'Built-in Modules', '#ffffff', '#ee77aa', contents)]
1969
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001970 seen = {}
Phillip J. Ebyceb30872006-04-18 00:59:55 +00001971 for dir in sys.path:
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001972 indices.append(html.index(dir, seen))
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001973 contents = heading + join(indices) + '''<p align=right>
Tim Peters2306d242001-09-25 03:18:32 +00001974<font color="#909090" face="helvetica, arial"><strong>
1975pydoc</strong> by Ka-Ping Yee &lt;ping@lfw.org&gt;</font>'''
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001976 self.send_document('Index of Modules', contents)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001977
1978 def log_message(self, *args): pass
1979
Ka-Ping Yeefd540692001-04-12 12:54:36 +00001980 class DocServer(BaseHTTPServer.HTTPServer):
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001981 def __init__(self, port, callback):
Ka-Ping Yeedb8ed152001-03-02 05:58:17 +00001982 host = (sys.platform == 'mac') and '127.0.0.1' or 'localhost'
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001983 self.address = ('', port)
Ka-Ping Yeedb8ed152001-03-02 05:58:17 +00001984 self.url = 'http://%s:%d/' % (host, port)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001985 self.callback = callback
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001986 self.base.__init__(self, self.address, self.handler)
1987
1988 def serve_until_quit(self):
1989 import select
Guido van Rossum8ca162f2002-04-07 06:36:23 +00001990 self.quit = False
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001991 while not self.quit:
1992 rd, wr, ex = select.select([self.socket.fileno()], [], [], 1)
1993 if rd: self.handle_request()
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001994
1995 def server_activate(self):
1996 self.base.server_activate(self)
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001997 if self.callback: self.callback(self)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001998
1999 DocServer.base = BaseHTTPServer.HTTPServer
2000 DocServer.handler = DocHandler
2001 DocHandler.MessageClass = Message
2002 try:
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00002003 try:
2004 DocServer(port, callback).serve_until_quit()
2005 except (KeyboardInterrupt, select.error):
2006 pass
2007 finally:
Ka-Ping Yee66246962001-04-12 11:59:50 +00002008 if completer: completer()
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002009
2010# ----------------------------------------------------- graphical interface
2011
2012def gui():
2013 """Graphical interface (starts web server and pops up a control window)."""
2014 class GUI:
2015 def __init__(self, window, port=7464):
2016 self.window = window
2017 self.server = None
2018 self.scanner = None
2019
2020 import Tkinter
2021 self.server_frm = Tkinter.Frame(window)
2022 self.title_lbl = Tkinter.Label(self.server_frm,
2023 text='Starting server...\n ')
2024 self.open_btn = Tkinter.Button(self.server_frm,
2025 text='open browser', command=self.open, state='disabled')
2026 self.quit_btn = Tkinter.Button(self.server_frm,
2027 text='quit serving', command=self.quit, state='disabled')
2028
2029 self.search_frm = Tkinter.Frame(window)
2030 self.search_lbl = Tkinter.Label(self.search_frm, text='Search for')
2031 self.search_ent = Tkinter.Entry(self.search_frm)
2032 self.search_ent.bind('<Return>', self.search)
2033 self.stop_btn = Tkinter.Button(self.search_frm,
2034 text='stop', pady=0, command=self.stop, state='disabled')
2035 if sys.platform == 'win32':
Ka-Ping Yeec92cdf72001-03-02 05:54:35 +00002036 # Trying to hide and show this button crashes under Windows.
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002037 self.stop_btn.pack(side='right')
2038
2039 self.window.title('pydoc')
2040 self.window.protocol('WM_DELETE_WINDOW', self.quit)
2041 self.title_lbl.pack(side='top', fill='x')
2042 self.open_btn.pack(side='left', fill='x', expand=1)
2043 self.quit_btn.pack(side='right', fill='x', expand=1)
2044 self.server_frm.pack(side='top', fill='x')
2045
2046 self.search_lbl.pack(side='left')
2047 self.search_ent.pack(side='right', fill='x', expand=1)
2048 self.search_frm.pack(side='top', fill='x')
2049 self.search_ent.focus_set()
2050
Ka-Ping Yeec92cdf72001-03-02 05:54:35 +00002051 font = ('helvetica', sys.platform == 'win32' and 8 or 10)
Ka-Ping Yee239432a2001-03-02 02:45:08 +00002052 self.result_lst = Tkinter.Listbox(window, font=font, height=6)
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002053 self.result_lst.bind('<Button-1>', self.select)
2054 self.result_lst.bind('<Double-Button-1>', self.goto)
2055 self.result_scr = Tkinter.Scrollbar(window,
2056 orient='vertical', command=self.result_lst.yview)
2057 self.result_lst.config(yscrollcommand=self.result_scr.set)
2058
2059 self.result_frm = Tkinter.Frame(window)
2060 self.goto_btn = Tkinter.Button(self.result_frm,
2061 text='go to selected', command=self.goto)
2062 self.hide_btn = Tkinter.Button(self.result_frm,
2063 text='hide results', command=self.hide)
2064 self.goto_btn.pack(side='left', fill='x', expand=1)
2065 self.hide_btn.pack(side='right', fill='x', expand=1)
2066
2067 self.window.update()
2068 self.minwidth = self.window.winfo_width()
2069 self.minheight = self.window.winfo_height()
2070 self.bigminheight = (self.server_frm.winfo_reqheight() +
2071 self.search_frm.winfo_reqheight() +
2072 self.result_lst.winfo_reqheight() +
2073 self.result_frm.winfo_reqheight())
2074 self.bigwidth, self.bigheight = self.minwidth, self.bigminheight
2075 self.expanded = 0
2076 self.window.wm_geometry('%dx%d' % (self.minwidth, self.minheight))
2077 self.window.wm_minsize(self.minwidth, self.minheight)
Martin v. Löwis5b26abb2002-12-28 09:23:09 +00002078 self.window.tk.willdispatch()
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002079
2080 import threading
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00002081 threading.Thread(
2082 target=serve, args=(port, self.ready, self.quit)).start()
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002083
2084 def ready(self, server):
2085 self.server = server
2086 self.title_lbl.config(
2087 text='Python documentation server at\n' + server.url)
2088 self.open_btn.config(state='normal')
2089 self.quit_btn.config(state='normal')
2090
Ka-Ping Yee239432a2001-03-02 02:45:08 +00002091 def open(self, event=None, url=None):
2092 url = url or self.server.url
2093 try:
2094 import webbrowser
2095 webbrowser.open(url)
2096 except ImportError: # pre-webbrowser.py compatibility
Ka-Ping Yeec92cdf72001-03-02 05:54:35 +00002097 if sys.platform == 'win32':
Ka-Ping Yee239432a2001-03-02 02:45:08 +00002098 os.system('start "%s"' % url)
2099 elif sys.platform == 'mac':
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +00002100 try: import ic
Ka-Ping Yee239432a2001-03-02 02:45:08 +00002101 except ImportError: pass
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +00002102 else: ic.launchurl(url)
Ka-Ping Yee239432a2001-03-02 02:45:08 +00002103 else:
2104 rc = os.system('netscape -remote "openURL(%s)" &' % url)
2105 if rc: os.system('netscape "%s" &' % url)
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002106
2107 def quit(self, event=None):
2108 if self.server:
2109 self.server.quit = 1
2110 self.window.quit()
2111
2112 def search(self, event=None):
2113 key = self.search_ent.get()
2114 self.stop_btn.pack(side='right')
2115 self.stop_btn.config(state='normal')
2116 self.search_lbl.config(text='Searching for "%s"...' % key)
2117 self.search_ent.forget()
2118 self.search_lbl.pack(side='left')
2119 self.result_lst.delete(0, 'end')
2120 self.goto_btn.config(state='disabled')
2121 self.expand()
2122
2123 import threading
2124 if self.scanner:
2125 self.scanner.quit = 1
2126 self.scanner = ModuleScanner()
2127 threading.Thread(target=self.scanner.run,
Ka-Ping Yee6dcfa382001-04-12 20:27:31 +00002128 args=(self.update, key, self.done)).start()
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002129
2130 def update(self, path, modname, desc):
2131 if modname[-9:] == '.__init__':
2132 modname = modname[:-9] + ' (package)'
2133 self.result_lst.insert('end',
2134 modname + ' - ' + (desc or '(no description)'))
2135
2136 def stop(self, event=None):
2137 if self.scanner:
2138 self.scanner.quit = 1
2139 self.scanner = None
2140
2141 def done(self):
2142 self.scanner = None
2143 self.search_lbl.config(text='Search for')
2144 self.search_lbl.pack(side='left')
2145 self.search_ent.pack(side='right', fill='x', expand=1)
2146 if sys.platform != 'win32': self.stop_btn.forget()
2147 self.stop_btn.config(state='disabled')
2148
2149 def select(self, event=None):
2150 self.goto_btn.config(state='normal')
2151
2152 def goto(self, event=None):
2153 selection = self.result_lst.curselection()
2154 if selection:
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002155 modname = split(self.result_lst.get(selection[0]))[0]
Ka-Ping Yee239432a2001-03-02 02:45:08 +00002156 self.open(url=self.server.url + modname + '.html')
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002157
2158 def collapse(self):
2159 if not self.expanded: return
2160 self.result_frm.forget()
2161 self.result_scr.forget()
2162 self.result_lst.forget()
2163 self.bigwidth = self.window.winfo_width()
2164 self.bigheight = self.window.winfo_height()
2165 self.window.wm_geometry('%dx%d' % (self.minwidth, self.minheight))
2166 self.window.wm_minsize(self.minwidth, self.minheight)
2167 self.expanded = 0
2168
2169 def expand(self):
2170 if self.expanded: return
2171 self.result_frm.pack(side='bottom', fill='x')
2172 self.result_scr.pack(side='right', fill='y')
2173 self.result_lst.pack(side='top', fill='both', expand=1)
2174 self.window.wm_geometry('%dx%d' % (self.bigwidth, self.bigheight))
2175 self.window.wm_minsize(self.minwidth, self.bigminheight)
2176 self.expanded = 1
2177
2178 def hide(self, event=None):
2179 self.stop()
2180 self.collapse()
2181
2182 import Tkinter
2183 try:
Martin v. Löwise09bd932004-08-22 16:13:26 +00002184 root = Tkinter.Tk()
2185 # Tk will crash if pythonw.exe has an XP .manifest
2186 # file and the root has is not destroyed explicitly.
2187 # If the problem is ever fixed in Tk, the explicit
2188 # destroy can go.
2189 try:
2190 gui = GUI(root)
2191 root.mainloop()
2192 finally:
2193 root.destroy()
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002194 except KeyboardInterrupt:
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002195 pass
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002196
2197# -------------------------------------------------- command-line interface
2198
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00002199def ispath(x):
Ka-Ping Yee45daeb02002-08-11 15:11:33 +00002200 return isinstance(x, str) and find(x, os.sep) >= 0
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00002201
Ka-Ping Yee1d384632001-03-01 00:24:32 +00002202def cli():
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002203 """Command-line interface (looks at sys.argv to decide what to do)."""
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002204 import getopt
2205 class BadUsage: pass
2206
Ka-Ping Yee3bda8792001-03-23 13:17:50 +00002207 # Scripts don't get the current directory in their path by default.
Ka-Ping Yeef78a81b2001-03-27 08:13:42 +00002208 scriptdir = os.path.dirname(sys.argv[0])
2209 if scriptdir in sys.path:
2210 sys.path.remove(scriptdir)
Ka-Ping Yee3bda8792001-03-23 13:17:50 +00002211 sys.path.insert(0, '.')
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002212
Ka-Ping Yee3bda8792001-03-23 13:17:50 +00002213 try:
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002214 opts, args = getopt.getopt(sys.argv[1:], 'gk:p:w')
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002215 writing = 0
2216
2217 for opt, val in opts:
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002218 if opt == '-g':
2219 gui()
2220 return
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002221 if opt == '-k':
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002222 apropos(val)
2223 return
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002224 if opt == '-p':
2225 try:
2226 port = int(val)
2227 except ValueError:
2228 raise BadUsage
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002229 def ready(server):
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00002230 print 'pydoc server ready at %s' % server.url
2231 def stopped():
2232 print 'pydoc server stopped'
2233 serve(port, ready, stopped)
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002234 return
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002235 if opt == '-w':
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002236 writing = 1
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002237
2238 if not args: raise BadUsage
2239 for arg in args:
Ka-Ping Yee45daeb02002-08-11 15:11:33 +00002240 if ispath(arg) and not os.path.exists(arg):
2241 print 'file %r does not exist' % arg
2242 break
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002243 try:
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00002244 if ispath(arg) and os.path.isfile(arg):
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002245 arg = importfile(arg)
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00002246 if writing:
2247 if ispath(arg) and os.path.isdir(arg):
2248 writedocs(arg)
2249 else:
2250 writedoc(arg)
2251 else:
Martin v. Löwisb8c084e2003-06-14 09:03:46 +00002252 help.help(arg)
Ka-Ping Yee3bda8792001-03-23 13:17:50 +00002253 except ErrorDuringImport, value:
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00002254 print value
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002255
2256 except (getopt.error, BadUsage):
Martin v. Löwisf9b08b82003-10-31 13:05:21 +00002257 cmd = os.path.basename(sys.argv[0])
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002258 print """pydoc - the Python documentation tool
2259
2260%s <name> ...
2261 Show text documentation on something. <name> may be the name of a
Martin v. Löwisb8c084e2003-06-14 09:03:46 +00002262 Python keyword, topic, function, module, or package, or a dotted
2263 reference to a class or function within a module or module in a
2264 package. If <name> contains a '%s', it is used as the path to a
2265 Python source file to document. If name is 'keywords', 'topics',
2266 or 'modules', a listing of these things is displayed.
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002267
2268%s -k <keyword>
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002269 Search for a keyword in the synopsis lines of all available modules.
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002270
2271%s -p <port>
2272 Start an HTTP server on the given port on the local machine.
2273
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002274%s -g
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +00002275 Pop up a graphical interface for finding and serving documentation.
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002276
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002277%s -w <name> ...
2278 Write out the HTML documentation for a module to a file in the current
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +00002279 directory. If <name> contains a '%s', it is treated as a filename; if
2280 it names a directory, documentation is written for all the contents.
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002281""" % (cmd, os.sep, cmd, cmd, cmd, cmd, os.sep)
Ka-Ping Yee1d384632001-03-01 00:24:32 +00002282
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002283if __name__ == '__main__': cli()