blob: 9551982315cf8e70963580ba89db5dcba0ba31a7 [file] [log] [blame]
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001#!/usr/bin/env python
Guido van Rossumfce538c2002-08-06 17:29:38 +00002# -*- coding: Latin-1 -*-
Ka-Ping Yee1d384632001-03-01 00:24:32 +00003"""Generate Python documentation in HTML or text for interactive use.
Ka-Ping Yeedd175342001-02-27 14:43:46 +00004
Ka-Ping Yeedd175342001-02-27 14:43:46 +00005In the Python interpreter, do "from pydoc import help" to provide online
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00006help. Calling help(thing) on a Python object documents the object.
7
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00008Or, at the shell command line outside of Python:
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00009
Ka-Ping Yee37f7b382001-03-23 00:12:53 +000010Run "pydoc <name>" to show documentation on something. <name> may be
11the name of a function, module, package, or a dotted reference to a
12class or function within a module or module in a package. If the
13argument contains a path segment delimiter (e.g. slash on Unix,
14backslash on Windows) it is treated as the path to a Python source file.
Ka-Ping Yee66efbc72001-03-01 13:55:20 +000015
Ka-Ping Yee37f7b382001-03-23 00:12:53 +000016Run "pydoc -k <keyword>" to search for a keyword in the synopsis lines
17of all available modules.
Ka-Ping Yee66efbc72001-03-01 13:55:20 +000018
Ka-Ping Yee37f7b382001-03-23 00:12:53 +000019Run "pydoc -p <port>" to start an HTTP server on a given port on the
20local machine to generate documentation web pages.
Ka-Ping Yee66efbc72001-03-01 13:55:20 +000021
Ka-Ping Yee37f7b382001-03-23 00:12:53 +000022For platforms without a command line, "pydoc -g" starts the HTTP server
23and also pops up a little window for controlling it.
24
25Run "pydoc -w <name>" to write out the HTML documentation for a module
26to a file named "<name>.html".
Skip Montanaro4997a692003-09-10 16:47:51 +000027
28Module docs for core modules are assumed to be in
29
30 http://www.python.org/doc/current/lib/
31
32This can be overridden by setting the PYTHONDOCS environment variable
33to a different URL or to a local directory containing the Library
34Reference Manual pages.
Ka-Ping Yee66efbc72001-03-01 13:55:20 +000035"""
Ka-Ping Yeedd175342001-02-27 14:43:46 +000036
37__author__ = "Ka-Ping Yee <ping@lfw.org>"
Ka-Ping Yee6f3f9a42001-02-27 22:42:36 +000038__date__ = "26 February 2001"
Jeremy Hylton3e0055f2005-10-20 19:59:25 +000039
Ka-Ping Yee09d7d9a2001-02-27 22:43:48 +000040__version__ = "$Revision$"
Martin v. Löwis6fe8f192004-11-14 10:21:04 +000041__credits__ = """Guido van Rossum, for an excellent programming language.
Ka-Ping Yee5e2b1732001-02-27 23:35:09 +000042Tommy Burnette, the original creator of manpy.
Ka-Ping Yee6f3f9a42001-02-27 22:42:36 +000043Paul Prescod, for all his work on onlinehelp.
44Richard Chamberlain, for the first implementation of textdoc.
Raymond Hettingered2dbe32005-01-01 07:51:01 +000045"""
Ka-Ping Yeedd175342001-02-27 14:43:46 +000046
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +000047# Known bugs that can't be fixed here:
48# - imp.load_module() cannot be prevented from clobbering existing
49# loaded modules, so calling synopsis() on a binary module file
50# changes the contents of any existing module with the same name.
51# - If the __file__ attribute on a module is a relative path and
52# the current directory is changed with os.chdir(), an incorrect
53# path will be displayed.
Ka-Ping Yee66efbc72001-03-01 13:55:20 +000054
Thomas Wouters49fd7fa2006-04-21 10:40:58 +000055import sys, imp, os, re, types, inspect, __builtin__, pkgutil
Ka-Ping Yeedd175342001-02-27 14:43:46 +000056from repr import Repr
Ka-Ping Yee3bda8792001-03-23 13:17:50 +000057from string import expandtabs, find, join, lower, split, strip, rfind, rstrip
Thomas Wouters49fd7fa2006-04-21 10:40:58 +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
Thomas Wouters49fd7fa2006-04-21 10:40:58 +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
Thomas Wouters49fd7fa2006-04-21 10:40:58 +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
Thomas Wouters49fd7fa2006-04-21 10:40:58 +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.
Thomas Wouters0e3f5912006-08-11 14:57:12 +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
Thomas Wouters0e3f5912006-08-11 14:57:12 +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",
349 "http://www.python.org/doc/current/lib")
350 basedir = os.path.join(sys.exec_prefix, "lib",
351 "python"+sys.version[0:3])
352 if (isinstance(object, type(os)) and
353 (object.__name__ in ('errno', 'exceptions', 'gc', 'imp',
354 'marshal', 'posix', 'signal', 'sys',
355 'thread', 'zipimport') or
356 (file.startswith(basedir) and
357 not file.startswith(os.path.join(basedir, 'site-packages'))))):
Skip Montanarof2134842004-06-07 02:40:05 +0000358 htmlfile = "module-%s.html" % object.__name__
Skip Montanaro4997a692003-09-10 16:47:51 +0000359 if docloc.startswith("http://"):
Skip Montanarof2134842004-06-07 02:40:05 +0000360 docloc = "%s/%s" % (docloc.rstrip("/"), htmlfile)
Skip Montanaro4997a692003-09-10 16:47:51 +0000361 else:
Skip Montanarof2134842004-06-07 02:40:05 +0000362 docloc = os.path.join(docloc, htmlfile)
Skip Montanaro4997a692003-09-10 16:47:51 +0000363 else:
364 docloc = None
365 return docloc
366
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000367# -------------------------------------------- HTML documentation generator
368
369class HTMLRepr(Repr):
370 """Class for safely making an HTML representation of a Python object."""
371 def __init__(self):
372 Repr.__init__(self)
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000373 self.maxlist = self.maxtuple = 20
374 self.maxdict = 10
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000375 self.maxstring = self.maxother = 100
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000376
377 def escape(self, text):
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +0000378 return replace(text, '&', '&amp;', '<', '&lt;', '>', '&gt;')
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000379
380 def repr(self, object):
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000381 return Repr.repr(self, object)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000382
383 def repr1(self, x, level):
Skip Montanaro0fe8fce2003-06-27 15:45:41 +0000384 if hasattr(type(x), '__name__'):
385 methodname = 'repr_' + join(split(type(x).__name__), '_')
386 if hasattr(self, methodname):
387 return getattr(self, methodname)(x, level)
388 return self.escape(cram(stripid(repr(x)), self.maxother))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000389
390 def repr_string(self, x, level):
Ka-Ping Yeea2fe1032001-03-02 01:19:14 +0000391 test = cram(x, self.maxstring)
392 testrepr = repr(test)
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +0000393 if '\\' in test and '\\' not in replace(testrepr, r'\\', ''):
Ka-Ping Yeea2fe1032001-03-02 01:19:14 +0000394 # Backslashes are only literal in the string and are never
395 # needed to make any special characters, so show a raw string.
396 return 'r' + testrepr[0] + self.escape(test) + testrepr[0]
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +0000397 return re.sub(r'((\\[\\abfnrtv\'"]|\\[0-9]..|\\x..|\\u....)+)',
Ka-Ping Yeea2fe1032001-03-02 01:19:14 +0000398 r'<font color="#c040c0">\1</font>',
399 self.escape(testrepr))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000400
Skip Montanarodf708782002-03-07 22:58:02 +0000401 repr_str = repr_string
402
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000403 def repr_instance(self, x, level):
404 try:
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000405 return self.escape(cram(stripid(repr(x)), self.maxstring))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000406 except:
407 return self.escape('<%s instance>' % x.__class__.__name__)
408
409 repr_unicode = repr_string
410
411class HTMLDoc(Doc):
412 """Formatter class for HTML documentation."""
413
414 # ------------------------------------------- HTML formatting utilities
415
416 _repr_instance = HTMLRepr()
417 repr = _repr_instance.repr
418 escape = _repr_instance.escape
419
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000420 def page(self, title, contents):
421 """Format an HTML page."""
422 return '''
Tim Peters59ed4482001-10-31 04:20:26 +0000423<!doctype html PUBLIC "-//W3C//DTD HTML 4.0 Transitional//EN">
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000424<html><head><title>Python: %s</title>
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000425</head><body bgcolor="#f0f0f8">
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000426%s
427</body></html>''' % (title, contents)
428
429 def heading(self, title, fgcol, bgcol, extras=''):
430 """Format a page heading."""
431 return '''
Tim Peters59ed4482001-10-31 04:20:26 +0000432<table width="100%%" cellspacing=0 cellpadding=2 border=0 summary="heading">
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000433<tr bgcolor="%s">
Tim Peters2306d242001-09-25 03:18:32 +0000434<td valign=bottom>&nbsp;<br>
435<font color="%s" face="helvetica, arial">&nbsp;<br>%s</font></td
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000436><td align=right valign=bottom
Ka-Ping Yee987ec902001-03-23 13:35:45 +0000437><font color="%s" face="helvetica, arial">%s</font></td></tr></table>
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000438 ''' % (bgcol, fgcol, title, fgcol, extras or '&nbsp;')
439
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000440 def section(self, title, fgcol, bgcol, contents, width=6,
441 prelude='', marginalia=None, gap='&nbsp;'):
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000442 """Format a section with a heading."""
443 if marginalia is None:
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +0000444 marginalia = '<tt>' + '&nbsp;' * width + '</tt>'
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000445 result = '''<p>
Tim Peters59ed4482001-10-31 04:20:26 +0000446<table width="100%%" cellspacing=0 cellpadding=2 border=0 summary="section">
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000447<tr bgcolor="%s">
Tim Peters2306d242001-09-25 03:18:32 +0000448<td colspan=3 valign=bottom>&nbsp;<br>
449<font color="%s" face="helvetica, arial">%s</font></td></tr>
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000450 ''' % (bgcol, fgcol, title)
451 if prelude:
452 result = result + '''
Ka-Ping Yee987ec902001-03-23 13:35:45 +0000453<tr bgcolor="%s"><td rowspan=2>%s</td>
454<td colspan=2>%s</td></tr>
455<tr><td>%s</td>''' % (bgcol, marginalia, prelude, gap)
456 else:
457 result = result + '''
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000458<tr><td bgcolor="%s">%s</td><td>%s</td>''' % (bgcol, marginalia, gap)
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000459
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +0000460 return result + '\n<td width="100%%">%s</td></tr></table>' % contents
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000461
462 def bigsection(self, title, *args):
463 """Format a section with a big heading."""
464 title = '<big><strong>%s</strong></big>' % title
Guido van Rossum68468eb2003-02-27 20:14:51 +0000465 return self.section(title, *args)
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000466
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000467 def preformat(self, text):
468 """Format literal preformatted text."""
469 text = self.escape(expandtabs(text))
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +0000470 return replace(text, '\n\n', '\n \n', '\n\n', '\n \n',
471 ' ', '&nbsp;', '\n', '<br>\n')
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000472
473 def multicolumn(self, list, format, cols=4):
474 """Format a list of items into a multi-column list."""
475 result = ''
476 rows = (len(list)+cols-1)/cols
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000477 for col in range(cols):
478 result = result + '<td width="%d%%" valign=top>' % (100/cols)
479 for i in range(rows*col, rows*col+rows):
480 if i < len(list):
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000481 result = result + format(list[i]) + '<br>\n'
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000482 result = result + '</td>'
Tim Peters59ed4482001-10-31 04:20:26 +0000483 return '<table width="100%%" summary="list"><tr>%s</tr></table>' % result
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000484
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000485 def grey(self, text): return '<font color="#909090">%s</font>' % text
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000486
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000487 def namelink(self, name, *dicts):
488 """Make a link for an identifier, given name-to-URL mappings."""
489 for dict in dicts:
Raymond Hettinger54f02222002-06-01 14:18:47 +0000490 if name in dict:
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000491 return '<a href="%s">%s</a>' % (dict[name], name)
492 return name
493
Ka-Ping Yeeb7a48302001-04-12 20:39:14 +0000494 def classlink(self, object, modname):
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000495 """Make a link for a class."""
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000496 name, module = object.__name__, sys.modules.get(object.__module__)
497 if hasattr(module, name) and getattr(module, name) is object:
Ka-Ping Yeeb7a48302001-04-12 20:39:14 +0000498 return '<a href="%s.html#%s">%s</a>' % (
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000499 module.__name__, name, classname(object, modname))
500 return classname(object, modname)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000501
502 def modulelink(self, object):
503 """Make a link for a module."""
504 return '<a href="%s.html">%s</a>' % (object.__name__, object.__name__)
505
506 def modpkglink(self, (name, path, ispackage, shadowed)):
507 """Make a link for a module or package to display in an index."""
508 if shadowed:
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000509 return self.grey(name)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000510 if path:
511 url = '%s.%s.html' % (path, name)
512 else:
513 url = '%s.html' % name
514 if ispackage:
515 text = '<strong>%s</strong>&nbsp;(package)' % name
516 else:
517 text = name
518 return '<a href="%s">%s</a>' % (url, text)
519
520 def markup(self, text, escape=None, funcs={}, classes={}, methods={}):
521 """Mark up some plain text, given a context of symbols to look for.
522 Each context dictionary maps object names to anchor names."""
523 escape = escape or self.escape
524 results = []
525 here = 0
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000526 pattern = re.compile(r'\b((http|ftp)://\S+[\w/]|'
527 r'RFC[- ]?(\d+)|'
Ka-Ping Yeef78a81b2001-03-27 08:13:42 +0000528 r'PEP[- ]?(\d+)|'
Neil Schemenauerd69711c2002-03-24 23:02:07 +0000529 r'(self\.)?(\w+))')
Guido van Rossum8ca162f2002-04-07 06:36:23 +0000530 while True:
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000531 match = pattern.search(text, here)
532 if not match: break
533 start, end = match.span()
534 results.append(escape(text[here:start]))
535
Ka-Ping Yeef78a81b2001-03-27 08:13:42 +0000536 all, scheme, rfc, pep, selfdot, name = match.groups()
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000537 if scheme:
Neil Schemenauercddc1a02002-03-24 23:11:21 +0000538 url = escape(all).replace('"', '&quot;')
539 results.append('<a href="%s">%s</a>' % (url, url))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000540 elif rfc:
Ka-Ping Yeef78a81b2001-03-27 08:13:42 +0000541 url = 'http://www.rfc-editor.org/rfc/rfc%d.txt' % int(rfc)
542 results.append('<a href="%s">%s</a>' % (url, escape(all)))
543 elif pep:
544 url = 'http://www.python.org/peps/pep-%04d.html' % int(pep)
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000545 results.append('<a href="%s">%s</a>' % (url, escape(all)))
546 elif text[end:end+1] == '(':
547 results.append(self.namelink(name, methods, funcs, classes))
548 elif selfdot:
549 results.append('self.<strong>%s</strong>' % name)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000550 else:
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000551 results.append(self.namelink(name, classes))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000552 here = end
553 results.append(escape(text[here:]))
554 return join(results, '')
555
556 # ---------------------------------------------- type-specific routines
557
Ka-Ping Yeeb7a48302001-04-12 20:39:14 +0000558 def formattree(self, tree, modname, parent=None):
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000559 """Produce HTML for a class tree as given by inspect.getclasstree()."""
560 result = ''
561 for entry in tree:
562 if type(entry) is type(()):
563 c, bases = entry
Tim Peters2306d242001-09-25 03:18:32 +0000564 result = result + '<dt><font face="helvetica, arial">'
Ka-Ping Yeeb7a48302001-04-12 20:39:14 +0000565 result = result + self.classlink(c, modname)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000566 if bases and bases != (parent,):
567 parents = []
568 for base in bases:
Ka-Ping Yeeb7a48302001-04-12 20:39:14 +0000569 parents.append(self.classlink(base, modname))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000570 result = result + '(' + join(parents, ', ') + ')'
Tim Peters2306d242001-09-25 03:18:32 +0000571 result = result + '\n</font></dt>'
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000572 elif type(entry) is type([]):
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000573 result = result + '<dd>\n%s</dd>\n' % self.formattree(
Ka-Ping Yeeb7a48302001-04-12 20:39:14 +0000574 entry, modname, c)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000575 return '<dl>\n%s</dl>\n' % result
576
Tim Peters8dd7ade2001-10-18 19:56:17 +0000577 def docmodule(self, object, name=None, mod=None, *ignored):
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000578 """Produce HTML documentation for a module object."""
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000579 name = object.__name__ # ignore the passed-in name
Skip Montanaroa5616d22004-06-11 04:46:12 +0000580 try:
581 all = object.__all__
582 except AttributeError:
583 all = None
Ka-Ping Yee66efbc72001-03-01 13:55:20 +0000584 parts = split(name, '.')
585 links = []
586 for i in range(len(parts)-1):
587 links.append(
588 '<a href="%s.html"><font color="#ffffff">%s</font></a>' %
589 (join(parts[:i+1], '.'), parts[i]))
590 linkedname = join(links + parts[-1:], '.')
591 head = '<big><big><strong>%s</strong></big></big>' % linkedname
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000592 try:
Ka-Ping Yee239432a2001-03-02 02:45:08 +0000593 path = inspect.getabsfile(object)
Ka-Ping Yee6191a232001-04-13 15:00:27 +0000594 url = path
595 if sys.platform == 'win32':
596 import nturl2path
597 url = nturl2path.pathname2url(path)
598 filelink = '<a href="file:%s">%s</a>' % (url, path)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000599 except TypeError:
600 filelink = '(built-in)'
Ka-Ping Yee6f3f9a42001-02-27 22:42:36 +0000601 info = []
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000602 if hasattr(object, '__version__'):
Ka-Ping Yee6f3f9a42001-02-27 22:42:36 +0000603 version = str(object.__version__)
Ka-Ping Yee40c49912001-02-27 22:46:01 +0000604 if version[:11] == '$' + 'Revision: ' and version[-1:] == '$':
605 version = strip(version[11:-1])
Ka-Ping Yee1d384632001-03-01 00:24:32 +0000606 info.append('version %s' % self.escape(version))
Ka-Ping Yee6f3f9a42001-02-27 22:42:36 +0000607 if hasattr(object, '__date__'):
608 info.append(self.escape(str(object.__date__)))
609 if info:
610 head = head + ' (%s)' % join(info, ', ')
Skip Montanaro4997a692003-09-10 16:47:51 +0000611 docloc = self.getdocloc(object)
612 if docloc is not None:
613 docloc = '<br><a href="%(docloc)s">Module Docs</a>' % locals()
614 else:
615 docloc = ''
Ka-Ping Yee66efbc72001-03-01 13:55:20 +0000616 result = self.heading(
Skip Montanaro4997a692003-09-10 16:47:51 +0000617 head, '#ffffff', '#7799ee',
618 '<a href=".">index</a><br>' + filelink + docloc)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000619
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000620 modules = inspect.getmembers(object, inspect.ismodule)
621
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000622 classes, cdict = [], {}
623 for key, value in inspect.getmembers(object, inspect.isclass):
Johannes Gijsbers4c11f602004-08-30 14:13:04 +0000624 # if __all__ exists, believe it. Otherwise use old heuristic.
625 if (all is not None or
626 (inspect.getmodule(value) or object) is object):
Skip Montanaroa5616d22004-06-11 04:46:12 +0000627 if visiblename(key, all):
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000628 classes.append((key, value))
629 cdict[key] = cdict[value] = '#' + key
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000630 for key, value in classes:
631 for base in value.__bases__:
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000632 key, modname = base.__name__, base.__module__
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000633 module = sys.modules.get(modname)
634 if modname != name and module and hasattr(module, key):
635 if getattr(module, key) is base:
Raymond Hettinger54f02222002-06-01 14:18:47 +0000636 if not key in cdict:
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000637 cdict[key] = cdict[base] = modname + '.html#' + key
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000638 funcs, fdict = [], {}
639 for key, value in inspect.getmembers(object, inspect.isroutine):
Johannes Gijsbers4c11f602004-08-30 14:13:04 +0000640 # if __all__ exists, believe it. Otherwise use old heuristic.
641 if (all is not None or
642 inspect.isbuiltin(value) or inspect.getmodule(value) is object):
Skip Montanaroa5616d22004-06-11 04:46:12 +0000643 if visiblename(key, all):
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000644 funcs.append((key, value))
645 fdict[key] = '#-' + key
646 if inspect.isfunction(value): fdict[value] = fdict[key]
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000647 data = []
648 for key, value in inspect.getmembers(object, isdata):
Skip Montanaroa5616d22004-06-11 04:46:12 +0000649 if visiblename(key, all):
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000650 data.append((key, value))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000651
652 doc = self.markup(getdoc(object), self.preformat, fdict, cdict)
653 doc = doc and '<tt>%s</tt>' % doc
Tim Peters2306d242001-09-25 03:18:32 +0000654 result = result + '<p>%s</p>\n' % doc
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000655
656 if hasattr(object, '__path__'):
657 modpkgs = []
Thomas Wouters49fd7fa2006-04-21 10:40:58 +0000658 for importer, modname, ispkg in pkgutil.iter_modules(object.__path__):
659 modpkgs.append((modname, name, ispkg, 0))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000660 modpkgs.sort()
661 contents = self.multicolumn(modpkgs, self.modpkglink)
662 result = result + self.bigsection(
663 'Package Contents', '#ffffff', '#aa55cc', contents)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000664 elif modules:
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000665 contents = self.multicolumn(
666 modules, lambda (key, value), s=self: s.modulelink(value))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000667 result = result + self.bigsection(
668 'Modules', '#fffff', '#aa55cc', contents)
669
670 if classes:
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000671 classlist = map(lambda (key, value): value, classes)
Ka-Ping Yeeb7a48302001-04-12 20:39:14 +0000672 contents = [
673 self.formattree(inspect.getclasstree(classlist, 1), name)]
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000674 for key, value in classes:
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +0000675 contents.append(self.document(value, key, name, fdict, cdict))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000676 result = result + self.bigsection(
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000677 'Classes', '#ffffff', '#ee77aa', join(contents))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000678 if funcs:
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000679 contents = []
680 for key, value in funcs:
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +0000681 contents.append(self.document(value, key, name, fdict, cdict))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000682 result = result + self.bigsection(
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000683 'Functions', '#ffffff', '#eeaa77', join(contents))
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000684 if data:
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000685 contents = []
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000686 for key, value in data:
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000687 contents.append(self.document(value, key))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000688 result = result + self.bigsection(
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000689 'Data', '#ffffff', '#55aa55', join(contents, '<br>\n'))
Ka-Ping Yee6f3f9a42001-02-27 22:42:36 +0000690 if hasattr(object, '__author__'):
691 contents = self.markup(str(object.__author__), self.preformat)
692 result = result + self.bigsection(
693 'Author', '#ffffff', '#7799ee', contents)
Ka-Ping Yee6f3f9a42001-02-27 22:42:36 +0000694 if hasattr(object, '__credits__'):
695 contents = self.markup(str(object.__credits__), self.preformat)
696 result = result + self.bigsection(
697 'Credits', '#ffffff', '#7799ee', contents)
698
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000699 return result
700
Tim Peters8dd7ade2001-10-18 19:56:17 +0000701 def docclass(self, object, name=None, mod=None, funcs={}, classes={},
702 *ignored):
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000703 """Produce HTML documentation for a class object."""
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000704 realname = object.__name__
705 name = name or realname
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000706 bases = object.__bases__
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000707
Tim Petersb47879b2001-09-24 04:47:19 +0000708 contents = []
709 push = contents.append
710
Tim Petersfa26f7c2001-09-24 08:05:11 +0000711 # Cute little class to pump out a horizontal rule between sections.
712 class HorizontalRule:
713 def __init__(self):
714 self.needone = 0
715 def maybe(self):
716 if self.needone:
717 push('<hr>\n')
718 self.needone = 1
719 hr = HorizontalRule()
720
Tim Petersc86f6ca2001-09-26 21:31:51 +0000721 # List the mro, if non-trivial.
Raymond Hettinger756b3f32004-01-29 06:37:52 +0000722 mro = deque(inspect.getmro(object))
Tim Petersc86f6ca2001-09-26 21:31:51 +0000723 if len(mro) > 2:
724 hr.maybe()
725 push('<dl><dt>Method resolution order:</dt>\n')
726 for base in mro:
727 push('<dd>%s</dd>\n' % self.classlink(base,
728 object.__module__))
729 push('</dl>\n')
730
Tim Petersb47879b2001-09-24 04:47:19 +0000731 def spill(msg, attrs, predicate):
Tim Petersfa26f7c2001-09-24 08:05:11 +0000732 ok, attrs = _split_list(attrs, predicate)
Tim Petersb47879b2001-09-24 04:47:19 +0000733 if ok:
Tim Petersfa26f7c2001-09-24 08:05:11 +0000734 hr.maybe()
Tim Petersb47879b2001-09-24 04:47:19 +0000735 push(msg)
736 for name, kind, homecls, value in ok:
737 push(self.document(getattr(object, name), name, mod,
738 funcs, classes, mdict, object))
739 push('\n')
740 return attrs
741
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +0000742 def spilldescriptors(msg, attrs, predicate):
Tim Petersfa26f7c2001-09-24 08:05:11 +0000743 ok, attrs = _split_list(attrs, predicate)
Tim Petersb47879b2001-09-24 04:47:19 +0000744 if ok:
Tim Petersfa26f7c2001-09-24 08:05:11 +0000745 hr.maybe()
Tim Petersb47879b2001-09-24 04:47:19 +0000746 push(msg)
747 for name, kind, homecls, value in ok:
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +0000748 push(self._docdescriptor(name, value, mod))
Tim Petersb47879b2001-09-24 04:47:19 +0000749 return attrs
750
Tim Petersfa26f7c2001-09-24 08:05:11 +0000751 def spilldata(msg, attrs, predicate):
752 ok, attrs = _split_list(attrs, predicate)
Tim Petersb47879b2001-09-24 04:47:19 +0000753 if ok:
Tim Petersfa26f7c2001-09-24 08:05:11 +0000754 hr.maybe()
Tim Petersb47879b2001-09-24 04:47:19 +0000755 push(msg)
756 for name, kind, homecls, value in ok:
757 base = self.docother(getattr(object, name), name, mod)
Martin v. Löwise59e2ba2003-05-03 09:09:02 +0000758 if callable(value) or inspect.isdatadescriptor(value):
Guido van Rossum5e355b22002-05-21 20:56:15 +0000759 doc = getattr(value, "__doc__", None)
760 else:
761 doc = None
Tim Petersb47879b2001-09-24 04:47:19 +0000762 if doc is None:
763 push('<dl><dt>%s</dl>\n' % base)
764 else:
765 doc = self.markup(getdoc(value), self.preformat,
766 funcs, classes, mdict)
Tim Peters2306d242001-09-25 03:18:32 +0000767 doc = '<dd><tt>%s</tt>' % doc
Tim Petersb47879b2001-09-24 04:47:19 +0000768 push('<dl><dt>%s%s</dl>\n' % (base, doc))
769 push('\n')
770 return attrs
771
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000772 attrs = filter(lambda (name, kind, cls, value): visiblename(name),
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +0000773 classify_class_attrs(object))
Tim Petersb47879b2001-09-24 04:47:19 +0000774 mdict = {}
775 for key, kind, homecls, value in attrs:
776 mdict[key] = anchor = '#' + name + '-' + key
777 value = getattr(object, key)
778 try:
779 # The value may not be hashable (e.g., a data attr with
780 # a dict or list value).
781 mdict[value] = anchor
782 except TypeError:
783 pass
784
Tim Petersfa26f7c2001-09-24 08:05:11 +0000785 while attrs:
Tim Peters351e3622001-09-27 03:29:51 +0000786 if mro:
Raymond Hettinger756b3f32004-01-29 06:37:52 +0000787 thisclass = mro.popleft()
Tim Peters351e3622001-09-27 03:29:51 +0000788 else:
789 thisclass = attrs[0][2]
Tim Petersfa26f7c2001-09-24 08:05:11 +0000790 attrs, inherited = _split_list(attrs, lambda t: t[2] is thisclass)
791
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000792 if thisclass is __builtin__.object:
793 attrs = inherited
794 continue
795 elif thisclass is object:
796 tag = 'defined here'
Tim Petersb47879b2001-09-24 04:47:19 +0000797 else:
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000798 tag = 'inherited from %s' % self.classlink(thisclass,
799 object.__module__)
Tim Petersb47879b2001-09-24 04:47:19 +0000800 tag += ':<br>\n'
801
802 # Sort attrs by name.
Thomas Wouters49fd7fa2006-04-21 10:40:58 +0000803 try:
804 attrs.sort(key=lambda t: t[0])
805 except TypeError:
806 attrs.sort(lambda t1, t2: cmp(t1[0], t2[0])) # 2.3 compat
Tim Petersb47879b2001-09-24 04:47:19 +0000807
808 # Pump out the attrs, segregated by kind.
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000809 attrs = spill('Methods %s' % tag, attrs,
Tim Petersb47879b2001-09-24 04:47:19 +0000810 lambda t: t[1] == 'method')
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000811 attrs = spill('Class methods %s' % tag, attrs,
Tim Petersb47879b2001-09-24 04:47:19 +0000812 lambda t: t[1] == 'class method')
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000813 attrs = spill('Static methods %s' % tag, attrs,
Tim Petersb47879b2001-09-24 04:47:19 +0000814 lambda t: t[1] == 'static method')
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +0000815 attrs = spilldescriptors('Data descriptors %s' % tag, attrs,
816 lambda t: t[1] == 'data descriptor')
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000817 attrs = spilldata('Data and other attributes %s' % tag, attrs,
Tim Petersfa26f7c2001-09-24 08:05:11 +0000818 lambda t: t[1] == 'data')
Tim Petersb47879b2001-09-24 04:47:19 +0000819 assert attrs == []
Tim Peters351e3622001-09-27 03:29:51 +0000820 attrs = inherited
Tim Petersb47879b2001-09-24 04:47:19 +0000821
822 contents = ''.join(contents)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000823
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000824 if name == realname:
825 title = '<a name="%s">class <strong>%s</strong></a>' % (
826 name, realname)
827 else:
828 title = '<strong>%s</strong> = <a name="%s">class %s</a>' % (
829 name, name, realname)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000830 if bases:
831 parents = []
832 for base in bases:
Ka-Ping Yeeb7a48302001-04-12 20:39:14 +0000833 parents.append(self.classlink(base, object.__module__))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000834 title = title + '(%s)' % join(parents, ', ')
Tim Peters2306d242001-09-25 03:18:32 +0000835 doc = self.markup(getdoc(object), self.preformat, funcs, classes, mdict)
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000836 doc = doc and '<tt>%s<br>&nbsp;</tt>' % doc
Tim Petersc86f6ca2001-09-26 21:31:51 +0000837
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000838 return self.section(title, '#000000', '#ffc8d8', contents, 3, doc)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000839
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000840 def formatvalue(self, object):
841 """Format an argument default value as text."""
Tim Peters2306d242001-09-25 03:18:32 +0000842 return self.grey('=' + self.repr(object))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000843
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +0000844 def docroutine(self, object, name=None, mod=None,
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000845 funcs={}, classes={}, methods={}, cl=None):
Ka-Ping Yee66efbc72001-03-01 13:55:20 +0000846 """Produce HTML documentation for a function or method object."""
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000847 realname = object.__name__
848 name = name or realname
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000849 anchor = (cl and cl.__name__ or '') + '-' + name
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000850 note = ''
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000851 skipdocs = 0
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000852 if inspect.ismethod(object):
Ka-Ping Yee6dcfa382001-04-12 20:27:31 +0000853 imclass = object.im_class
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000854 if cl:
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +0000855 if imclass is not cl:
Ka-Ping Yeeb7a48302001-04-12 20:39:14 +0000856 note = ' from ' + self.classlink(imclass, mod)
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000857 else:
Ka-Ping Yeeb7a48302001-04-12 20:39:14 +0000858 if object.im_self:
859 note = ' method of %s instance' % self.classlink(
860 object.im_self.__class__, mod)
861 else:
862 note = ' unbound %s method' % self.classlink(imclass,mod)
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000863 object = object.im_func
864
865 if name == realname:
866 title = '<a name="%s"><strong>%s</strong></a>' % (anchor, realname)
867 else:
Raymond Hettinger54f02222002-06-01 14:18:47 +0000868 if (cl and realname in cl.__dict__ and
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000869 cl.__dict__[realname] is object):
Ka-Ping Yeee280c062001-03-23 14:05:53 +0000870 reallink = '<a href="#%s">%s</a>' % (
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000871 cl.__name__ + '-' + realname, realname)
872 skipdocs = 1
873 else:
874 reallink = realname
875 title = '<a name="%s"><strong>%s</strong></a> = %s' % (
876 anchor, name, reallink)
Tim Peters4bcfa312001-09-20 06:08:24 +0000877 if inspect.isfunction(object):
Guido van Rossum2e65f892007-02-28 22:03:49 +0000878 args, varargs, kwonlyargs, kwdefaults, varkw, defaults, ann = \
879 inspect.getfullargspec(object)
Ka-Ping Yee66efbc72001-03-01 13:55:20 +0000880 argspec = inspect.formatargspec(
Guido van Rossum2e65f892007-02-28 22:03:49 +0000881 args, varargs, kwonlyargs, kwdefaults, varkw, defaults, ann,
882 formatvalue=self.formatvalue,
883 formatannotation=inspect.formatannotationrelativeto(object))
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000884 if realname == '<lambda>':
Tim Peters59ed4482001-10-31 04:20:26 +0000885 title = '<strong>%s</strong> <em>lambda</em> ' % name
Guido van Rossum2e65f892007-02-28 22:03:49 +0000886 # XXX lambda's won't usually have func_annotations['return']
887 # since the syntax doesn't support but it is possible.
888 # So removing parentheses isn't truly safe.
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000889 argspec = argspec[1:-1] # remove parentheses
Tim Peters4bcfa312001-09-20 06:08:24 +0000890 else:
891 argspec = '(...)'
Ka-Ping Yee66efbc72001-03-01 13:55:20 +0000892
Tim Peters2306d242001-09-25 03:18:32 +0000893 decl = title + argspec + (note and self.grey(
894 '<font face="helvetica, arial">%s</font>' % note))
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000895
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000896 if skipdocs:
Tim Peters2306d242001-09-25 03:18:32 +0000897 return '<dl><dt>%s</dt></dl>\n' % decl
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000898 else:
899 doc = self.markup(
900 getdoc(object), self.preformat, funcs, classes, methods)
Tim Peters2306d242001-09-25 03:18:32 +0000901 doc = doc and '<dd><tt>%s</tt></dd>' % doc
902 return '<dl><dt>%s</dt>%s</dl>\n' % (decl, doc)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000903
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +0000904 def _docdescriptor(self, name, value, mod):
Johannes Gijsbers8de645a2004-11-07 19:16:05 +0000905 results = []
906 push = results.append
907
908 if name:
909 push('<dl><dt><strong>%s</strong></dt>\n' % name)
910 if value.__doc__ is not None:
Ka-Ping Yeebba6acc2005-02-19 22:58:26 +0000911 doc = self.markup(getdoc(value), self.preformat)
Johannes Gijsbers8de645a2004-11-07 19:16:05 +0000912 push('<dd><tt>%s</tt></dd>\n' % doc)
Johannes Gijsbers8de645a2004-11-07 19:16:05 +0000913 push('</dl>\n')
914
915 return ''.join(results)
916
917 def docproperty(self, object, name=None, mod=None, cl=None):
918 """Produce html documentation for a property."""
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +0000919 return self._docdescriptor(name, object, mod)
Johannes Gijsbers8de645a2004-11-07 19:16:05 +0000920
Tim Peters8dd7ade2001-10-18 19:56:17 +0000921 def docother(self, object, name=None, mod=None, *ignored):
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000922 """Produce HTML documentation for a data object."""
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +0000923 lhs = name and '<strong>%s</strong> = ' % name or ''
924 return lhs + self.repr(object)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000925
Thomas Wouters0e3f5912006-08-11 14:57:12 +0000926 def docdata(self, object, name=None, mod=None, cl=None):
927 """Produce html documentation for a data descriptor."""
928 return self._docdescriptor(name, object, mod)
929
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000930 def index(self, dir, shadowed=None):
931 """Generate an HTML index for a directory of modules."""
932 modpkgs = []
933 if shadowed is None: shadowed = {}
Thomas Wouters49fd7fa2006-04-21 10:40:58 +0000934 for importer, name, ispkg in pkgutil.iter_modules([dir]):
935 modpkgs.append((name, '', ispkg, name in shadowed))
936 shadowed[name] = 1
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000937
938 modpkgs.sort()
939 contents = self.multicolumn(modpkgs, self.modpkglink)
940 return self.bigsection(dir, '#ffffff', '#ee77aa', contents)
941
942# -------------------------------------------- text documentation generator
943
944class TextRepr(Repr):
945 """Class for safely making a text representation of a Python object."""
946 def __init__(self):
947 Repr.__init__(self)
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000948 self.maxlist = self.maxtuple = 20
949 self.maxdict = 10
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000950 self.maxstring = self.maxother = 100
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000951
952 def repr1(self, x, level):
Skip Montanaro0fe8fce2003-06-27 15:45:41 +0000953 if hasattr(type(x), '__name__'):
954 methodname = 'repr_' + join(split(type(x).__name__), '_')
955 if hasattr(self, methodname):
956 return getattr(self, methodname)(x, level)
957 return cram(stripid(repr(x)), self.maxother)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000958
Ka-Ping Yeea2fe1032001-03-02 01:19:14 +0000959 def repr_string(self, x, level):
960 test = cram(x, self.maxstring)
961 testrepr = repr(test)
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +0000962 if '\\' in test and '\\' not in replace(testrepr, r'\\', ''):
Ka-Ping Yeea2fe1032001-03-02 01:19:14 +0000963 # Backslashes are only literal in the string and are never
964 # needed to make any special characters, so show a raw string.
965 return 'r' + testrepr[0] + test + testrepr[0]
966 return testrepr
967
Skip Montanarodf708782002-03-07 22:58:02 +0000968 repr_str = repr_string
969
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000970 def repr_instance(self, x, level):
971 try:
Ka-Ping Yee1d384632001-03-01 00:24:32 +0000972 return cram(stripid(repr(x)), self.maxstring)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000973 except:
974 return '<%s instance>' % x.__class__.__name__
975
976class TextDoc(Doc):
977 """Formatter class for text documentation."""
978
979 # ------------------------------------------- text formatting utilities
980
981 _repr_instance = TextRepr()
982 repr = _repr_instance.repr
983
984 def bold(self, text):
985 """Format a string in bold by overstriking."""
986 return join(map(lambda ch: ch + '\b' + ch, text), '')
987
988 def indent(self, text, prefix=' '):
989 """Indent text by prepending a given prefix to each line."""
990 if not text: return ''
991 lines = split(text, '\n')
992 lines = map(lambda line, prefix=prefix: prefix + line, lines)
993 if lines: lines[-1] = rstrip(lines[-1])
994 return join(lines, '\n')
995
996 def section(self, title, contents):
997 """Format a section with a given heading."""
998 return self.bold(title) + '\n' + rstrip(self.indent(contents)) + '\n\n'
999
1000 # ---------------------------------------------- type-specific routines
1001
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001002 def formattree(self, tree, modname, parent=None, prefix=''):
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001003 """Render in text a class tree as returned by inspect.getclasstree()."""
1004 result = ''
1005 for entry in tree:
1006 if type(entry) is type(()):
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001007 c, bases = entry
1008 result = result + prefix + classname(c, modname)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001009 if bases and bases != (parent,):
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001010 parents = map(lambda c, m=modname: classname(c, m), bases)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001011 result = result + '(%s)' % join(parents, ', ')
1012 result = result + '\n'
1013 elif type(entry) is type([]):
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001014 result = result + self.formattree(
1015 entry, modname, c, prefix + ' ')
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001016 return result
1017
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001018 def docmodule(self, object, name=None, mod=None):
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001019 """Produce text documentation for a given module object."""
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001020 name = object.__name__ # ignore the passed-in name
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +00001021 synop, desc = splitdoc(getdoc(object))
1022 result = self.section('NAME', name + (synop and ' - ' + synop))
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001023
1024 try:
Skip Montanaroa5616d22004-06-11 04:46:12 +00001025 all = object.__all__
1026 except AttributeError:
1027 all = None
1028
1029 try:
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001030 file = inspect.getabsfile(object)
1031 except TypeError:
1032 file = '(built-in)'
Ka-Ping Yee239432a2001-03-02 02:45:08 +00001033 result = result + self.section('FILE', file)
Skip Montanaro4997a692003-09-10 16:47:51 +00001034
1035 docloc = self.getdocloc(object)
1036 if docloc is not None:
1037 result = result + self.section('MODULE DOCS', docloc)
1038
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +00001039 if desc:
1040 result = result + self.section('DESCRIPTION', desc)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001041
1042 classes = []
1043 for key, value in inspect.getmembers(object, inspect.isclass):
Johannes Gijsbers4c11f602004-08-30 14:13:04 +00001044 # if __all__ exists, believe it. Otherwise use old heuristic.
1045 if (all is not None
1046 or (inspect.getmodule(value) or object) is object):
Skip Montanaroa5616d22004-06-11 04:46:12 +00001047 if visiblename(key, all):
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +00001048 classes.append((key, value))
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001049 funcs = []
1050 for key, value in inspect.getmembers(object, inspect.isroutine):
Johannes Gijsbers4c11f602004-08-30 14:13:04 +00001051 # if __all__ exists, believe it. Otherwise use old heuristic.
1052 if (all is not None or
1053 inspect.isbuiltin(value) or inspect.getmodule(value) is object):
Skip Montanaroa5616d22004-06-11 04:46:12 +00001054 if visiblename(key, all):
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +00001055 funcs.append((key, value))
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001056 data = []
1057 for key, value in inspect.getmembers(object, isdata):
Skip Montanaroa5616d22004-06-11 04:46:12 +00001058 if visiblename(key, all):
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001059 data.append((key, value))
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001060
1061 if hasattr(object, '__path__'):
1062 modpkgs = []
Thomas Wouters49fd7fa2006-04-21 10:40:58 +00001063 for importer, modname, ispkg in pkgutil.iter_modules(object.__path__):
1064 if ispkg:
1065 modpkgs.append(modname + ' (package)')
1066 else:
1067 modpkgs.append(modname)
1068
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001069 modpkgs.sort()
1070 result = result + self.section(
1071 'PACKAGE CONTENTS', join(modpkgs, '\n'))
1072
1073 if classes:
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001074 classlist = map(lambda (key, value): value, classes)
1075 contents = [self.formattree(
1076 inspect.getclasstree(classlist, 1), name)]
1077 for key, value in classes:
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001078 contents.append(self.document(value, key, name))
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001079 result = result + self.section('CLASSES', join(contents, '\n'))
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001080
1081 if funcs:
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001082 contents = []
1083 for key, value in funcs:
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001084 contents.append(self.document(value, key, name))
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001085 result = result + self.section('FUNCTIONS', join(contents, '\n'))
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001086
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001087 if data:
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001088 contents = []
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001089 for key, value in data:
Georg Brandl8b813db2005-10-01 16:32:31 +00001090 contents.append(self.docother(value, key, name, maxlen=70))
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001091 result = result + self.section('DATA', join(contents, '\n'))
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001092
1093 if hasattr(object, '__version__'):
1094 version = str(object.__version__)
Ka-Ping Yee1d384632001-03-01 00:24:32 +00001095 if version[:11] == '$' + 'Revision: ' and version[-1:] == '$':
1096 version = strip(version[11:-1])
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001097 result = result + self.section('VERSION', version)
Ka-Ping Yee6f3f9a42001-02-27 22:42:36 +00001098 if hasattr(object, '__date__'):
1099 result = result + self.section('DATE', str(object.__date__))
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001100 if hasattr(object, '__author__'):
Ka-Ping Yee6f3f9a42001-02-27 22:42:36 +00001101 result = result + self.section('AUTHOR', str(object.__author__))
1102 if hasattr(object, '__credits__'):
1103 result = result + self.section('CREDITS', str(object.__credits__))
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001104 return result
1105
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001106 def docclass(self, object, name=None, mod=None):
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001107 """Produce text documentation for a given class object."""
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001108 realname = object.__name__
1109 name = name or realname
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001110 bases = object.__bases__
1111
Tim Petersc86f6ca2001-09-26 21:31:51 +00001112 def makename(c, m=object.__module__):
1113 return classname(c, m)
1114
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001115 if name == realname:
1116 title = 'class ' + self.bold(realname)
1117 else:
1118 title = self.bold(name) + ' = class ' + realname
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001119 if bases:
Ka-Ping Yee3bda8792001-03-23 13:17:50 +00001120 parents = map(makename, bases)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001121 title = title + '(%s)' % join(parents, ', ')
1122
1123 doc = getdoc(object)
Tim Peters28355492001-09-23 21:29:55 +00001124 contents = doc and [doc + '\n'] or []
1125 push = contents.append
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001126
Tim Petersc86f6ca2001-09-26 21:31:51 +00001127 # List the mro, if non-trivial.
Raymond Hettinger756b3f32004-01-29 06:37:52 +00001128 mro = deque(inspect.getmro(object))
Tim Petersc86f6ca2001-09-26 21:31:51 +00001129 if len(mro) > 2:
1130 push("Method resolution order:")
1131 for base in mro:
1132 push(' ' + makename(base))
1133 push('')
1134
Tim Petersf4aad8e2001-09-24 22:40:47 +00001135 # Cute little class to pump out a horizontal rule between sections.
1136 class HorizontalRule:
1137 def __init__(self):
1138 self.needone = 0
1139 def maybe(self):
1140 if self.needone:
1141 push('-' * 70)
1142 self.needone = 1
1143 hr = HorizontalRule()
1144
Tim Peters28355492001-09-23 21:29:55 +00001145 def spill(msg, attrs, predicate):
Tim Petersfa26f7c2001-09-24 08:05:11 +00001146 ok, attrs = _split_list(attrs, predicate)
Tim Peters28355492001-09-23 21:29:55 +00001147 if ok:
Tim Petersf4aad8e2001-09-24 22:40:47 +00001148 hr.maybe()
Tim Peters28355492001-09-23 21:29:55 +00001149 push(msg)
1150 for name, kind, homecls, value in ok:
1151 push(self.document(getattr(object, name),
1152 name, mod, object))
1153 return attrs
1154
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +00001155 def spilldescriptors(msg, attrs, predicate):
Tim Petersfa26f7c2001-09-24 08:05:11 +00001156 ok, attrs = _split_list(attrs, predicate)
Tim Peters28355492001-09-23 21:29:55 +00001157 if ok:
Tim Petersf4aad8e2001-09-24 22:40:47 +00001158 hr.maybe()
Tim Peters28355492001-09-23 21:29:55 +00001159 push(msg)
1160 for name, kind, homecls, value in ok:
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +00001161 push(self._docdescriptor(name, value, mod))
Tim Peters28355492001-09-23 21:29:55 +00001162 return attrs
Tim Petersb47879b2001-09-24 04:47:19 +00001163
Tim Petersfa26f7c2001-09-24 08:05:11 +00001164 def spilldata(msg, attrs, predicate):
1165 ok, attrs = _split_list(attrs, predicate)
Tim Peters28355492001-09-23 21:29:55 +00001166 if ok:
Tim Petersf4aad8e2001-09-24 22:40:47 +00001167 hr.maybe()
Tim Peters28355492001-09-23 21:29:55 +00001168 push(msg)
1169 for name, kind, homecls, value in ok:
Martin v. Löwise59e2ba2003-05-03 09:09:02 +00001170 if callable(value) or inspect.isdatadescriptor(value):
Ka-Ping Yeebba6acc2005-02-19 22:58:26 +00001171 doc = getdoc(value)
Guido van Rossum5e355b22002-05-21 20:56:15 +00001172 else:
1173 doc = None
Tim Peters28355492001-09-23 21:29:55 +00001174 push(self.docother(getattr(object, name),
Georg Brandl8b813db2005-10-01 16:32:31 +00001175 name, mod, maxlen=70, doc=doc) + '\n')
Tim Peters28355492001-09-23 21:29:55 +00001176 return attrs
1177
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +00001178 attrs = filter(lambda (name, kind, cls, value): visiblename(name),
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +00001179 classify_class_attrs(object))
Tim Petersfa26f7c2001-09-24 08:05:11 +00001180 while attrs:
Tim Peters351e3622001-09-27 03:29:51 +00001181 if mro:
Raymond Hettinger756b3f32004-01-29 06:37:52 +00001182 thisclass = mro.popleft()
Tim Peters351e3622001-09-27 03:29:51 +00001183 else:
1184 thisclass = attrs[0][2]
Tim Petersfa26f7c2001-09-24 08:05:11 +00001185 attrs, inherited = _split_list(attrs, lambda t: t[2] is thisclass)
1186
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +00001187 if thisclass is __builtin__.object:
1188 attrs = inherited
1189 continue
1190 elif thisclass is object:
Tim Peters28355492001-09-23 21:29:55 +00001191 tag = "defined here"
1192 else:
Tim Petersfa26f7c2001-09-24 08:05:11 +00001193 tag = "inherited from %s" % classname(thisclass,
1194 object.__module__)
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +00001195 filter(lambda t: not t[0].startswith('_'), attrs)
Tim Peters28355492001-09-23 21:29:55 +00001196
1197 # Sort attrs by name.
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +00001198 attrs.sort()
Tim Peters28355492001-09-23 21:29:55 +00001199
1200 # Pump out the attrs, segregated by kind.
Tim Petersf4aad8e2001-09-24 22:40:47 +00001201 attrs = spill("Methods %s:\n" % tag, attrs,
Tim Peters28355492001-09-23 21:29:55 +00001202 lambda t: t[1] == 'method')
Tim Petersf4aad8e2001-09-24 22:40:47 +00001203 attrs = spill("Class methods %s:\n" % tag, attrs,
Tim Peters28355492001-09-23 21:29:55 +00001204 lambda t: t[1] == 'class method')
Tim Petersf4aad8e2001-09-24 22:40:47 +00001205 attrs = spill("Static methods %s:\n" % tag, attrs,
Tim Peters28355492001-09-23 21:29:55 +00001206 lambda t: t[1] == 'static method')
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +00001207 attrs = spilldescriptors("Data descriptors %s:\n" % tag, attrs,
1208 lambda t: t[1] == 'data descriptor')
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +00001209 attrs = spilldata("Data and other attributes %s:\n" % tag, attrs,
1210 lambda t: t[1] == 'data')
Tim Peters28355492001-09-23 21:29:55 +00001211 assert attrs == []
Tim Peters351e3622001-09-27 03:29:51 +00001212 attrs = inherited
Tim Peters28355492001-09-23 21:29:55 +00001213
1214 contents = '\n'.join(contents)
1215 if not contents:
1216 return title + '\n'
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001217 return title + '\n' + self.indent(rstrip(contents), ' | ') + '\n'
1218
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001219 def formatvalue(self, object):
1220 """Format an argument default value as text."""
1221 return '=' + self.repr(object)
1222
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001223 def docroutine(self, object, name=None, mod=None, cl=None):
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001224 """Produce text documentation for a function or method object."""
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001225 realname = object.__name__
1226 name = name or realname
1227 note = ''
Ka-Ping Yee3bda8792001-03-23 13:17:50 +00001228 skipdocs = 0
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001229 if inspect.ismethod(object):
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001230 imclass = object.im_class
Ka-Ping Yee3bda8792001-03-23 13:17:50 +00001231 if cl:
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001232 if imclass is not cl:
1233 note = ' from ' + classname(imclass, mod)
Ka-Ping Yee3bda8792001-03-23 13:17:50 +00001234 else:
Ka-Ping Yeeb7a48302001-04-12 20:39:14 +00001235 if object.im_self:
1236 note = ' method of %s instance' % classname(
1237 object.im_self.__class__, mod)
1238 else:
1239 note = ' unbound %s method' % classname(imclass,mod)
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001240 object = object.im_func
1241
1242 if name == realname:
1243 title = self.bold(realname)
1244 else:
Raymond Hettinger54f02222002-06-01 14:18:47 +00001245 if (cl and realname in cl.__dict__ and
Ka-Ping Yee3bda8792001-03-23 13:17:50 +00001246 cl.__dict__[realname] is object):
1247 skipdocs = 1
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001248 title = self.bold(name) + ' = ' + realname
Tim Peters4bcfa312001-09-20 06:08:24 +00001249 if inspect.isfunction(object):
Guido van Rossum2e65f892007-02-28 22:03:49 +00001250 args, varargs, varkw, defaults, kwonlyargs, kwdefaults, ann = \
1251 inspect.getfullargspec(object)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001252 argspec = inspect.formatargspec(
Guido van Rossum2e65f892007-02-28 22:03:49 +00001253 args, varargs, varkw, defaults, kwonlyargs, kwdefaults, ann,
1254 formatvalue=self.formatvalue,
1255 formatannotation=inspect.formatannotationrelativeto(object))
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001256 if realname == '<lambda>':
Georg Brandl501dd0d2006-02-17 09:45:40 +00001257 title = self.bold(name) + ' lambda '
Guido van Rossum2e65f892007-02-28 22:03:49 +00001258 # XXX lambda's won't usually have func_annotations['return']
1259 # since the syntax doesn't support but it is possible.
1260 # So removing parentheses isn't truly safe.
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001261 argspec = argspec[1:-1] # remove parentheses
Tim Peters4bcfa312001-09-20 06:08:24 +00001262 else:
1263 argspec = '(...)'
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001264 decl = title + argspec + note
1265
Ka-Ping Yee3bda8792001-03-23 13:17:50 +00001266 if skipdocs:
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001267 return decl + '\n'
Ka-Ping Yee3bda8792001-03-23 13:17:50 +00001268 else:
1269 doc = getdoc(object) or ''
1270 return decl + '\n' + (doc and rstrip(self.indent(doc)) + '\n')
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001271
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +00001272 def _docdescriptor(self, name, value, mod):
Johannes Gijsbers8de645a2004-11-07 19:16:05 +00001273 results = []
1274 push = results.append
1275
1276 if name:
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +00001277 push(self.bold(name))
1278 push('\n')
Johannes Gijsbers8de645a2004-11-07 19:16:05 +00001279 doc = getdoc(value) or ''
1280 if doc:
1281 push(self.indent(doc))
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +00001282 push('\n')
1283 return ''.join(results)
Johannes Gijsbers8de645a2004-11-07 19:16:05 +00001284
1285 def docproperty(self, object, name=None, mod=None, cl=None):
1286 """Produce text documentation for a property."""
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +00001287 return self._docdescriptor(name, object, mod)
Johannes Gijsbers8de645a2004-11-07 19:16:05 +00001288
Thomas Wouters0e3f5912006-08-11 14:57:12 +00001289 def docdata(self, object, name=None, mod=None, cl=None):
1290 """Produce text documentation for a data descriptor."""
1291 return self._docdescriptor(name, object, mod)
1292
Georg Brandl8b813db2005-10-01 16:32:31 +00001293 def docother(self, object, name=None, mod=None, parent=None, maxlen=None, doc=None):
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001294 """Produce text documentation for a data object."""
1295 repr = self.repr(object)
1296 if maxlen:
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001297 line = (name and name + ' = ' or '') + repr
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001298 chop = maxlen - len(line)
1299 if chop < 0: repr = repr[:chop] + '...'
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001300 line = (name and self.bold(name) + ' = ' or '') + repr
Tim Peters28355492001-09-23 21:29:55 +00001301 if doc is not None:
1302 line += '\n' + self.indent(str(doc))
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001303 return line
1304
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001305# --------------------------------------------------------- user interfaces
1306
1307def pager(text):
1308 """The first time this is called, determine what kind of pager to use."""
1309 global pager
1310 pager = getpager()
1311 pager(text)
1312
1313def getpager():
1314 """Decide what method to use for paging through text."""
1315 if type(sys.stdout) is not types.FileType:
1316 return plainpager
1317 if not sys.stdin.isatty() or not sys.stdout.isatty():
1318 return plainpager
Raymond Hettinger54f02222002-06-01 14:18:47 +00001319 if 'PAGER' in os.environ:
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +00001320 if sys.platform == 'win32': # pipes completely broken in Windows
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001321 return lambda text: tempfilepager(plain(text), os.environ['PAGER'])
Raymond Hettingerdbecd932005-02-06 06:57:08 +00001322 elif os.environ.get('TERM') in ('dumb', 'emacs'):
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001323 return lambda text: pipepager(plain(text), os.environ['PAGER'])
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +00001324 else:
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001325 return lambda text: pipepager(text, os.environ['PAGER'])
Ka-Ping Yeea487e4e2005-11-05 04:49:18 +00001326 if os.environ.get('TERM') in ('dumb', 'emacs'):
1327 return plainpager
Andrew MacIntyre54e0eab2002-03-03 03:12:30 +00001328 if sys.platform == 'win32' or sys.platform.startswith('os2'):
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001329 return lambda text: tempfilepager(plain(text), 'more <')
Skip Montanarod404bee2002-09-26 21:44:57 +00001330 if hasattr(os, 'system') and os.system('(less) 2>/dev/null') == 0:
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001331 return lambda text: pipepager(text, 'less')
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001332
1333 import tempfile
Guido van Rossum3b0a3292002-08-09 16:38:32 +00001334 (fd, filename) = tempfile.mkstemp()
1335 os.close(fd)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001336 try:
1337 if hasattr(os, 'system') and os.system('more %s' % filename) == 0:
1338 return lambda text: pipepager(text, 'more')
1339 else:
1340 return ttypager
1341 finally:
1342 os.unlink(filename)
1343
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001344def plain(text):
1345 """Remove boldface formatting from text."""
1346 return re.sub('.\b', '', text)
1347
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001348def pipepager(text, cmd):
1349 """Page through text by feeding it to another program."""
1350 pipe = os.popen(cmd, 'w')
1351 try:
1352 pipe.write(text)
1353 pipe.close()
1354 except IOError:
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001355 pass # Ignore broken pipes caused by quitting the pager program.
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001356
1357def tempfilepager(text, cmd):
1358 """Page through text by invoking a program on a temporary file."""
1359 import tempfile
Tim Peters550e4e52003-02-07 01:53:46 +00001360 filename = tempfile.mktemp()
1361 file = open(filename, 'w')
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001362 file.write(text)
1363 file.close()
1364 try:
Ka-Ping Yeec92cdf72001-03-02 05:54:35 +00001365 os.system(cmd + ' ' + filename)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001366 finally:
1367 os.unlink(filename)
1368
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001369def ttypager(text):
1370 """Page through text on a text terminal."""
1371 lines = split(plain(text), '\n')
1372 try:
1373 import tty
1374 fd = sys.stdin.fileno()
1375 old = tty.tcgetattr(fd)
1376 tty.setcbreak(fd)
1377 getchar = lambda: sys.stdin.read(1)
Ka-Ping Yee457aab22001-02-27 23:36:29 +00001378 except (ImportError, AttributeError):
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001379 tty = None
1380 getchar = lambda: sys.stdin.readline()[:-1][:1]
1381
1382 try:
1383 r = inc = os.environ.get('LINES', 25) - 1
1384 sys.stdout.write(join(lines[:inc], '\n') + '\n')
1385 while lines[r:]:
1386 sys.stdout.write('-- more --')
1387 sys.stdout.flush()
1388 c = getchar()
1389
Raymond Hettingerdbecd932005-02-06 06:57:08 +00001390 if c in ('q', 'Q'):
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001391 sys.stdout.write('\r \r')
1392 break
Raymond Hettingerdbecd932005-02-06 06:57:08 +00001393 elif c in ('\r', '\n'):
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001394 sys.stdout.write('\r \r' + lines[r] + '\n')
1395 r = r + 1
1396 continue
Raymond Hettingerdbecd932005-02-06 06:57:08 +00001397 if c in ('b', 'B', '\x1b'):
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001398 r = r - inc - inc
1399 if r < 0: r = 0
1400 sys.stdout.write('\n' + join(lines[r:r+inc], '\n') + '\n')
1401 r = r + inc
1402
1403 finally:
1404 if tty:
1405 tty.tcsetattr(fd, tty.TCSAFLUSH, old)
1406
1407def plainpager(text):
1408 """Simply print unformatted text. This is the ultimate fallback."""
1409 sys.stdout.write(plain(text))
1410
1411def describe(thing):
Ka-Ping Yee3bda8792001-03-23 13:17:50 +00001412 """Produce a short description of the given thing."""
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001413 if inspect.ismodule(thing):
1414 if thing.__name__ in sys.builtin_module_names:
1415 return 'built-in module ' + thing.__name__
1416 if hasattr(thing, '__path__'):
1417 return 'package ' + thing.__name__
1418 else:
1419 return 'module ' + thing.__name__
1420 if inspect.isbuiltin(thing):
1421 return 'built-in function ' + thing.__name__
Thomas Wouters0e3f5912006-08-11 14:57:12 +00001422 if inspect.isgetsetdescriptor(thing):
1423 return 'getset descriptor %s.%s.%s' % (
1424 thing.__objclass__.__module__, thing.__objclass__.__name__,
1425 thing.__name__)
1426 if inspect.ismemberdescriptor(thing):
1427 return 'member descriptor %s.%s.%s' % (
1428 thing.__objclass__.__module__, thing.__objclass__.__name__,
1429 thing.__name__)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001430 if inspect.isclass(thing):
1431 return 'class ' + thing.__name__
1432 if inspect.isfunction(thing):
1433 return 'function ' + thing.__name__
1434 if inspect.ismethod(thing):
1435 return 'method ' + thing.__name__
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001436 if type(thing) is types.InstanceType:
1437 return 'instance of ' + thing.__class__.__name__
1438 return type(thing).__name__
1439
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001440def locate(path, forceload=0):
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001441 """Locate an object by name or dotted path, importing as necessary."""
Guido van Rossum97dede02003-02-16 01:12:32 +00001442 parts = [part for part in split(path, '.') if part]
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001443 module, n = None, 0
1444 while n < len(parts):
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001445 nextmodule = safeimport(join(parts[:n+1], '.'), forceload)
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001446 if nextmodule: module, n = nextmodule, n + 1
1447 else: break
1448 if module:
1449 object = module
1450 for part in parts[n:]:
1451 try: object = getattr(object, part)
1452 except AttributeError: return None
1453 return object
1454 else:
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001455 if hasattr(__builtin__, path):
1456 return getattr(__builtin__, path)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001457
1458# --------------------------------------- interactive interpreter interface
1459
1460text = TextDoc()
1461html = HTMLDoc()
1462
Ka-Ping Yee45daeb02002-08-11 15:11:33 +00001463def resolve(thing, forceload=0):
1464 """Given an object or a path to an object, get the object and its name."""
1465 if isinstance(thing, str):
1466 object = locate(thing, forceload)
1467 if not object:
1468 raise ImportError, 'no Python documentation found for %r' % thing
1469 return object, thing
1470 else:
1471 return thing, getattr(thing, '__name__', None)
1472
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001473def doc(thing, title='Python Library Documentation: %s', forceload=0):
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001474 """Display text documentation, given an object or a path to an object."""
Ka-Ping Yee45daeb02002-08-11 15:11:33 +00001475 try:
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__
Thomas Woutersfc7bb8c2007-01-15 15:49:28 +00001483 elif not (inspect.ismodule(object) or
1484 inspect.isclass(object) or
1485 inspect.isroutine(object) or
1486 inspect.isgetsetdescriptor(object) or
1487 inspect.ismemberdescriptor(object) or
1488 isinstance(object, property)):
Ka-Ping Yeebba6acc2005-02-19 22:58:26 +00001489 # If the passed object is a piece of data or an instance,
1490 # document its available methods instead of its value.
1491 object = type(object)
1492 desc += ' object'
Ka-Ping Yee45daeb02002-08-11 15:11:33 +00001493 pager(title % desc + '\n\n' + text.document(object, name))
Guido van Rossumb940e112007-01-10 16:19:56 +00001494 except (ImportError, ErrorDuringImport) as value:
Guido van Rossumbe19ed72007-02-09 05:37:30 +00001495 print(value)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001496
Ka-Ping Yee45daeb02002-08-11 15:11:33 +00001497def writedoc(thing, forceload=0):
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001498 """Write HTML documentation to a file in the current directory."""
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001499 try:
Ka-Ping Yee45daeb02002-08-11 15:11:33 +00001500 object, name = resolve(thing, forceload)
1501 page = html.page(describe(object), html.document(object, name))
1502 file = open(name + '.html', 'w')
1503 file.write(page)
1504 file.close()
Guido van Rossumbe19ed72007-02-09 05:37:30 +00001505 print('wrote', name + '.html')
Guido van Rossumb940e112007-01-10 16:19:56 +00001506 except (ImportError, ErrorDuringImport) as value:
Guido van Rossumbe19ed72007-02-09 05:37:30 +00001507 print(value)
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001508
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001509def writedocs(dir, pkgpath='', done=None):
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001510 """Write out HTML documentation for all modules in a directory tree."""
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001511 if done is None: done = {}
Thomas Wouters49fd7fa2006-04-21 10:40:58 +00001512 for importer, modname, ispkg in pkgutil.walk_packages([dir], pkgpath):
1513 writedoc(modname)
1514 return
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001515
Neal Norwitzce96f692006-03-17 06:49:51 +00001516def raw_input(prompt):
1517 sys.stdout.write(prompt)
1518 sys.stdout.flush()
1519 return sys.stdin.readline()
1520
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001521class Helper:
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001522 keywords = {
1523 'and': 'BOOLEAN',
Ka-Ping Yee79c009d2001-04-13 10:53:25 +00001524 'assert': ('ref/assert', ''),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001525 'break': ('ref/break', 'while for'),
1526 'class': ('ref/class', 'CLASSES SPECIALMETHODS'),
1527 'continue': ('ref/continue', 'while for'),
1528 'def': ('ref/function', ''),
1529 'del': ('ref/del', 'BASICMETHODS'),
1530 'elif': 'if',
1531 'else': ('ref/if', 'while for'),
1532 'except': 'try',
1533 'exec': ('ref/exec', ''),
1534 'finally': 'try',
1535 'for': ('ref/for', 'break continue while'),
1536 'from': 'import',
1537 'global': ('ref/global', 'NAMESPACES'),
1538 'if': ('ref/if', 'TRUTHVALUE'),
1539 'import': ('ref/import', 'MODULES'),
1540 'in': ('ref/comparisons', 'SEQUENCEMETHODS2'),
1541 'is': 'COMPARISON',
Neal Norwitz742dde42003-03-30 20:31:34 +00001542 'lambda': ('ref/lambdas', 'FUNCTIONS'),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001543 'not': 'BOOLEAN',
1544 'or': 'BOOLEAN',
Neal Norwitz742dde42003-03-30 20:31:34 +00001545 'pass': ('ref/pass', ''),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001546 'print': ('ref/print', ''),
1547 'raise': ('ref/raise', 'EXCEPTIONS'),
Ka-Ping Yee79c009d2001-04-13 10:53:25 +00001548 'return': ('ref/return', 'FUNCTIONS'),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001549 'try': ('ref/try', 'EXCEPTIONS'),
1550 'while': ('ref/while', 'break continue if TRUTHVALUE'),
Tim Petersfb05c4e2002-10-30 05:21:00 +00001551 'yield': ('ref/yield', ''),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001552 }
1553
1554 topics = {
1555 'TYPES': ('ref/types', 'STRINGS UNICODE NUMBERS SEQUENCES MAPPINGS FUNCTIONS CLASSES MODULES FILES inspect'),
Ka-Ping Yee79c009d2001-04-13 10:53:25 +00001556 'STRINGS': ('ref/strings', 'str UNICODE SEQUENCES STRINGMETHODS FORMATTING TYPES'),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001557 'STRINGMETHODS': ('lib/string-methods', 'STRINGS FORMATTING'),
1558 'FORMATTING': ('lib/typesseq-strings', 'OPERATORS'),
Neal Norwitz54f871e2003-05-26 13:49:54 +00001559 'UNICODE': ('ref/strings', 'encodings unicode SEQUENCES STRINGMETHODS FORMATTING TYPES'),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001560 'NUMBERS': ('ref/numbers', 'INTEGER FLOAT COMPLEX TYPES'),
1561 'INTEGER': ('ref/integers', 'int range'),
1562 'FLOAT': ('ref/floating', 'float math'),
1563 'COMPLEX': ('ref/imaginary', 'complex cmath'),
Ka-Ping Yee79c009d2001-04-13 10:53:25 +00001564 'SEQUENCES': ('lib/typesseq', 'STRINGMETHODS FORMATTING xrange LISTS'),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001565 'MAPPINGS': 'DICTIONARIES',
1566 'FUNCTIONS': ('lib/typesfunctions', 'def TYPES'),
1567 'METHODS': ('lib/typesmethods', 'class def CLASSES TYPES'),
1568 'CODEOBJECTS': ('lib/bltin-code-objects', 'compile FUNCTIONS TYPES'),
Ka-Ping Yee79c009d2001-04-13 10:53:25 +00001569 'TYPEOBJECTS': ('lib/bltin-type-objects', 'types TYPES'),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001570 'FRAMEOBJECTS': 'TYPES',
1571 'TRACEBACKS': 'TYPES',
1572 'NONE': ('lib/bltin-null-object', ''),
1573 'ELLIPSIS': ('lib/bltin-ellipsis-object', 'SLICINGS'),
1574 'FILES': ('lib/bltin-file-objects', ''),
1575 'SPECIALATTRIBUTES': ('lib/specialattrs', ''),
1576 'CLASSES': ('ref/types', 'class SPECIALMETHODS PRIVATENAMES'),
1577 'MODULES': ('lib/typesmodules', 'import'),
1578 'PACKAGES': 'import',
Neal Norwitz2eca4402006-08-29 04:40:24 +00001579 'EXPRESSIONS': ('ref/summary', 'lambda or and not in is BOOLEAN COMPARISON BITWISE SHIFTING BINARY FORMATTING POWER UNARY ATTRIBUTES SUBSCRIPTS SLICINGS CALLS TUPLES LISTS DICTIONARIES'),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001580 'OPERATORS': 'EXPRESSIONS',
1581 'PRECEDENCE': 'EXPRESSIONS',
1582 'OBJECTS': ('ref/objects', 'TYPES'),
1583 'SPECIALMETHODS': ('ref/specialnames', 'BASICMETHODS ATTRIBUTEMETHODS CALLABLEMETHODS SEQUENCEMETHODS1 MAPPINGMETHODS SEQUENCEMETHODS2 NUMBERMETHODS CLASSES'),
Ka-Ping Yee79c009d2001-04-13 10:53:25 +00001584 'BASICMETHODS': ('ref/customization', 'cmp hash repr str SPECIALMETHODS'),
1585 'ATTRIBUTEMETHODS': ('ref/attribute-access', 'ATTRIBUTES SPECIALMETHODS'),
1586 'CALLABLEMETHODS': ('ref/callable-types', 'CALLS SPECIALMETHODS'),
1587 'SEQUENCEMETHODS1': ('ref/sequence-types', 'SEQUENCES SEQUENCEMETHODS2 SPECIALMETHODS'),
1588 'SEQUENCEMETHODS2': ('ref/sequence-methods', 'SEQUENCES SEQUENCEMETHODS1 SPECIALMETHODS'),
1589 'MAPPINGMETHODS': ('ref/sequence-types', 'MAPPINGS SPECIALMETHODS'),
1590 'NUMBERMETHODS': ('ref/numeric-types', 'NUMBERS AUGMENTEDASSIGNMENT SPECIALMETHODS'),
Neal Norwitz54f871e2003-05-26 13:49:54 +00001591 'EXECUTION': ('ref/execmodel', 'NAMESPACES DYNAMICFEATURES EXCEPTIONS'),
1592 'NAMESPACES': ('ref/naming', 'global ASSIGNMENT DELETION DYNAMICFEATURES'),
1593 'DYNAMICFEATURES': ('ref/dynamic-features', ''),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001594 'SCOPING': 'NAMESPACES',
1595 'FRAMES': 'NAMESPACES',
1596 'EXCEPTIONS': ('ref/exceptions', 'try except finally raise'),
Neal Norwitz54f871e2003-05-26 13:49:54 +00001597 'COERCIONS': ('ref/coercion-rules','CONVERSIONS'),
1598 'CONVERSIONS': ('ref/conversions', 'COERCIONS'),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001599 'IDENTIFIERS': ('ref/identifiers', 'keywords SPECIALIDENTIFIERS'),
1600 'SPECIALIDENTIFIERS': ('ref/id-classes', ''),
Ka-Ping Yee79c009d2001-04-13 10:53:25 +00001601 'PRIVATENAMES': ('ref/atom-identifiers', ''),
Neal Norwitz2eca4402006-08-29 04:40:24 +00001602 'LITERALS': ('ref/atom-literals', 'STRINGS NUMBERS TUPLELITERALS LISTLITERALS DICTIONARYLITERALS'),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001603 'TUPLES': 'SEQUENCES',
Ka-Ping Yee79c009d2001-04-13 10:53:25 +00001604 'TUPLELITERALS': ('ref/exprlists', 'TUPLES LITERALS'),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001605 'LISTS': ('lib/typesseq-mutable', 'LISTLITERALS'),
Ka-Ping Yee79c009d2001-04-13 10:53:25 +00001606 'LISTLITERALS': ('ref/lists', 'LISTS LITERALS'),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001607 'DICTIONARIES': ('lib/typesmapping', 'DICTIONARYLITERALS'),
Ka-Ping Yee79c009d2001-04-13 10:53:25 +00001608 'DICTIONARYLITERALS': ('ref/dict', 'DICTIONARIES LITERALS'),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001609 'ATTRIBUTES': ('ref/attribute-references', 'getattr hasattr setattr ATTRIBUTEMETHODS'),
1610 'SUBSCRIPTS': ('ref/subscriptions', 'SEQUENCEMETHODS1'),
1611 'SLICINGS': ('ref/slicings', 'SEQUENCEMETHODS2'),
1612 'CALLS': ('ref/calls', 'EXPRESSIONS'),
1613 'POWER': ('ref/power', 'EXPRESSIONS'),
1614 'UNARY': ('ref/unary', 'EXPRESSIONS'),
1615 'BINARY': ('ref/binary', 'EXPRESSIONS'),
1616 'SHIFTING': ('ref/shifting', 'EXPRESSIONS'),
1617 'BITWISE': ('ref/bitwise', 'EXPRESSIONS'),
1618 'COMPARISON': ('ref/comparisons', 'EXPRESSIONS BASICMETHODS'),
Neal Norwitzf98159c2003-02-07 20:49:40 +00001619 'BOOLEAN': ('ref/Booleans', 'EXPRESSIONS TRUTHVALUE'),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001620 'ASSERTION': 'assert',
1621 'ASSIGNMENT': ('ref/assignment', 'AUGMENTEDASSIGNMENT'),
Ka-Ping Yee79c009d2001-04-13 10:53:25 +00001622 'AUGMENTEDASSIGNMENT': ('ref/augassign', 'NUMBERMETHODS'),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001623 'DELETION': 'del',
1624 'PRINTING': 'print',
1625 'RETURNING': 'return',
1626 'IMPORTING': 'import',
1627 'CONDITIONAL': 'if',
1628 'LOOPING': ('ref/compound', 'for while break continue'),
1629 'TRUTHVALUE': ('lib/truth', 'if while and or not BASICMETHODS'),
Ka-Ping Yee79c009d2001-04-13 10:53:25 +00001630 'DEBUGGING': ('lib/module-pdb', 'pdb'),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001631 }
1632
1633 def __init__(self, input, output):
1634 self.input = input
1635 self.output = output
1636 self.docdir = None
1637 execdir = os.path.dirname(sys.executable)
1638 homedir = os.environ.get('PYTHONHOME')
1639 for dir in [os.environ.get('PYTHONDOCS'),
1640 homedir and os.path.join(homedir, 'doc'),
1641 os.path.join(execdir, 'doc'),
1642 '/usr/doc/python-docs-' + split(sys.version)[0],
1643 '/usr/doc/python-' + split(sys.version)[0],
1644 '/usr/doc/python-docs-' + sys.version[:3],
Jack Jansenb2628b02002-08-23 08:40:42 +00001645 '/usr/doc/python-' + sys.version[:3],
1646 os.path.join(sys.prefix, 'Resources/English.lproj/Documentation')]:
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001647 if dir and os.path.isdir(os.path.join(dir, 'lib')):
1648 self.docdir = dir
1649
Ka-Ping Yee79c009d2001-04-13 10:53:25 +00001650 def __repr__(self):
Ka-Ping Yee9bc576b2001-04-13 13:57:31 +00001651 if inspect.stack()[1][3] == '?':
Ka-Ping Yee79c009d2001-04-13 10:53:25 +00001652 self()
1653 return ''
Ka-Ping Yee9bc576b2001-04-13 13:57:31 +00001654 return '<pydoc.Helper instance>'
Ka-Ping Yee79c009d2001-04-13 10:53:25 +00001655
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001656 def __call__(self, request=None):
1657 if request is not None:
1658 self.help(request)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001659 else:
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001660 self.intro()
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001661 self.interact()
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001662 self.output.write('''
Fred Drakee61967f2001-05-10 18:41:02 +00001663You are now leaving help and returning to the Python interpreter.
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001664If you want to ask for help on a particular object directly from the
1665interpreter, you can type "help(object)". Executing "help('string')"
1666has the same effect as typing a particular string at the help> prompt.
1667''')
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001668
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001669 def interact(self):
1670 self.output.write('\n')
Guido van Rossum8ca162f2002-04-07 06:36:23 +00001671 while True:
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001672 try:
Johannes Gijsberse7691d32004-08-17 13:21:53 +00001673 request = self.getline('help> ')
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001674 if not request: break
Johannes Gijsberse7691d32004-08-17 13:21:53 +00001675 except (KeyboardInterrupt, EOFError):
1676 break
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001677 request = strip(replace(request, '"', '', "'", ''))
Raymond Hettingerdbecd932005-02-06 06:57:08 +00001678 if lower(request) in ('q', 'quit'): break
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001679 self.help(request)
1680
Johannes Gijsberse7691d32004-08-17 13:21:53 +00001681 def getline(self, prompt):
1682 """Read one line, using raw_input when available."""
1683 if self.input is sys.stdin:
1684 return raw_input(prompt)
1685 else:
1686 self.output.write(prompt)
1687 self.output.flush()
1688 return self.input.readline()
1689
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001690 def help(self, request):
1691 if type(request) is type(''):
1692 if request == 'help': self.intro()
1693 elif request == 'keywords': self.listkeywords()
1694 elif request == 'topics': self.listtopics()
1695 elif request == 'modules': self.listmodules()
1696 elif request[:8] == 'modules ':
1697 self.listmodules(split(request)[1])
Raymond Hettinger54f02222002-06-01 14:18:47 +00001698 elif request in self.keywords: self.showtopic(request)
1699 elif request in self.topics: self.showtopic(request)
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001700 elif request: doc(request, 'Help on %s:')
1701 elif isinstance(request, Helper): self()
1702 else: doc(request, 'Help on %s:')
1703 self.output.write('\n')
1704
1705 def intro(self):
1706 self.output.write('''
1707Welcome to Python %s! This is the online help utility.
1708
1709If this is your first time using Python, you should definitely check out
1710the tutorial on the Internet at http://www.python.org/doc/tut/.
1711
1712Enter the name of any module, keyword, or topic to get help on writing
1713Python programs and using Python modules. To quit this help utility and
1714return to the interpreter, just type "quit".
1715
1716To get a list of available modules, keywords, or topics, type "modules",
1717"keywords", or "topics". Each module also comes with a one-line summary
1718of what it does; to list the modules whose summaries contain a given word
1719such as "spam", type "modules spam".
1720''' % sys.version[:3])
1721
1722 def list(self, items, columns=4, width=80):
1723 items = items[:]
1724 items.sort()
1725 colw = width / columns
1726 rows = (len(items) + columns - 1) / columns
1727 for row in range(rows):
1728 for col in range(columns):
1729 i = col * rows + row
1730 if i < len(items):
1731 self.output.write(items[i])
1732 if col < columns - 1:
1733 self.output.write(' ' + ' ' * (colw-1 - len(items[i])))
1734 self.output.write('\n')
1735
1736 def listkeywords(self):
1737 self.output.write('''
1738Here is a list of the Python keywords. Enter any keyword to get more help.
1739
1740''')
1741 self.list(self.keywords.keys())
1742
1743 def listtopics(self):
1744 self.output.write('''
1745Here is a list of available topics. Enter any topic name to get more help.
1746
1747''')
1748 self.list(self.topics.keys())
1749
1750 def showtopic(self, topic):
1751 if not self.docdir:
1752 self.output.write('''
1753Sorry, topic and keyword documentation is not available because the Python
1754HTML documentation files could not be found. If you have installed them,
1755please set the environment variable PYTHONDOCS to indicate their location.
Thomas Wouters902d6eb2007-01-09 23:18:33 +00001756
1757On the Microsoft Windows operating system, the files can be built by
1758running "hh -decompile . PythonNN.chm" in the C:\PythonNN\Doc> directory.
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001759''')
1760 return
1761 target = self.topics.get(topic, self.keywords.get(topic))
1762 if not target:
1763 self.output.write('no documentation found for %s\n' % repr(topic))
1764 return
1765 if type(target) is type(''):
1766 return self.showtopic(target)
1767
1768 filename, xrefs = target
1769 filename = self.docdir + '/' + filename + '.html'
1770 try:
1771 file = open(filename)
1772 except:
1773 self.output.write('could not read docs from %s\n' % filename)
1774 return
1775
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001776 divpat = re.compile('<div[^>]*navigat.*?</div.*?>', re.I | re.S)
1777 addrpat = re.compile('<address.*?>.*?</address.*?>', re.I | re.S)
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001778 document = re.sub(addrpat, '', re.sub(divpat, '', file.read()))
1779 file.close()
1780
1781 import htmllib, formatter, StringIO
1782 buffer = StringIO.StringIO()
1783 parser = htmllib.HTMLParser(
1784 formatter.AbstractFormatter(formatter.DumbWriter(buffer)))
1785 parser.start_table = parser.do_p
1786 parser.end_table = lambda parser=parser: parser.do_p({})
1787 parser.start_tr = parser.do_br
1788 parser.start_td = parser.start_th = lambda a, b=buffer: b.write('\t')
1789 parser.feed(document)
1790 buffer = replace(buffer.getvalue(), '\xa0', ' ', '\n', '\n ')
1791 pager(' ' + strip(buffer) + '\n')
Ka-Ping Yeeda793892001-04-13 11:02:51 +00001792 if xrefs:
1793 buffer = StringIO.StringIO()
1794 formatter.DumbWriter(buffer).send_flowing_data(
1795 'Related help topics: ' + join(split(xrefs), ', ') + '\n')
1796 self.output.write('\n%s\n' % buffer.getvalue())
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001797
1798 def listmodules(self, key=''):
1799 if key:
1800 self.output.write('''
1801Here is a list of matching modules. Enter any module name to get more help.
1802
1803''')
1804 apropos(key)
1805 else:
1806 self.output.write('''
1807Please wait a moment while I gather a list of all available modules...
1808
1809''')
1810 modules = {}
1811 def callback(path, modname, desc, modules=modules):
1812 if modname and modname[-9:] == '.__init__':
1813 modname = modname[:-9] + ' (package)'
1814 if find(modname, '.') < 0:
1815 modules[modname] = 1
1816 ModuleScanner().run(callback)
1817 self.list(modules.keys())
1818 self.output.write('''
1819Enter any module name to get more help. Or, type "modules spam" to search
1820for modules whose descriptions contain the word "spam".
1821''')
1822
1823help = Helper(sys.stdin, sys.stdout)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001824
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001825class Scanner:
1826 """A generic tree iterator."""
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001827 def __init__(self, roots, children, descendp):
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001828 self.roots = roots[:]
1829 self.state = []
1830 self.children = children
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001831 self.descendp = descendp
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001832
1833 def next(self):
1834 if not self.state:
1835 if not self.roots:
1836 return None
1837 root = self.roots.pop(0)
1838 self.state = [(root, self.children(root))]
1839 node, children = self.state[-1]
1840 if not children:
1841 self.state.pop()
1842 return self.next()
1843 child = children.pop(0)
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001844 if self.descendp(child):
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001845 self.state.append((child, self.children(child)))
1846 return child
1847
Thomas Wouters49fd7fa2006-04-21 10:40:58 +00001848
1849class ModuleScanner:
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001850 """An interruptible scanner that searches module synopses."""
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001851
Ka-Ping Yee66246962001-04-12 11:59:50 +00001852 def run(self, callback, key=None, completer=None):
1853 if key: key = lower(key)
Guido van Rossum8ca162f2002-04-07 06:36:23 +00001854 self.quit = False
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001855 seen = {}
1856
1857 for modname in sys.builtin_module_names:
Ka-Ping Yee239432a2001-03-02 02:45:08 +00001858 if modname != '__main__':
1859 seen[modname] = 1
Ka-Ping Yee66246962001-04-12 11:59:50 +00001860 if key is None:
1861 callback(None, modname, '')
1862 else:
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001863 desc = split(__import__(modname).__doc__ or '', '\n')[0]
Ka-Ping Yee66246962001-04-12 11:59:50 +00001864 if find(lower(modname + ' - ' + desc), key) >= 0:
1865 callback(None, modname, desc)
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001866
Thomas Wouters49fd7fa2006-04-21 10:40:58 +00001867 for importer, modname, ispkg in pkgutil.walk_packages():
1868 if self.quit:
1869 break
1870 if key is None:
1871 callback(None, modname, '')
1872 else:
1873 loader = importer.find_module(modname)
1874 if hasattr(loader,'get_source'):
1875 import StringIO
1876 desc = source_synopsis(
1877 StringIO.StringIO(loader.get_source(modname))
1878 ) or ''
1879 if hasattr(loader,'get_filename'):
1880 path = loader.get_filename(modname)
Ka-Ping Yee66246962001-04-12 11:59:50 +00001881 else:
Thomas Wouters49fd7fa2006-04-21 10:40:58 +00001882 path = None
1883 else:
1884 module = loader.load_module(modname)
1885 desc = (module.__doc__ or '').splitlines()[0]
1886 path = getattr(module,'__file__',None)
1887 if find(lower(modname + ' - ' + desc), key) >= 0:
1888 callback(path, modname, desc)
1889
1890 if completer:
1891 completer()
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001892
1893def apropos(key):
1894 """Print all the one-line module summaries that contain a substring."""
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001895 def callback(path, modname, desc):
1896 if modname[-9:] == '.__init__':
1897 modname = modname[:-9] + ' (package)'
Guido van Rossumbe19ed72007-02-09 05:37:30 +00001898 print(modname, desc and '- ' + desc)
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +00001899 try: import warnings
1900 except ImportError: pass
1901 else: warnings.filterwarnings('ignore') # ignore problems during import
Ka-Ping Yee66246962001-04-12 11:59:50 +00001902 ModuleScanner().run(callback, key)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001903
1904# --------------------------------------------------- web browser interface
1905
Ka-Ping Yee66246962001-04-12 11:59:50 +00001906def serve(port, callback=None, completer=None):
Ka-Ping Yeefd540692001-04-12 12:54:36 +00001907 import BaseHTTPServer, mimetools, select
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001908
1909 # Patch up mimetools.Message so it doesn't break if rfc822 is reloaded.
1910 class Message(mimetools.Message):
1911 def __init__(self, fp, seekable=1):
1912 Message = self.__class__
1913 Message.__bases__[0].__bases__[0].__init__(self, fp, seekable)
1914 self.encodingheader = self.getheader('content-transfer-encoding')
1915 self.typeheader = self.getheader('content-type')
1916 self.parsetype()
1917 self.parseplist()
1918
1919 class DocHandler(BaseHTTPServer.BaseHTTPRequestHandler):
1920 def send_document(self, title, contents):
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001921 try:
1922 self.send_response(200)
1923 self.send_header('Content-Type', 'text/html')
1924 self.end_headers()
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001925 self.wfile.write(html.page(title, contents))
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001926 except IOError: pass
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001927
1928 def do_GET(self):
1929 path = self.path
1930 if path[-5:] == '.html': path = path[:-5]
1931 if path[:1] == '/': path = path[1:]
1932 if path and path != '.':
1933 try:
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001934 obj = locate(path, forceload=1)
Guido van Rossumb940e112007-01-10 16:19:56 +00001935 except ErrorDuringImport as value:
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001936 self.send_document(path, html.escape(str(value)))
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001937 return
Ka-Ping Yee3bda8792001-03-23 13:17:50 +00001938 if obj:
1939 self.send_document(describe(obj), html.document(obj, path))
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001940 else:
1941 self.send_document(path,
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001942'no Python documentation found for %s' % repr(path))
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001943 else:
1944 heading = html.heading(
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001945'<big><big><strong>Python: Index of Modules</strong></big></big>',
1946'#ffffff', '#7799ee')
Ka-Ping Yee239432a2001-03-02 02:45:08 +00001947 def bltinlink(name):
1948 return '<a href="%s.html">%s</a>' % (name, name)
Ka-Ping Yee3bda8792001-03-23 13:17:50 +00001949 names = filter(lambda x: x != '__main__',
1950 sys.builtin_module_names)
Ka-Ping Yee239432a2001-03-02 02:45:08 +00001951 contents = html.multicolumn(names, bltinlink)
1952 indices = ['<p>' + html.bigsection(
1953 'Built-in Modules', '#ffffff', '#ee77aa', contents)]
1954
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001955 seen = {}
Thomas Wouters49fd7fa2006-04-21 10:40:58 +00001956 for dir in sys.path:
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001957 indices.append(html.index(dir, seen))
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001958 contents = heading + join(indices) + '''<p align=right>
Tim Peters2306d242001-09-25 03:18:32 +00001959<font color="#909090" face="helvetica, arial"><strong>
1960pydoc</strong> by Ka-Ping Yee &lt;ping@lfw.org&gt;</font>'''
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001961 self.send_document('Index of Modules', contents)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001962
1963 def log_message(self, *args): pass
1964
Ka-Ping Yeefd540692001-04-12 12:54:36 +00001965 class DocServer(BaseHTTPServer.HTTPServer):
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001966 def __init__(self, port, callback):
Ka-Ping Yeedb8ed152001-03-02 05:58:17 +00001967 host = (sys.platform == 'mac') and '127.0.0.1' or 'localhost'
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001968 self.address = ('', port)
Ka-Ping Yeedb8ed152001-03-02 05:58:17 +00001969 self.url = 'http://%s:%d/' % (host, port)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001970 self.callback = callback
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001971 self.base.__init__(self, self.address, self.handler)
1972
1973 def serve_until_quit(self):
1974 import select
Guido van Rossum8ca162f2002-04-07 06:36:23 +00001975 self.quit = False
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001976 while not self.quit:
1977 rd, wr, ex = select.select([self.socket.fileno()], [], [], 1)
1978 if rd: self.handle_request()
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001979
1980 def server_activate(self):
1981 self.base.server_activate(self)
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001982 if self.callback: self.callback(self)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001983
1984 DocServer.base = BaseHTTPServer.HTTPServer
1985 DocServer.handler = DocHandler
1986 DocHandler.MessageClass = Message
1987 try:
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001988 try:
1989 DocServer(port, callback).serve_until_quit()
1990 except (KeyboardInterrupt, select.error):
1991 pass
1992 finally:
Ka-Ping Yee66246962001-04-12 11:59:50 +00001993 if completer: completer()
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001994
1995# ----------------------------------------------------- graphical interface
1996
1997def gui():
1998 """Graphical interface (starts web server and pops up a control window)."""
1999 class GUI:
2000 def __init__(self, window, port=7464):
2001 self.window = window
2002 self.server = None
2003 self.scanner = None
2004
2005 import Tkinter
2006 self.server_frm = Tkinter.Frame(window)
2007 self.title_lbl = Tkinter.Label(self.server_frm,
2008 text='Starting server...\n ')
2009 self.open_btn = Tkinter.Button(self.server_frm,
2010 text='open browser', command=self.open, state='disabled')
2011 self.quit_btn = Tkinter.Button(self.server_frm,
2012 text='quit serving', command=self.quit, state='disabled')
2013
2014 self.search_frm = Tkinter.Frame(window)
2015 self.search_lbl = Tkinter.Label(self.search_frm, text='Search for')
2016 self.search_ent = Tkinter.Entry(self.search_frm)
2017 self.search_ent.bind('<Return>', self.search)
2018 self.stop_btn = Tkinter.Button(self.search_frm,
2019 text='stop', pady=0, command=self.stop, state='disabled')
2020 if sys.platform == 'win32':
Ka-Ping Yeec92cdf72001-03-02 05:54:35 +00002021 # Trying to hide and show this button crashes under Windows.
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002022 self.stop_btn.pack(side='right')
2023
2024 self.window.title('pydoc')
2025 self.window.protocol('WM_DELETE_WINDOW', self.quit)
2026 self.title_lbl.pack(side='top', fill='x')
2027 self.open_btn.pack(side='left', fill='x', expand=1)
2028 self.quit_btn.pack(side='right', fill='x', expand=1)
2029 self.server_frm.pack(side='top', fill='x')
2030
2031 self.search_lbl.pack(side='left')
2032 self.search_ent.pack(side='right', fill='x', expand=1)
2033 self.search_frm.pack(side='top', fill='x')
2034 self.search_ent.focus_set()
2035
Ka-Ping Yeec92cdf72001-03-02 05:54:35 +00002036 font = ('helvetica', sys.platform == 'win32' and 8 or 10)
Ka-Ping Yee239432a2001-03-02 02:45:08 +00002037 self.result_lst = Tkinter.Listbox(window, font=font, height=6)
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002038 self.result_lst.bind('<Button-1>', self.select)
2039 self.result_lst.bind('<Double-Button-1>', self.goto)
2040 self.result_scr = Tkinter.Scrollbar(window,
2041 orient='vertical', command=self.result_lst.yview)
2042 self.result_lst.config(yscrollcommand=self.result_scr.set)
2043
2044 self.result_frm = Tkinter.Frame(window)
2045 self.goto_btn = Tkinter.Button(self.result_frm,
2046 text='go to selected', command=self.goto)
2047 self.hide_btn = Tkinter.Button(self.result_frm,
2048 text='hide results', command=self.hide)
2049 self.goto_btn.pack(side='left', fill='x', expand=1)
2050 self.hide_btn.pack(side='right', fill='x', expand=1)
2051
2052 self.window.update()
2053 self.minwidth = self.window.winfo_width()
2054 self.minheight = self.window.winfo_height()
2055 self.bigminheight = (self.server_frm.winfo_reqheight() +
2056 self.search_frm.winfo_reqheight() +
2057 self.result_lst.winfo_reqheight() +
2058 self.result_frm.winfo_reqheight())
2059 self.bigwidth, self.bigheight = self.minwidth, self.bigminheight
2060 self.expanded = 0
2061 self.window.wm_geometry('%dx%d' % (self.minwidth, self.minheight))
2062 self.window.wm_minsize(self.minwidth, self.minheight)
Martin v. Löwis5b26abb2002-12-28 09:23:09 +00002063 self.window.tk.willdispatch()
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002064
2065 import threading
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00002066 threading.Thread(
2067 target=serve, args=(port, self.ready, self.quit)).start()
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002068
2069 def ready(self, server):
2070 self.server = server
2071 self.title_lbl.config(
2072 text='Python documentation server at\n' + server.url)
2073 self.open_btn.config(state='normal')
2074 self.quit_btn.config(state='normal')
2075
Ka-Ping Yee239432a2001-03-02 02:45:08 +00002076 def open(self, event=None, url=None):
2077 url = url or self.server.url
2078 try:
2079 import webbrowser
2080 webbrowser.open(url)
2081 except ImportError: # pre-webbrowser.py compatibility
Ka-Ping Yeec92cdf72001-03-02 05:54:35 +00002082 if sys.platform == 'win32':
Ka-Ping Yee239432a2001-03-02 02:45:08 +00002083 os.system('start "%s"' % url)
2084 elif sys.platform == 'mac':
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +00002085 try: import ic
Ka-Ping Yee239432a2001-03-02 02:45:08 +00002086 except ImportError: pass
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +00002087 else: ic.launchurl(url)
Ka-Ping Yee239432a2001-03-02 02:45:08 +00002088 else:
2089 rc = os.system('netscape -remote "openURL(%s)" &' % url)
2090 if rc: os.system('netscape "%s" &' % url)
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002091
2092 def quit(self, event=None):
2093 if self.server:
2094 self.server.quit = 1
2095 self.window.quit()
2096
2097 def search(self, event=None):
2098 key = self.search_ent.get()
2099 self.stop_btn.pack(side='right')
2100 self.stop_btn.config(state='normal')
2101 self.search_lbl.config(text='Searching for "%s"...' % key)
2102 self.search_ent.forget()
2103 self.search_lbl.pack(side='left')
2104 self.result_lst.delete(0, 'end')
2105 self.goto_btn.config(state='disabled')
2106 self.expand()
2107
2108 import threading
2109 if self.scanner:
2110 self.scanner.quit = 1
2111 self.scanner = ModuleScanner()
2112 threading.Thread(target=self.scanner.run,
Ka-Ping Yee6dcfa382001-04-12 20:27:31 +00002113 args=(self.update, key, self.done)).start()
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002114
2115 def update(self, path, modname, desc):
2116 if modname[-9:] == '.__init__':
2117 modname = modname[:-9] + ' (package)'
2118 self.result_lst.insert('end',
2119 modname + ' - ' + (desc or '(no description)'))
2120
2121 def stop(self, event=None):
2122 if self.scanner:
2123 self.scanner.quit = 1
2124 self.scanner = None
2125
2126 def done(self):
2127 self.scanner = None
2128 self.search_lbl.config(text='Search for')
2129 self.search_lbl.pack(side='left')
2130 self.search_ent.pack(side='right', fill='x', expand=1)
2131 if sys.platform != 'win32': self.stop_btn.forget()
2132 self.stop_btn.config(state='disabled')
2133
2134 def select(self, event=None):
2135 self.goto_btn.config(state='normal')
2136
2137 def goto(self, event=None):
2138 selection = self.result_lst.curselection()
2139 if selection:
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002140 modname = split(self.result_lst.get(selection[0]))[0]
Ka-Ping Yee239432a2001-03-02 02:45:08 +00002141 self.open(url=self.server.url + modname + '.html')
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002142
2143 def collapse(self):
2144 if not self.expanded: return
2145 self.result_frm.forget()
2146 self.result_scr.forget()
2147 self.result_lst.forget()
2148 self.bigwidth = self.window.winfo_width()
2149 self.bigheight = self.window.winfo_height()
2150 self.window.wm_geometry('%dx%d' % (self.minwidth, self.minheight))
2151 self.window.wm_minsize(self.minwidth, self.minheight)
2152 self.expanded = 0
2153
2154 def expand(self):
2155 if self.expanded: return
2156 self.result_frm.pack(side='bottom', fill='x')
2157 self.result_scr.pack(side='right', fill='y')
2158 self.result_lst.pack(side='top', fill='both', expand=1)
2159 self.window.wm_geometry('%dx%d' % (self.bigwidth, self.bigheight))
2160 self.window.wm_minsize(self.minwidth, self.bigminheight)
2161 self.expanded = 1
2162
2163 def hide(self, event=None):
2164 self.stop()
2165 self.collapse()
2166
2167 import Tkinter
2168 try:
Martin v. Löwise09bd932004-08-22 16:13:26 +00002169 root = Tkinter.Tk()
2170 # Tk will crash if pythonw.exe has an XP .manifest
2171 # file and the root has is not destroyed explicitly.
2172 # If the problem is ever fixed in Tk, the explicit
2173 # destroy can go.
2174 try:
2175 gui = GUI(root)
2176 root.mainloop()
2177 finally:
2178 root.destroy()
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002179 except KeyboardInterrupt:
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002180 pass
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002181
2182# -------------------------------------------------- command-line interface
2183
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00002184def ispath(x):
Ka-Ping Yee45daeb02002-08-11 15:11:33 +00002185 return isinstance(x, str) and find(x, os.sep) >= 0
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00002186
Ka-Ping Yee1d384632001-03-01 00:24:32 +00002187def cli():
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002188 """Command-line interface (looks at sys.argv to decide what to do)."""
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002189 import getopt
2190 class BadUsage: pass
2191
Ka-Ping Yee3bda8792001-03-23 13:17:50 +00002192 # Scripts don't get the current directory in their path by default.
Ka-Ping Yeef78a81b2001-03-27 08:13:42 +00002193 scriptdir = os.path.dirname(sys.argv[0])
2194 if scriptdir in sys.path:
2195 sys.path.remove(scriptdir)
Ka-Ping Yee3bda8792001-03-23 13:17:50 +00002196 sys.path.insert(0, '.')
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002197
Ka-Ping Yee3bda8792001-03-23 13:17:50 +00002198 try:
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002199 opts, args = getopt.getopt(sys.argv[1:], 'gk:p:w')
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002200 writing = 0
2201
2202 for opt, val in opts:
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002203 if opt == '-g':
2204 gui()
2205 return
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002206 if opt == '-k':
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002207 apropos(val)
2208 return
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002209 if opt == '-p':
2210 try:
2211 port = int(val)
2212 except ValueError:
2213 raise BadUsage
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002214 def ready(server):
Guido van Rossumbe19ed72007-02-09 05:37:30 +00002215 print('pydoc server ready at %s' % server.url)
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00002216 def stopped():
Guido van Rossumbe19ed72007-02-09 05:37:30 +00002217 print('pydoc server stopped')
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00002218 serve(port, ready, stopped)
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002219 return
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002220 if opt == '-w':
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002221 writing = 1
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002222
2223 if not args: raise BadUsage
2224 for arg in args:
Ka-Ping Yee45daeb02002-08-11 15:11:33 +00002225 if ispath(arg) and not os.path.exists(arg):
Guido van Rossumbe19ed72007-02-09 05:37:30 +00002226 print('file %r does not exist' % arg)
Ka-Ping Yee45daeb02002-08-11 15:11:33 +00002227 break
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002228 try:
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00002229 if ispath(arg) and os.path.isfile(arg):
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002230 arg = importfile(arg)
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00002231 if writing:
2232 if ispath(arg) and os.path.isdir(arg):
2233 writedocs(arg)
2234 else:
2235 writedoc(arg)
2236 else:
Martin v. Löwisb8c084e2003-06-14 09:03:46 +00002237 help.help(arg)
Guido van Rossumb940e112007-01-10 16:19:56 +00002238 except ErrorDuringImport as value:
Guido van Rossumbe19ed72007-02-09 05:37:30 +00002239 print(value)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002240
2241 except (getopt.error, BadUsage):
Martin v. Löwisf9b08b82003-10-31 13:05:21 +00002242 cmd = os.path.basename(sys.argv[0])
Guido van Rossumbe19ed72007-02-09 05:37:30 +00002243 print("""pydoc - the Python documentation tool
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002244
2245%s <name> ...
2246 Show text documentation on something. <name> may be the name of a
Martin v. Löwisb8c084e2003-06-14 09:03:46 +00002247 Python keyword, topic, function, module, or package, or a dotted
2248 reference to a class or function within a module or module in a
2249 package. If <name> contains a '%s', it is used as the path to a
2250 Python source file to document. If name is 'keywords', 'topics',
2251 or 'modules', a listing of these things is displayed.
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002252
2253%s -k <keyword>
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002254 Search for a keyword in the synopsis lines of all available modules.
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002255
2256%s -p <port>
2257 Start an HTTP server on the given port on the local machine.
2258
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002259%s -g
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +00002260 Pop up a graphical interface for finding and serving documentation.
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002261
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002262%s -w <name> ...
2263 Write out the HTML documentation for a module to a file in the current
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +00002264 directory. If <name> contains a '%s', it is treated as a filename; if
2265 it names a directory, documentation is written for all the contents.
Guido van Rossumbe19ed72007-02-09 05:37:30 +00002266""" % (cmd, os.sep, cmd, cmd, cmd, cmd, os.sep))
Ka-Ping Yee1d384632001-03-01 00:24:32 +00002267
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002268if __name__ == '__main__': cli()