blob: 97474569ba4e38b9075184cc63024a390736193c [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
Guido van Rossuma01a8b62007-05-27 09:20:14 +000055import sys, imp, os, re, inspect, __builtin__, pkgutil
Ka-Ping Yeedd175342001-02-27 14:43:46 +000056from repr import Repr
Thomas Wouters49fd7fa2006-04-21 10:40:58 +000057try:
58 from collections import deque
59except ImportError:
60 # Python 2.3 compatibility
61 class deque(list):
62 def popleft(self):
63 return self.pop(0)
Ka-Ping Yeedd175342001-02-27 14:43:46 +000064
65# --------------------------------------------------------- common routines
66
Ka-Ping Yeedd175342001-02-27 14:43:46 +000067def pathdirs():
68 """Convert sys.path into a list of absolute, existing, unique paths."""
69 dirs = []
Ka-Ping Yee1d384632001-03-01 00:24:32 +000070 normdirs = []
Ka-Ping Yeedd175342001-02-27 14:43:46 +000071 for dir in sys.path:
72 dir = os.path.abspath(dir or '.')
Ka-Ping Yee1d384632001-03-01 00:24:32 +000073 normdir = os.path.normcase(dir)
74 if normdir not in normdirs and os.path.isdir(dir):
Ka-Ping Yeedd175342001-02-27 14:43:46 +000075 dirs.append(dir)
Ka-Ping Yee1d384632001-03-01 00:24:32 +000076 normdirs.append(normdir)
Ka-Ping Yeedd175342001-02-27 14:43:46 +000077 return dirs
78
79def getdoc(object):
80 """Get the doc string or comments for an object."""
Ka-Ping Yee3bda8792001-03-23 13:17:50 +000081 result = inspect.getdoc(object) or inspect.getcomments(object)
Neal Norwitz9d72bb42007-04-17 08:48:32 +000082 return result and re.sub('^ *\n', '', result.rstrip()) or ''
Ka-Ping Yeedd175342001-02-27 14:43:46 +000083
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +000084def splitdoc(doc):
85 """Split a doc string into a synopsis line (if any) and the rest."""
Neal Norwitz9d72bb42007-04-17 08:48:32 +000086 lines = doc.strip().split('\n')
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +000087 if len(lines) == 1:
88 return lines[0], ''
Neal Norwitz9d72bb42007-04-17 08:48:32 +000089 elif len(lines) >= 2 and not lines[1].rstrip():
90 return lines[0], '\n'.join(lines[2:])
91 return '', '\n'.join(lines)
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +000092
Ka-Ping Yeedd175342001-02-27 14:43:46 +000093def classname(object, modname):
94 """Get a class name and qualify it with a module name if necessary."""
95 name = object.__name__
96 if object.__module__ != modname:
97 name = object.__module__ + '.' + name
98 return name
99
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000100def isdata(object):
Georg Brandl94432422005-07-22 21:52:25 +0000101 """Check if an object is of a type that probably means it's data."""
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000102 return not (inspect.ismodule(object) or inspect.isclass(object) or
103 inspect.isroutine(object) or inspect.isframe(object) or
104 inspect.istraceback(object) or inspect.iscode(object))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000105
106def replace(text, *pairs):
107 """Do a series of global replacements on a string."""
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +0000108 while pairs:
Neal Norwitz9d72bb42007-04-17 08:48:32 +0000109 text = pairs[1].join(text.split(pairs[0]))
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +0000110 pairs = pairs[2:]
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000111 return text
112
113def cram(text, maxlen):
114 """Omit part of a string if needed to make it fit in a maximum length."""
115 if len(text) > maxlen:
Raymond Hettingerfca3bb62002-10-21 04:44:11 +0000116 pre = max(0, (maxlen-3)//2)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000117 post = max(0, maxlen-3-pre)
118 return text[:pre] + '...' + text[len(text)-post:]
119 return text
120
Brett Cannon84601f12004-06-19 01:22:48 +0000121_re_stripid = re.compile(r' at 0x[0-9a-f]{6,16}(>+)$', re.IGNORECASE)
Ka-Ping Yee1d384632001-03-01 00:24:32 +0000122def stripid(text):
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000123 """Remove the hexadecimal id from a Python object representation."""
Brett Cannonc6c1f472004-06-19 01:02:51 +0000124 # The behaviour of %p is implementation-dependent in terms of case.
125 if _re_stripid.search(repr(Exception)):
126 return _re_stripid.sub(r'\1', text)
Ka-Ping Yee1d384632001-03-01 00:24:32 +0000127 return text
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000128
Brett Cannonc6c1f472004-06-19 01:02:51 +0000129def _is_some_method(obj):
130 return inspect.ismethod(obj) or inspect.ismethoddescriptor(obj)
Tim Peters536d2262001-09-20 05:13:38 +0000131
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000132def allmethods(cl):
133 methods = {}
Tim Peters536d2262001-09-20 05:13:38 +0000134 for key, value in inspect.getmembers(cl, _is_some_method):
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000135 methods[key] = 1
136 for base in cl.__bases__:
137 methods.update(allmethods(base)) # all your base are belong to us
138 for key in methods.keys():
139 methods[key] = getattr(cl, key)
140 return methods
141
Tim Petersfa26f7c2001-09-24 08:05:11 +0000142def _split_list(s, predicate):
143 """Split sequence s via predicate, and return pair ([true], [false]).
144
145 The return value is a 2-tuple of lists,
146 ([x for x in s if predicate(x)],
147 [x for x in s if not predicate(x)])
148 """
149
Tim Peters28355492001-09-23 21:29:55 +0000150 yes = []
151 no = []
Tim Petersfa26f7c2001-09-24 08:05:11 +0000152 for x in s:
153 if predicate(x):
154 yes.append(x)
Tim Peters28355492001-09-23 21:29:55 +0000155 else:
Tim Petersfa26f7c2001-09-24 08:05:11 +0000156 no.append(x)
Tim Peters28355492001-09-23 21:29:55 +0000157 return yes, no
158
Skip Montanaroa5616d22004-06-11 04:46:12 +0000159def visiblename(name, all=None):
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000160 """Decide whether to show documentation on a variable."""
161 # Certain special names are redundant.
Raymond Hettingerdbecd932005-02-06 06:57:08 +0000162 if name in ('__builtins__', '__doc__', '__file__', '__path__',
163 '__module__', '__name__', '__slots__'): return 0
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000164 # Private names are hidden, but special names are displayed.
165 if name.startswith('__') and name.endswith('__'): return 1
Skip Montanaroa5616d22004-06-11 04:46:12 +0000166 if all is not None:
167 # only document that which the programmer exported in __all__
168 return name in all
169 else:
170 return not name.startswith('_')
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000171
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +0000172def classify_class_attrs(object):
173 """Wrap inspect.classify_class_attrs, with fixup for data descriptors."""
Guido van Rossum1bc535d2007-05-15 18:46:22 +0000174 results = []
175 for (name, kind, cls, value) in inspect.classify_class_attrs(object):
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +0000176 if inspect.isdatadescriptor(value):
177 kind = 'data descriptor'
Guido van Rossum1bc535d2007-05-15 18:46:22 +0000178 results.append((name, kind, cls, value))
179 return results
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +0000180
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()
Neal Norwitz9d72bb42007-04-17 08:48:32 +0000193 while line[:1] == '#' or not line.strip():
Thomas Wouters49fd7fa2006-04-21 10:40:58 +0000194 line = file.readline()
195 if not line: break
Neal Norwitz9d72bb42007-04-17 08:48:32 +0000196 line = line.strip()
Thomas Wouters49fd7fa2006-04-21 10:40:58 +0000197 if line[:4] == 'r"""': line = line[1:]
198 if line[:3] == '"""':
199 line = line[3:]
200 if line[-1:] == '\\': line = line[:-1]
Neal Norwitz9d72bb42007-04-17 08:48:32 +0000201 while not line.strip():
Thomas Wouters49fd7fa2006-04-21 10:40:58 +0000202 line = file.readline()
203 if not line: break
Neal Norwitz9d72bb42007-04-17 08:48:32 +0000204 result = line.split('"""')[0].strip()
Thomas Wouters49fd7fa2006-04-21 10:40:58 +0000205 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."""
Guido van Rossum1bc535d2007-05-15 18:46:22 +0000232 def __init__(self, filename, exc_info):
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000233 self.filename = filename
Guido van Rossum1bc535d2007-05-15 18:46:22 +0000234 self.exc, self.value, self.tb = exc_info
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000235
236 def __str__(self):
Guido van Rossuma01a8b62007-05-27 09:20:14 +0000237 exc = self.exc.__name__
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000238 return 'problem in %s - %s: %s' % (self.filename, exc, self.value)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000239
240def importfile(path):
241 """Import a Python source file or compiled file given its path."""
242 magic = imp.get_magic()
243 file = open(path, 'r')
244 if file.read(len(magic)) == magic:
245 kind = imp.PY_COMPILED
246 else:
247 kind = imp.PY_SOURCE
248 file.close()
249 filename = os.path.basename(path)
250 name, ext = os.path.splitext(filename)
251 file = open(path, 'r')
252 try:
253 module = imp.load_module(name, file, path, (ext, 'r', kind))
254 except:
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000255 raise ErrorDuringImport(path, sys.exc_info())
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000256 file.close()
257 return module
258
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000259def safeimport(path, forceload=0, cache={}):
260 """Import a module; handle errors; return None if the module isn't found.
261
262 If the module *is* found but an exception occurs, it's wrapped in an
263 ErrorDuringImport exception and reraised. Unlike __import__, if a
264 package path is specified, the module at the end of the path is returned,
265 not the package at the beginning. If the optional 'forceload' argument
266 is 1, we reload the module from disk (unless it's a dynamic extension)."""
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000267 try:
Ka-Ping Yee9a2dcf82005-11-05 05:04:41 +0000268 # If forceload is 1 and the module has been previously loaded from
269 # disk, we always have to reload the module. Checking the file's
270 # mtime isn't good enough (e.g. the module could contain a class
271 # that inherits from another module that has changed).
272 if forceload and path in sys.modules:
273 if path not in sys.builtin_module_names:
274 # Avoid simply calling reload() because it leaves names in
275 # the currently loaded module lying around if they're not
276 # defined in the new source file. Instead, remove the
277 # module from sys.modules and re-import. Also remove any
278 # submodules because they won't appear in the newly loaded
279 # module's namespace if they're already in sys.modules.
280 subs = [m for m in sys.modules if m.startswith(path + '.')]
281 for key in [path] + subs:
282 # Prevent garbage collection.
283 cache[key] = sys.modules[key]
284 del sys.modules[key]
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000285 module = __import__(path)
286 except:
287 # Did the error occur before or after the module was found?
288 (exc, value, tb) = info = sys.exc_info()
Raymond Hettinger54f02222002-06-01 14:18:47 +0000289 if path in sys.modules:
Fred Drakedb390c12005-10-28 14:39:47 +0000290 # An error occurred while executing the imported module.
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000291 raise ErrorDuringImport(sys.modules[path].__file__, info)
292 elif exc is SyntaxError:
293 # A SyntaxError occurred before we could execute the module.
294 raise ErrorDuringImport(value.filename, info)
295 elif exc is ImportError and \
Neal Norwitz9d72bb42007-04-17 08:48:32 +0000296 str(value).lower().split()[:2] == ['no', 'module']:
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000297 # The module was not found.
298 return None
299 else:
300 # Some other error occurred during the importing process.
301 raise ErrorDuringImport(path, sys.exc_info())
Neal Norwitz9d72bb42007-04-17 08:48:32 +0000302 for part in path.split('.')[1:]:
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000303 try: module = getattr(module, part)
304 except AttributeError: return None
305 return module
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000306
307# ---------------------------------------------------- formatter base class
308
309class Doc:
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000310 def document(self, object, name=None, *args):
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000311 """Generate documentation for an object."""
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000312 args = (object, name) + args
Brett Cannon28a4f0f2003-06-11 23:38:55 +0000313 # 'try' clause is to attempt to handle the possibility that inspect
314 # identifies something in a way that pydoc itself has issues handling;
315 # think 'super' and how it is a descriptor (which raises the exception
316 # by lacking a __name__ attribute) and an instance.
Thomas Wouters0e3f5912006-08-11 14:57:12 +0000317 if inspect.isgetsetdescriptor(object): return self.docdata(*args)
318 if inspect.ismemberdescriptor(object): return self.docdata(*args)
Brett Cannon28a4f0f2003-06-11 23:38:55 +0000319 try:
320 if inspect.ismodule(object): return self.docmodule(*args)
321 if inspect.isclass(object): return self.docclass(*args)
322 if inspect.isroutine(object): return self.docroutine(*args)
323 except AttributeError:
324 pass
Johannes Gijsbers8de645a2004-11-07 19:16:05 +0000325 if isinstance(object, property): return self.docproperty(*args)
Guido van Rossum68468eb2003-02-27 20:14:51 +0000326 return self.docother(*args)
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000327
328 def fail(self, object, name=None, *args):
329 """Raise an exception for unimplemented types."""
330 message = "don't know how to document object%s of type %s" % (
331 name and ' ' + repr(name), type(object).__name__)
332 raise TypeError, message
333
Thomas Wouters0e3f5912006-08-11 14:57:12 +0000334 docmodule = docclass = docroutine = docother = docproperty = docdata = fail
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000335
Skip Montanaro4997a692003-09-10 16:47:51 +0000336 def getdocloc(self, object):
337 """Return the location of module docs or None"""
338
339 try:
340 file = inspect.getabsfile(object)
341 except TypeError:
342 file = '(built-in)'
343
344 docloc = os.environ.get("PYTHONDOCS",
345 "http://www.python.org/doc/current/lib")
346 basedir = os.path.join(sys.exec_prefix, "lib",
347 "python"+sys.version[0:3])
348 if (isinstance(object, type(os)) and
349 (object.__name__ in ('errno', 'exceptions', 'gc', 'imp',
350 'marshal', 'posix', 'signal', 'sys',
351 'thread', 'zipimport') or
352 (file.startswith(basedir) and
353 not file.startswith(os.path.join(basedir, 'site-packages'))))):
Skip Montanarof2134842004-06-07 02:40:05 +0000354 htmlfile = "module-%s.html" % object.__name__
Skip Montanaro4997a692003-09-10 16:47:51 +0000355 if docloc.startswith("http://"):
Skip Montanarof2134842004-06-07 02:40:05 +0000356 docloc = "%s/%s" % (docloc.rstrip("/"), htmlfile)
Skip Montanaro4997a692003-09-10 16:47:51 +0000357 else:
Skip Montanarof2134842004-06-07 02:40:05 +0000358 docloc = os.path.join(docloc, htmlfile)
Skip Montanaro4997a692003-09-10 16:47:51 +0000359 else:
360 docloc = None
361 return docloc
362
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000363# -------------------------------------------- HTML documentation generator
364
365class HTMLRepr(Repr):
366 """Class for safely making an HTML representation of a Python object."""
367 def __init__(self):
368 Repr.__init__(self)
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000369 self.maxlist = self.maxtuple = 20
370 self.maxdict = 10
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000371 self.maxstring = self.maxother = 100
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000372
373 def escape(self, text):
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +0000374 return replace(text, '&', '&amp;', '<', '&lt;', '>', '&gt;')
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000375
376 def repr(self, object):
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000377 return Repr.repr(self, object)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000378
379 def repr1(self, x, level):
Skip Montanaro0fe8fce2003-06-27 15:45:41 +0000380 if hasattr(type(x), '__name__'):
Neal Norwitz9d72bb42007-04-17 08:48:32 +0000381 methodname = 'repr_' + '_'.join(type(x).__name__.split())
Skip Montanaro0fe8fce2003-06-27 15:45:41 +0000382 if hasattr(self, methodname):
383 return getattr(self, methodname)(x, level)
384 return self.escape(cram(stripid(repr(x)), self.maxother))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000385
386 def repr_string(self, x, level):
Ka-Ping Yeea2fe1032001-03-02 01:19:14 +0000387 test = cram(x, self.maxstring)
388 testrepr = repr(test)
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +0000389 if '\\' in test and '\\' not in replace(testrepr, r'\\', ''):
Ka-Ping Yeea2fe1032001-03-02 01:19:14 +0000390 # Backslashes are only literal in the string and are never
391 # needed to make any special characters, so show a raw string.
392 return 'r' + testrepr[0] + self.escape(test) + testrepr[0]
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +0000393 return re.sub(r'((\\[\\abfnrtv\'"]|\\[0-9]..|\\x..|\\u....)+)',
Ka-Ping Yeea2fe1032001-03-02 01:19:14 +0000394 r'<font color="#c040c0">\1</font>',
395 self.escape(testrepr))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000396
Skip Montanarodf708782002-03-07 22:58:02 +0000397 repr_str = repr_string
398
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000399 def repr_instance(self, x, level):
400 try:
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000401 return self.escape(cram(stripid(repr(x)), self.maxstring))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000402 except:
403 return self.escape('<%s instance>' % x.__class__.__name__)
404
405 repr_unicode = repr_string
406
407class HTMLDoc(Doc):
408 """Formatter class for HTML documentation."""
409
410 # ------------------------------------------- HTML formatting utilities
411
412 _repr_instance = HTMLRepr()
413 repr = _repr_instance.repr
414 escape = _repr_instance.escape
415
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000416 def page(self, title, contents):
417 """Format an HTML page."""
418 return '''
Tim Peters59ed4482001-10-31 04:20:26 +0000419<!doctype html PUBLIC "-//W3C//DTD HTML 4.0 Transitional//EN">
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000420<html><head><title>Python: %s</title>
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000421</head><body bgcolor="#f0f0f8">
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000422%s
423</body></html>''' % (title, contents)
424
425 def heading(self, title, fgcol, bgcol, extras=''):
426 """Format a page heading."""
427 return '''
Tim Peters59ed4482001-10-31 04:20:26 +0000428<table width="100%%" cellspacing=0 cellpadding=2 border=0 summary="heading">
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000429<tr bgcolor="%s">
Tim Peters2306d242001-09-25 03:18:32 +0000430<td valign=bottom>&nbsp;<br>
431<font color="%s" face="helvetica, arial">&nbsp;<br>%s</font></td
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000432><td align=right valign=bottom
Ka-Ping Yee987ec902001-03-23 13:35:45 +0000433><font color="%s" face="helvetica, arial">%s</font></td></tr></table>
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000434 ''' % (bgcol, fgcol, title, fgcol, extras or '&nbsp;')
435
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000436 def section(self, title, fgcol, bgcol, contents, width=6,
437 prelude='', marginalia=None, gap='&nbsp;'):
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000438 """Format a section with a heading."""
439 if marginalia is None:
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +0000440 marginalia = '<tt>' + '&nbsp;' * width + '</tt>'
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000441 result = '''<p>
Tim Peters59ed4482001-10-31 04:20:26 +0000442<table width="100%%" cellspacing=0 cellpadding=2 border=0 summary="section">
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000443<tr bgcolor="%s">
Tim Peters2306d242001-09-25 03:18:32 +0000444<td colspan=3 valign=bottom>&nbsp;<br>
445<font color="%s" face="helvetica, arial">%s</font></td></tr>
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000446 ''' % (bgcol, fgcol, title)
447 if prelude:
448 result = result + '''
Ka-Ping Yee987ec902001-03-23 13:35:45 +0000449<tr bgcolor="%s"><td rowspan=2>%s</td>
450<td colspan=2>%s</td></tr>
451<tr><td>%s</td>''' % (bgcol, marginalia, prelude, gap)
452 else:
453 result = result + '''
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000454<tr><td bgcolor="%s">%s</td><td>%s</td>''' % (bgcol, marginalia, gap)
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000455
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +0000456 return result + '\n<td width="100%%">%s</td></tr></table>' % contents
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000457
458 def bigsection(self, title, *args):
459 """Format a section with a big heading."""
460 title = '<big><strong>%s</strong></big>' % title
Guido van Rossum68468eb2003-02-27 20:14:51 +0000461 return self.section(title, *args)
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000462
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000463 def preformat(self, text):
464 """Format literal preformatted text."""
Neal Norwitz9d72bb42007-04-17 08:48:32 +0000465 text = self.escape(text.expandtabs())
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +0000466 return replace(text, '\n\n', '\n \n', '\n\n', '\n \n',
467 ' ', '&nbsp;', '\n', '<br>\n')
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000468
469 def multicolumn(self, list, format, cols=4):
470 """Format a list of items into a multi-column list."""
471 result = ''
472 rows = (len(list)+cols-1)/cols
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000473 for col in range(cols):
474 result = result + '<td width="%d%%" valign=top>' % (100/cols)
475 for i in range(rows*col, rows*col+rows):
476 if i < len(list):
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000477 result = result + format(list[i]) + '<br>\n'
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000478 result = result + '</td>'
Tim Peters59ed4482001-10-31 04:20:26 +0000479 return '<table width="100%%" summary="list"><tr>%s</tr></table>' % result
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000480
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000481 def grey(self, text): return '<font color="#909090">%s</font>' % text
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000482
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000483 def namelink(self, name, *dicts):
484 """Make a link for an identifier, given name-to-URL mappings."""
485 for dict in dicts:
Raymond Hettinger54f02222002-06-01 14:18:47 +0000486 if name in dict:
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000487 return '<a href="%s">%s</a>' % (dict[name], name)
488 return name
489
Ka-Ping Yeeb7a48302001-04-12 20:39:14 +0000490 def classlink(self, object, modname):
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000491 """Make a link for a class."""
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000492 name, module = object.__name__, sys.modules.get(object.__module__)
493 if hasattr(module, name) and getattr(module, name) is object:
Ka-Ping Yeeb7a48302001-04-12 20:39:14 +0000494 return '<a href="%s.html#%s">%s</a>' % (
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000495 module.__name__, name, classname(object, modname))
496 return classname(object, modname)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000497
498 def modulelink(self, object):
499 """Make a link for a module."""
500 return '<a href="%s.html">%s</a>' % (object.__name__, object.__name__)
501
Guido van Rossum1bc535d2007-05-15 18:46:22 +0000502 def modpkglink(self, modpkginfo):
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000503 """Make a link for a module or package to display in an index."""
Guido van Rossum1bc535d2007-05-15 18:46:22 +0000504 name, path, ispackage, shadowed = modpkginfo
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000505 if shadowed:
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000506 return self.grey(name)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000507 if path:
508 url = '%s.%s.html' % (path, name)
509 else:
510 url = '%s.html' % name
511 if ispackage:
512 text = '<strong>%s</strong>&nbsp;(package)' % name
513 else:
514 text = name
515 return '<a href="%s">%s</a>' % (url, text)
516
517 def markup(self, text, escape=None, funcs={}, classes={}, methods={}):
518 """Mark up some plain text, given a context of symbols to look for.
519 Each context dictionary maps object names to anchor names."""
520 escape = escape or self.escape
521 results = []
522 here = 0
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000523 pattern = re.compile(r'\b((http|ftp)://\S+[\w/]|'
524 r'RFC[- ]?(\d+)|'
Ka-Ping Yeef78a81b2001-03-27 08:13:42 +0000525 r'PEP[- ]?(\d+)|'
Neil Schemenauerd69711c2002-03-24 23:02:07 +0000526 r'(self\.)?(\w+))')
Guido van Rossum8ca162f2002-04-07 06:36:23 +0000527 while True:
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000528 match = pattern.search(text, here)
529 if not match: break
530 start, end = match.span()
531 results.append(escape(text[here:start]))
532
Ka-Ping Yeef78a81b2001-03-27 08:13:42 +0000533 all, scheme, rfc, pep, selfdot, name = match.groups()
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000534 if scheme:
Neil Schemenauercddc1a02002-03-24 23:11:21 +0000535 url = escape(all).replace('"', '&quot;')
536 results.append('<a href="%s">%s</a>' % (url, url))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000537 elif rfc:
Ka-Ping Yeef78a81b2001-03-27 08:13:42 +0000538 url = 'http://www.rfc-editor.org/rfc/rfc%d.txt' % int(rfc)
539 results.append('<a href="%s">%s</a>' % (url, escape(all)))
540 elif pep:
541 url = 'http://www.python.org/peps/pep-%04d.html' % int(pep)
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000542 results.append('<a href="%s">%s</a>' % (url, escape(all)))
543 elif text[end:end+1] == '(':
544 results.append(self.namelink(name, methods, funcs, classes))
545 elif selfdot:
546 results.append('self.<strong>%s</strong>' % name)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000547 else:
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000548 results.append(self.namelink(name, classes))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000549 here = end
550 results.append(escape(text[here:]))
Neal Norwitz9d72bb42007-04-17 08:48:32 +0000551 return ''.join(results)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000552
553 # ---------------------------------------------- type-specific routines
554
Ka-Ping Yeeb7a48302001-04-12 20:39:14 +0000555 def formattree(self, tree, modname, parent=None):
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000556 """Produce HTML for a class tree as given by inspect.getclasstree()."""
557 result = ''
558 for entry in tree:
559 if type(entry) is type(()):
560 c, bases = entry
Tim Peters2306d242001-09-25 03:18:32 +0000561 result = result + '<dt><font face="helvetica, arial">'
Ka-Ping Yeeb7a48302001-04-12 20:39:14 +0000562 result = result + self.classlink(c, modname)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000563 if bases and bases != (parent,):
564 parents = []
565 for base in bases:
Ka-Ping Yeeb7a48302001-04-12 20:39:14 +0000566 parents.append(self.classlink(base, modname))
Neal Norwitz9d72bb42007-04-17 08:48:32 +0000567 result = result + '(' + ', '.join(parents) + ')'
Tim Peters2306d242001-09-25 03:18:32 +0000568 result = result + '\n</font></dt>'
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000569 elif type(entry) is type([]):
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000570 result = result + '<dd>\n%s</dd>\n' % self.formattree(
Ka-Ping Yeeb7a48302001-04-12 20:39:14 +0000571 entry, modname, c)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000572 return '<dl>\n%s</dl>\n' % result
573
Tim Peters8dd7ade2001-10-18 19:56:17 +0000574 def docmodule(self, object, name=None, mod=None, *ignored):
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000575 """Produce HTML documentation for a module object."""
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000576 name = object.__name__ # ignore the passed-in name
Skip Montanaroa5616d22004-06-11 04:46:12 +0000577 try:
578 all = object.__all__
579 except AttributeError:
580 all = None
Neal Norwitz9d72bb42007-04-17 08:48:32 +0000581 parts = name.split('.')
Ka-Ping Yee66efbc72001-03-01 13:55:20 +0000582 links = []
583 for i in range(len(parts)-1):
584 links.append(
585 '<a href="%s.html"><font color="#ffffff">%s</font></a>' %
Neal Norwitz9d72bb42007-04-17 08:48:32 +0000586 ('.'.join(parts[:i+1]), parts[i]))
587 linkedname = '.'.join(links + parts[-1:])
Ka-Ping Yee66efbc72001-03-01 13:55:20 +0000588 head = '<big><big><strong>%s</strong></big></big>' % linkedname
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000589 try:
Ka-Ping Yee239432a2001-03-02 02:45:08 +0000590 path = inspect.getabsfile(object)
Ka-Ping Yee6191a232001-04-13 15:00:27 +0000591 url = path
592 if sys.platform == 'win32':
593 import nturl2path
594 url = nturl2path.pathname2url(path)
595 filelink = '<a href="file:%s">%s</a>' % (url, path)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000596 except TypeError:
597 filelink = '(built-in)'
Ka-Ping Yee6f3f9a42001-02-27 22:42:36 +0000598 info = []
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000599 if hasattr(object, '__version__'):
Ka-Ping Yee6f3f9a42001-02-27 22:42:36 +0000600 version = str(object.__version__)
Ka-Ping Yee40c49912001-02-27 22:46:01 +0000601 if version[:11] == '$' + 'Revision: ' and version[-1:] == '$':
Neal Norwitz9d72bb42007-04-17 08:48:32 +0000602 version = version[11:-1].strip()
Ka-Ping Yee1d384632001-03-01 00:24:32 +0000603 info.append('version %s' % self.escape(version))
Ka-Ping Yee6f3f9a42001-02-27 22:42:36 +0000604 if hasattr(object, '__date__'):
605 info.append(self.escape(str(object.__date__)))
606 if info:
Neal Norwitz9d72bb42007-04-17 08:48:32 +0000607 head = head + ' (%s)' % ', '.join(info)
Skip Montanaro4997a692003-09-10 16:47:51 +0000608 docloc = self.getdocloc(object)
609 if docloc is not None:
610 docloc = '<br><a href="%(docloc)s">Module Docs</a>' % locals()
611 else:
612 docloc = ''
Ka-Ping Yee66efbc72001-03-01 13:55:20 +0000613 result = self.heading(
Skip Montanaro4997a692003-09-10 16:47:51 +0000614 head, '#ffffff', '#7799ee',
615 '<a href=".">index</a><br>' + filelink + docloc)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000616
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000617 modules = inspect.getmembers(object, inspect.ismodule)
618
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000619 classes, cdict = [], {}
620 for key, value in inspect.getmembers(object, inspect.isclass):
Johannes Gijsbers4c11f602004-08-30 14:13:04 +0000621 # if __all__ exists, believe it. Otherwise use old heuristic.
622 if (all is not None or
623 (inspect.getmodule(value) or object) is object):
Skip Montanaroa5616d22004-06-11 04:46:12 +0000624 if visiblename(key, all):
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000625 classes.append((key, value))
626 cdict[key] = cdict[value] = '#' + key
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000627 for key, value in classes:
628 for base in value.__bases__:
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000629 key, modname = base.__name__, base.__module__
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000630 module = sys.modules.get(modname)
631 if modname != name and module and hasattr(module, key):
632 if getattr(module, key) is base:
Raymond Hettinger54f02222002-06-01 14:18:47 +0000633 if not key in cdict:
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000634 cdict[key] = cdict[base] = modname + '.html#' + key
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000635 funcs, fdict = [], {}
636 for key, value in inspect.getmembers(object, inspect.isroutine):
Johannes Gijsbers4c11f602004-08-30 14:13:04 +0000637 # if __all__ exists, believe it. Otherwise use old heuristic.
638 if (all is not None or
639 inspect.isbuiltin(value) or inspect.getmodule(value) is object):
Skip Montanaroa5616d22004-06-11 04:46:12 +0000640 if visiblename(key, all):
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000641 funcs.append((key, value))
642 fdict[key] = '#-' + key
643 if inspect.isfunction(value): fdict[value] = fdict[key]
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000644 data = []
645 for key, value in inspect.getmembers(object, isdata):
Skip Montanaroa5616d22004-06-11 04:46:12 +0000646 if visiblename(key, all):
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000647 data.append((key, value))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000648
649 doc = self.markup(getdoc(object), self.preformat, fdict, cdict)
650 doc = doc and '<tt>%s</tt>' % doc
Tim Peters2306d242001-09-25 03:18:32 +0000651 result = result + '<p>%s</p>\n' % doc
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000652
653 if hasattr(object, '__path__'):
654 modpkgs = []
Thomas Wouters49fd7fa2006-04-21 10:40:58 +0000655 for importer, modname, ispkg in pkgutil.iter_modules(object.__path__):
656 modpkgs.append((modname, name, ispkg, 0))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000657 modpkgs.sort()
658 contents = self.multicolumn(modpkgs, self.modpkglink)
659 result = result + self.bigsection(
660 'Package Contents', '#ffffff', '#aa55cc', contents)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000661 elif modules:
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000662 contents = self.multicolumn(
Guido van Rossum1bc535d2007-05-15 18:46:22 +0000663 modules, lambda t: self.modulelink(t[1]))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000664 result = result + self.bigsection(
665 'Modules', '#fffff', '#aa55cc', contents)
666
667 if classes:
Guido van Rossum1bc535d2007-05-15 18:46:22 +0000668 classlist = [value for (key, value) in classes]
Ka-Ping Yeeb7a48302001-04-12 20:39:14 +0000669 contents = [
670 self.formattree(inspect.getclasstree(classlist, 1), name)]
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000671 for key, value in classes:
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +0000672 contents.append(self.document(value, key, name, fdict, cdict))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000673 result = result + self.bigsection(
Neal Norwitz9d72bb42007-04-17 08:48:32 +0000674 'Classes', '#ffffff', '#ee77aa', ' '.join(contents))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000675 if funcs:
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000676 contents = []
677 for key, value in funcs:
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +0000678 contents.append(self.document(value, key, name, fdict, cdict))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000679 result = result + self.bigsection(
Neal Norwitz9d72bb42007-04-17 08:48:32 +0000680 'Functions', '#ffffff', '#eeaa77', ' '.join(contents))
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000681 if data:
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000682 contents = []
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000683 for key, value in data:
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000684 contents.append(self.document(value, key))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000685 result = result + self.bigsection(
Neal Norwitz9d72bb42007-04-17 08:48:32 +0000686 'Data', '#ffffff', '#55aa55', '<br>\n'.join(contents))
Ka-Ping Yee6f3f9a42001-02-27 22:42:36 +0000687 if hasattr(object, '__author__'):
688 contents = self.markup(str(object.__author__), self.preformat)
689 result = result + self.bigsection(
690 'Author', '#ffffff', '#7799ee', contents)
Ka-Ping Yee6f3f9a42001-02-27 22:42:36 +0000691 if hasattr(object, '__credits__'):
692 contents = self.markup(str(object.__credits__), self.preformat)
693 result = result + self.bigsection(
694 'Credits', '#ffffff', '#7799ee', contents)
695
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000696 return result
697
Tim Peters8dd7ade2001-10-18 19:56:17 +0000698 def docclass(self, object, name=None, mod=None, funcs={}, classes={},
699 *ignored):
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000700 """Produce HTML documentation for a class object."""
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000701 realname = object.__name__
702 name = name or realname
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000703 bases = object.__bases__
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000704
Tim Petersb47879b2001-09-24 04:47:19 +0000705 contents = []
706 push = contents.append
707
Tim Petersfa26f7c2001-09-24 08:05:11 +0000708 # Cute little class to pump out a horizontal rule between sections.
709 class HorizontalRule:
710 def __init__(self):
711 self.needone = 0
712 def maybe(self):
713 if self.needone:
714 push('<hr>\n')
715 self.needone = 1
716 hr = HorizontalRule()
717
Tim Petersc86f6ca2001-09-26 21:31:51 +0000718 # List the mro, if non-trivial.
Raymond Hettinger756b3f32004-01-29 06:37:52 +0000719 mro = deque(inspect.getmro(object))
Tim Petersc86f6ca2001-09-26 21:31:51 +0000720 if len(mro) > 2:
721 hr.maybe()
722 push('<dl><dt>Method resolution order:</dt>\n')
723 for base in mro:
724 push('<dd>%s</dd>\n' % self.classlink(base,
725 object.__module__))
726 push('</dl>\n')
727
Tim Petersb47879b2001-09-24 04:47:19 +0000728 def spill(msg, attrs, predicate):
Tim Petersfa26f7c2001-09-24 08:05:11 +0000729 ok, attrs = _split_list(attrs, predicate)
Tim Petersb47879b2001-09-24 04:47:19 +0000730 if ok:
Tim Petersfa26f7c2001-09-24 08:05:11 +0000731 hr.maybe()
Tim Petersb47879b2001-09-24 04:47:19 +0000732 push(msg)
733 for name, kind, homecls, value in ok:
734 push(self.document(getattr(object, name), name, mod,
735 funcs, classes, mdict, object))
736 push('\n')
737 return attrs
738
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +0000739 def spilldescriptors(msg, attrs, predicate):
Tim Petersfa26f7c2001-09-24 08:05:11 +0000740 ok, attrs = _split_list(attrs, predicate)
Tim Petersb47879b2001-09-24 04:47:19 +0000741 if ok:
Tim Petersfa26f7c2001-09-24 08:05:11 +0000742 hr.maybe()
Tim Petersb47879b2001-09-24 04:47:19 +0000743 push(msg)
744 for name, kind, homecls, value in ok:
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +0000745 push(self._docdescriptor(name, value, mod))
Tim Petersb47879b2001-09-24 04:47:19 +0000746 return attrs
747
Tim Petersfa26f7c2001-09-24 08:05:11 +0000748 def spilldata(msg, attrs, predicate):
749 ok, attrs = _split_list(attrs, predicate)
Tim Petersb47879b2001-09-24 04:47:19 +0000750 if ok:
Tim Petersfa26f7c2001-09-24 08:05:11 +0000751 hr.maybe()
Tim Petersb47879b2001-09-24 04:47:19 +0000752 push(msg)
753 for name, kind, homecls, value in ok:
754 base = self.docother(getattr(object, name), name, mod)
Guido van Rossumd59da4b2007-05-22 18:11:13 +0000755 if hasattr(value, '__call__') or inspect.isdatadescriptor(value):
Guido van Rossum5e355b22002-05-21 20:56:15 +0000756 doc = getattr(value, "__doc__", None)
757 else:
758 doc = None
Tim Petersb47879b2001-09-24 04:47:19 +0000759 if doc is None:
760 push('<dl><dt>%s</dl>\n' % base)
761 else:
762 doc = self.markup(getdoc(value), self.preformat,
763 funcs, classes, mdict)
Tim Peters2306d242001-09-25 03:18:32 +0000764 doc = '<dd><tt>%s</tt>' % doc
Tim Petersb47879b2001-09-24 04:47:19 +0000765 push('<dl><dt>%s%s</dl>\n' % (base, doc))
766 push('\n')
767 return attrs
768
Guido van Rossum1bc535d2007-05-15 18:46:22 +0000769 attrs = [(name, kind, cls, value)
770 for name, kind, cls, value in classify_class_attrs(object)
771 if visiblename(name)]
772
Tim Petersb47879b2001-09-24 04:47:19 +0000773 mdict = {}
774 for key, kind, homecls, value in attrs:
775 mdict[key] = anchor = '#' + name + '-' + key
776 value = getattr(object, key)
777 try:
778 # The value may not be hashable (e.g., a data attr with
779 # a dict or list value).
780 mdict[value] = anchor
781 except TypeError:
782 pass
783
Tim Petersfa26f7c2001-09-24 08:05:11 +0000784 while attrs:
Tim Peters351e3622001-09-27 03:29:51 +0000785 if mro:
Raymond Hettinger756b3f32004-01-29 06:37:52 +0000786 thisclass = mro.popleft()
Tim Peters351e3622001-09-27 03:29:51 +0000787 else:
788 thisclass = attrs[0][2]
Tim Petersfa26f7c2001-09-24 08:05:11 +0000789 attrs, inherited = _split_list(attrs, lambda t: t[2] is thisclass)
790
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000791 if thisclass is __builtin__.object:
792 attrs = inherited
793 continue
794 elif thisclass is object:
795 tag = 'defined here'
Tim Petersb47879b2001-09-24 04:47:19 +0000796 else:
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000797 tag = 'inherited from %s' % self.classlink(thisclass,
798 object.__module__)
Tim Petersb47879b2001-09-24 04:47:19 +0000799 tag += ':<br>\n'
800
801 # Sort attrs by name.
Thomas Wouters49fd7fa2006-04-21 10:40:58 +0000802 try:
803 attrs.sort(key=lambda t: t[0])
804 except TypeError:
805 attrs.sort(lambda t1, t2: cmp(t1[0], t2[0])) # 2.3 compat
Tim Petersb47879b2001-09-24 04:47:19 +0000806
807 # Pump out the attrs, segregated by kind.
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000808 attrs = spill('Methods %s' % tag, attrs,
Tim Petersb47879b2001-09-24 04:47:19 +0000809 lambda t: t[1] == 'method')
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000810 attrs = spill('Class methods %s' % tag, attrs,
Tim Petersb47879b2001-09-24 04:47:19 +0000811 lambda t: t[1] == 'class method')
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000812 attrs = spill('Static methods %s' % tag, attrs,
Tim Petersb47879b2001-09-24 04:47:19 +0000813 lambda t: t[1] == 'static method')
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +0000814 attrs = spilldescriptors('Data descriptors %s' % tag, attrs,
815 lambda t: t[1] == 'data descriptor')
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000816 attrs = spilldata('Data and other attributes %s' % tag, attrs,
Tim Petersfa26f7c2001-09-24 08:05:11 +0000817 lambda t: t[1] == 'data')
Tim Petersb47879b2001-09-24 04:47:19 +0000818 assert attrs == []
Tim Peters351e3622001-09-27 03:29:51 +0000819 attrs = inherited
Tim Petersb47879b2001-09-24 04:47:19 +0000820
821 contents = ''.join(contents)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000822
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000823 if name == realname:
824 title = '<a name="%s">class <strong>%s</strong></a>' % (
825 name, realname)
826 else:
827 title = '<strong>%s</strong> = <a name="%s">class %s</a>' % (
828 name, name, realname)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000829 if bases:
830 parents = []
831 for base in bases:
Ka-Ping Yeeb7a48302001-04-12 20:39:14 +0000832 parents.append(self.classlink(base, object.__module__))
Neal Norwitz9d72bb42007-04-17 08:48:32 +0000833 title = title + '(%s)' % ', '.join(parents)
Tim Peters2306d242001-09-25 03:18:32 +0000834 doc = self.markup(getdoc(object), self.preformat, funcs, classes, mdict)
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000835 doc = doc and '<tt>%s<br>&nbsp;</tt>' % doc
Tim Petersc86f6ca2001-09-26 21:31:51 +0000836
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000837 return self.section(title, '#000000', '#ffc8d8', contents, 3, doc)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000838
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000839 def formatvalue(self, object):
840 """Format an argument default value as text."""
Tim Peters2306d242001-09-25 03:18:32 +0000841 return self.grey('=' + self.repr(object))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000842
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +0000843 def docroutine(self, object, name=None, mod=None,
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000844 funcs={}, classes={}, methods={}, cl=None):
Ka-Ping Yee66efbc72001-03-01 13:55:20 +0000845 """Produce HTML documentation for a function or method object."""
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000846 realname = object.__name__
847 name = name or realname
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000848 anchor = (cl and cl.__name__ or '') + '-' + name
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000849 note = ''
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000850 skipdocs = 0
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000851 if inspect.ismethod(object):
Ka-Ping Yee6dcfa382001-04-12 20:27:31 +0000852 imclass = object.im_class
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000853 if cl:
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +0000854 if imclass is not cl:
Ka-Ping Yeeb7a48302001-04-12 20:39:14 +0000855 note = ' from ' + self.classlink(imclass, mod)
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000856 else:
Guido van Rossumd8faa362007-04-27 19:54:29 +0000857 if object.im_self is not None:
Ka-Ping Yeeb7a48302001-04-12 20:39:14 +0000858 note = ' method of %s instance' % self.classlink(
859 object.im_self.__class__, mod)
860 else:
861 note = ' unbound %s method' % self.classlink(imclass,mod)
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000862 object = object.im_func
863
864 if name == realname:
865 title = '<a name="%s"><strong>%s</strong></a>' % (anchor, realname)
866 else:
Raymond Hettinger54f02222002-06-01 14:18:47 +0000867 if (cl and realname in cl.__dict__ and
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000868 cl.__dict__[realname] is object):
Ka-Ping Yeee280c062001-03-23 14:05:53 +0000869 reallink = '<a href="#%s">%s</a>' % (
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000870 cl.__name__ + '-' + realname, realname)
871 skipdocs = 1
872 else:
873 reallink = realname
874 title = '<a name="%s"><strong>%s</strong></a> = %s' % (
875 anchor, name, reallink)
Tim Peters4bcfa312001-09-20 06:08:24 +0000876 if inspect.isfunction(object):
Guido van Rossum2e65f892007-02-28 22:03:49 +0000877 args, varargs, kwonlyargs, kwdefaults, varkw, defaults, ann = \
878 inspect.getfullargspec(object)
Ka-Ping Yee66efbc72001-03-01 13:55:20 +0000879 argspec = inspect.formatargspec(
Guido van Rossum2e65f892007-02-28 22:03:49 +0000880 args, varargs, kwonlyargs, kwdefaults, varkw, defaults, ann,
881 formatvalue=self.formatvalue,
882 formatannotation=inspect.formatannotationrelativeto(object))
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000883 if realname == '<lambda>':
Tim Peters59ed4482001-10-31 04:20:26 +0000884 title = '<strong>%s</strong> <em>lambda</em> ' % name
Guido van Rossum2e65f892007-02-28 22:03:49 +0000885 # XXX lambda's won't usually have func_annotations['return']
886 # since the syntax doesn't support but it is possible.
887 # So removing parentheses isn't truly safe.
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000888 argspec = argspec[1:-1] # remove parentheses
Tim Peters4bcfa312001-09-20 06:08:24 +0000889 else:
890 argspec = '(...)'
Ka-Ping Yee66efbc72001-03-01 13:55:20 +0000891
Tim Peters2306d242001-09-25 03:18:32 +0000892 decl = title + argspec + (note and self.grey(
893 '<font face="helvetica, arial">%s</font>' % note))
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000894
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000895 if skipdocs:
Tim Peters2306d242001-09-25 03:18:32 +0000896 return '<dl><dt>%s</dt></dl>\n' % decl
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000897 else:
898 doc = self.markup(
899 getdoc(object), self.preformat, funcs, classes, methods)
Tim Peters2306d242001-09-25 03:18:32 +0000900 doc = doc and '<dd><tt>%s</tt></dd>' % doc
901 return '<dl><dt>%s</dt>%s</dl>\n' % (decl, doc)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000902
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +0000903 def _docdescriptor(self, name, value, mod):
Johannes Gijsbers8de645a2004-11-07 19:16:05 +0000904 results = []
905 push = results.append
906
907 if name:
908 push('<dl><dt><strong>%s</strong></dt>\n' % name)
909 if value.__doc__ is not None:
Ka-Ping Yeebba6acc2005-02-19 22:58:26 +0000910 doc = self.markup(getdoc(value), self.preformat)
Johannes Gijsbers8de645a2004-11-07 19:16:05 +0000911 push('<dd><tt>%s</tt></dd>\n' % doc)
Johannes Gijsbers8de645a2004-11-07 19:16:05 +0000912 push('</dl>\n')
913
914 return ''.join(results)
915
916 def docproperty(self, object, name=None, mod=None, cl=None):
917 """Produce html documentation for a property."""
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +0000918 return self._docdescriptor(name, object, mod)
Johannes Gijsbers8de645a2004-11-07 19:16:05 +0000919
Tim Peters8dd7ade2001-10-18 19:56:17 +0000920 def docother(self, object, name=None, mod=None, *ignored):
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000921 """Produce HTML documentation for a data object."""
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +0000922 lhs = name and '<strong>%s</strong> = ' % name or ''
923 return lhs + self.repr(object)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000924
Thomas Wouters0e3f5912006-08-11 14:57:12 +0000925 def docdata(self, object, name=None, mod=None, cl=None):
926 """Produce html documentation for a data descriptor."""
927 return self._docdescriptor(name, object, mod)
928
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000929 def index(self, dir, shadowed=None):
930 """Generate an HTML index for a directory of modules."""
931 modpkgs = []
932 if shadowed is None: shadowed = {}
Thomas Wouters49fd7fa2006-04-21 10:40:58 +0000933 for importer, name, ispkg in pkgutil.iter_modules([dir]):
934 modpkgs.append((name, '', ispkg, name in shadowed))
935 shadowed[name] = 1
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000936
937 modpkgs.sort()
938 contents = self.multicolumn(modpkgs, self.modpkglink)
939 return self.bigsection(dir, '#ffffff', '#ee77aa', contents)
940
941# -------------------------------------------- text documentation generator
942
943class TextRepr(Repr):
944 """Class for safely making a text representation of a Python object."""
945 def __init__(self):
946 Repr.__init__(self)
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000947 self.maxlist = self.maxtuple = 20
948 self.maxdict = 10
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000949 self.maxstring = self.maxother = 100
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000950
951 def repr1(self, x, level):
Skip Montanaro0fe8fce2003-06-27 15:45:41 +0000952 if hasattr(type(x), '__name__'):
Neal Norwitz9d72bb42007-04-17 08:48:32 +0000953 methodname = 'repr_' + '_'.join(type(x).__name__.split())
Skip Montanaro0fe8fce2003-06-27 15:45:41 +0000954 if hasattr(self, methodname):
955 return getattr(self, methodname)(x, level)
956 return cram(stripid(repr(x)), self.maxother)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000957
Ka-Ping Yeea2fe1032001-03-02 01:19:14 +0000958 def repr_string(self, x, level):
959 test = cram(x, self.maxstring)
960 testrepr = repr(test)
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +0000961 if '\\' in test and '\\' not in replace(testrepr, r'\\', ''):
Ka-Ping Yeea2fe1032001-03-02 01:19:14 +0000962 # Backslashes are only literal in the string and are never
963 # needed to make any special characters, so show a raw string.
964 return 'r' + testrepr[0] + test + testrepr[0]
965 return testrepr
966
Skip Montanarodf708782002-03-07 22:58:02 +0000967 repr_str = repr_string
968
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000969 def repr_instance(self, x, level):
970 try:
Ka-Ping Yee1d384632001-03-01 00:24:32 +0000971 return cram(stripid(repr(x)), self.maxstring)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000972 except:
973 return '<%s instance>' % x.__class__.__name__
974
975class TextDoc(Doc):
976 """Formatter class for text documentation."""
977
978 # ------------------------------------------- text formatting utilities
979
980 _repr_instance = TextRepr()
981 repr = _repr_instance.repr
982
983 def bold(self, text):
984 """Format a string in bold by overstriking."""
Neal Norwitz9d72bb42007-04-17 08:48:32 +0000985 return ''.join(map(lambda ch: ch + '\b' + ch, text))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000986
987 def indent(self, text, prefix=' '):
988 """Indent text by prepending a given prefix to each line."""
989 if not text: return ''
Neal Norwitz9d72bb42007-04-17 08:48:32 +0000990 lines = text.split('\n')
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000991 lines = map(lambda line, prefix=prefix: prefix + line, lines)
Neal Norwitz9d72bb42007-04-17 08:48:32 +0000992 if lines: lines[-1] = lines[-1].rstrip()
993 return '\n'.join(lines)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000994
995 def section(self, title, contents):
996 """Format a section with a given heading."""
Neal Norwitz9d72bb42007-04-17 08:48:32 +0000997 clean_contents = self.indent(contents).rstrip()
998 return self.bold(title) + '\n' + clean_contents + '\n\n'
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000999
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)
Neal Norwitz9d72bb42007-04-17 08:48:32 +00001011 result = result + '(%s)' % ', '.join(parents)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001012 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(
Neal Norwitz9d72bb42007-04-17 08:48:32 +00001071 'PACKAGE CONTENTS', '\n'.join(modpkgs))
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001072
1073 if classes:
Guido van Rossum1bc535d2007-05-15 18:46:22 +00001074 classlist = [value for key, value in classes]
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001075 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))
Neal Norwitz9d72bb42007-04-17 08:48:32 +00001079 result = result + self.section('CLASSES', '\n'.join(contents))
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))
Neal Norwitz9d72bb42007-04-17 08:48:32 +00001085 result = result + self.section('FUNCTIONS', '\n'.join(contents))
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))
Neal Norwitz9d72bb42007-04-17 08:48:32 +00001091 result = result + self.section('DATA', '\n'.join(contents))
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:] == '$':
Neal Norwitz9d72bb42007-04-17 08:48:32 +00001096 version = version[11:-1].strip()
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)
Neal Norwitz9d72bb42007-04-17 08:48:32 +00001121 title = title + '(%s)' % ', '.join(parents)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001122
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:
Guido van Rossumd59da4b2007-05-22 18:11:13 +00001170 if hasattr(value, '__call__') 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
Guido van Rossum1bc535d2007-05-15 18:46:22 +00001178 attrs = [(name, kind, cls, value)
1179 for name, kind, cls, value in classify_class_attrs(object)
1180 if visiblename(name)]
1181
Tim Petersfa26f7c2001-09-24 08:05:11 +00001182 while attrs:
Tim Peters351e3622001-09-27 03:29:51 +00001183 if mro:
Raymond Hettinger756b3f32004-01-29 06:37:52 +00001184 thisclass = mro.popleft()
Tim Peters351e3622001-09-27 03:29:51 +00001185 else:
1186 thisclass = attrs[0][2]
Tim Petersfa26f7c2001-09-24 08:05:11 +00001187 attrs, inherited = _split_list(attrs, lambda t: t[2] is thisclass)
1188
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +00001189 if thisclass is __builtin__.object:
1190 attrs = inherited
1191 continue
1192 elif thisclass is object:
Tim Peters28355492001-09-23 21:29:55 +00001193 tag = "defined here"
1194 else:
Tim Petersfa26f7c2001-09-24 08:05:11 +00001195 tag = "inherited from %s" % classname(thisclass,
1196 object.__module__)
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +00001197 filter(lambda t: not t[0].startswith('_'), attrs)
Tim Peters28355492001-09-23 21:29:55 +00001198
1199 # Sort attrs by name.
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +00001200 attrs.sort()
Tim Peters28355492001-09-23 21:29:55 +00001201
1202 # Pump out the attrs, segregated by kind.
Tim Petersf4aad8e2001-09-24 22:40:47 +00001203 attrs = spill("Methods %s:\n" % tag, attrs,
Tim Peters28355492001-09-23 21:29:55 +00001204 lambda t: t[1] == 'method')
Tim Petersf4aad8e2001-09-24 22:40:47 +00001205 attrs = spill("Class methods %s:\n" % tag, attrs,
Tim Peters28355492001-09-23 21:29:55 +00001206 lambda t: t[1] == 'class method')
Tim Petersf4aad8e2001-09-24 22:40:47 +00001207 attrs = spill("Static methods %s:\n" % tag, attrs,
Tim Peters28355492001-09-23 21:29:55 +00001208 lambda t: t[1] == 'static method')
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +00001209 attrs = spilldescriptors("Data descriptors %s:\n" % tag, attrs,
1210 lambda t: t[1] == 'data descriptor')
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +00001211 attrs = spilldata("Data and other attributes %s:\n" % tag, attrs,
1212 lambda t: t[1] == 'data')
Tim Peters28355492001-09-23 21:29:55 +00001213 assert attrs == []
Tim Peters351e3622001-09-27 03:29:51 +00001214 attrs = inherited
Tim Peters28355492001-09-23 21:29:55 +00001215
1216 contents = '\n'.join(contents)
1217 if not contents:
1218 return title + '\n'
Neal Norwitz9d72bb42007-04-17 08:48:32 +00001219 return title + '\n' + self.indent(contents.rstrip(), ' | ') + '\n'
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001220
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001221 def formatvalue(self, object):
1222 """Format an argument default value as text."""
1223 return '=' + self.repr(object)
1224
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001225 def docroutine(self, object, name=None, mod=None, cl=None):
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001226 """Produce text documentation for a function or method object."""
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001227 realname = object.__name__
1228 name = name or realname
1229 note = ''
Ka-Ping Yee3bda8792001-03-23 13:17:50 +00001230 skipdocs = 0
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001231 if inspect.ismethod(object):
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001232 imclass = object.im_class
Ka-Ping Yee3bda8792001-03-23 13:17:50 +00001233 if cl:
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001234 if imclass is not cl:
1235 note = ' from ' + classname(imclass, mod)
Ka-Ping Yee3bda8792001-03-23 13:17:50 +00001236 else:
Guido van Rossumd8faa362007-04-27 19:54:29 +00001237 if object.im_self is not None:
Ka-Ping Yeeb7a48302001-04-12 20:39:14 +00001238 note = ' method of %s instance' % classname(
1239 object.im_self.__class__, mod)
1240 else:
1241 note = ' unbound %s method' % classname(imclass,mod)
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001242 object = object.im_func
1243
1244 if name == realname:
1245 title = self.bold(realname)
1246 else:
Raymond Hettinger54f02222002-06-01 14:18:47 +00001247 if (cl and realname in cl.__dict__ and
Ka-Ping Yee3bda8792001-03-23 13:17:50 +00001248 cl.__dict__[realname] is object):
1249 skipdocs = 1
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001250 title = self.bold(name) + ' = ' + realname
Tim Peters4bcfa312001-09-20 06:08:24 +00001251 if inspect.isfunction(object):
Guido van Rossum2e65f892007-02-28 22:03:49 +00001252 args, varargs, varkw, defaults, kwonlyargs, kwdefaults, ann = \
1253 inspect.getfullargspec(object)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001254 argspec = inspect.formatargspec(
Guido van Rossum2e65f892007-02-28 22:03:49 +00001255 args, varargs, varkw, defaults, kwonlyargs, kwdefaults, ann,
1256 formatvalue=self.formatvalue,
1257 formatannotation=inspect.formatannotationrelativeto(object))
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001258 if realname == '<lambda>':
Georg Brandl501dd0d2006-02-17 09:45:40 +00001259 title = self.bold(name) + ' lambda '
Guido van Rossum2e65f892007-02-28 22:03:49 +00001260 # XXX lambda's won't usually have func_annotations['return']
1261 # since the syntax doesn't support but it is possible.
1262 # So removing parentheses isn't truly safe.
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001263 argspec = argspec[1:-1] # remove parentheses
Tim Peters4bcfa312001-09-20 06:08:24 +00001264 else:
1265 argspec = '(...)'
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001266 decl = title + argspec + note
1267
Ka-Ping Yee3bda8792001-03-23 13:17:50 +00001268 if skipdocs:
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001269 return decl + '\n'
Ka-Ping Yee3bda8792001-03-23 13:17:50 +00001270 else:
1271 doc = getdoc(object) or ''
Neal Norwitz9d72bb42007-04-17 08:48:32 +00001272 return decl + '\n' + (doc and self.indent(doc).rstrip() + '\n')
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001273
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +00001274 def _docdescriptor(self, name, value, mod):
Johannes Gijsbers8de645a2004-11-07 19:16:05 +00001275 results = []
1276 push = results.append
1277
1278 if name:
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +00001279 push(self.bold(name))
1280 push('\n')
Johannes Gijsbers8de645a2004-11-07 19:16:05 +00001281 doc = getdoc(value) or ''
1282 if doc:
1283 push(self.indent(doc))
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +00001284 push('\n')
1285 return ''.join(results)
Johannes Gijsbers8de645a2004-11-07 19:16:05 +00001286
1287 def docproperty(self, object, name=None, mod=None, cl=None):
1288 """Produce text documentation for a property."""
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +00001289 return self._docdescriptor(name, object, mod)
Johannes Gijsbers8de645a2004-11-07 19:16:05 +00001290
Thomas Wouters0e3f5912006-08-11 14:57:12 +00001291 def docdata(self, object, name=None, mod=None, cl=None):
1292 """Produce text documentation for a data descriptor."""
1293 return self._docdescriptor(name, object, mod)
1294
Georg Brandl8b813db2005-10-01 16:32:31 +00001295 def docother(self, object, name=None, mod=None, parent=None, maxlen=None, doc=None):
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001296 """Produce text documentation for a data object."""
1297 repr = self.repr(object)
1298 if maxlen:
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001299 line = (name and name + ' = ' or '') + repr
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001300 chop = maxlen - len(line)
1301 if chop < 0: repr = repr[:chop] + '...'
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001302 line = (name and self.bold(name) + ' = ' or '') + repr
Tim Peters28355492001-09-23 21:29:55 +00001303 if doc is not None:
1304 line += '\n' + self.indent(str(doc))
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001305 return line
1306
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001307# --------------------------------------------------------- user interfaces
1308
1309def pager(text):
1310 """The first time this is called, determine what kind of pager to use."""
1311 global pager
1312 pager = getpager()
1313 pager(text)
1314
1315def getpager():
1316 """Decide what method to use for paging through text."""
Guido van Rossuma01a8b62007-05-27 09:20:14 +00001317 if not hasattr(sys.stdout, "isatty"):
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001318 return plainpager
1319 if not sys.stdin.isatty() or not sys.stdout.isatty():
1320 return plainpager
Raymond Hettinger54f02222002-06-01 14:18:47 +00001321 if 'PAGER' in os.environ:
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +00001322 if sys.platform == 'win32': # pipes completely broken in Windows
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001323 return lambda text: tempfilepager(plain(text), os.environ['PAGER'])
Raymond Hettingerdbecd932005-02-06 06:57:08 +00001324 elif os.environ.get('TERM') in ('dumb', 'emacs'):
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001325 return lambda text: pipepager(plain(text), os.environ['PAGER'])
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +00001326 else:
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001327 return lambda text: pipepager(text, os.environ['PAGER'])
Ka-Ping Yeea487e4e2005-11-05 04:49:18 +00001328 if os.environ.get('TERM') in ('dumb', 'emacs'):
1329 return plainpager
Andrew MacIntyre54e0eab2002-03-03 03:12:30 +00001330 if sys.platform == 'win32' or sys.platform.startswith('os2'):
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001331 return lambda text: tempfilepager(plain(text), 'more <')
Skip Montanarod404bee2002-09-26 21:44:57 +00001332 if hasattr(os, 'system') and os.system('(less) 2>/dev/null') == 0:
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001333 return lambda text: pipepager(text, 'less')
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001334
1335 import tempfile
Guido van Rossum3b0a3292002-08-09 16:38:32 +00001336 (fd, filename) = tempfile.mkstemp()
1337 os.close(fd)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001338 try:
1339 if hasattr(os, 'system') and os.system('more %s' % filename) == 0:
1340 return lambda text: pipepager(text, 'more')
1341 else:
1342 return ttypager
1343 finally:
1344 os.unlink(filename)
1345
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001346def plain(text):
1347 """Remove boldface formatting from text."""
1348 return re.sub('.\b', '', text)
1349
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001350def pipepager(text, cmd):
1351 """Page through text by feeding it to another program."""
1352 pipe = os.popen(cmd, 'w')
1353 try:
1354 pipe.write(text)
1355 pipe.close()
1356 except IOError:
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001357 pass # Ignore broken pipes caused by quitting the pager program.
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001358
1359def tempfilepager(text, cmd):
1360 """Page through text by invoking a program on a temporary file."""
1361 import tempfile
Tim Peters550e4e52003-02-07 01:53:46 +00001362 filename = tempfile.mktemp()
1363 file = open(filename, 'w')
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001364 file.write(text)
1365 file.close()
1366 try:
Ka-Ping Yeec92cdf72001-03-02 05:54:35 +00001367 os.system(cmd + ' ' + filename)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001368 finally:
1369 os.unlink(filename)
1370
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001371def ttypager(text):
1372 """Page through text on a text terminal."""
Neal Norwitz9d72bb42007-04-17 08:48:32 +00001373 lines = plain(text).split('\n')
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001374 try:
1375 import tty
1376 fd = sys.stdin.fileno()
1377 old = tty.tcgetattr(fd)
1378 tty.setcbreak(fd)
1379 getchar = lambda: sys.stdin.read(1)
Ka-Ping Yee457aab22001-02-27 23:36:29 +00001380 except (ImportError, AttributeError):
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001381 tty = None
1382 getchar = lambda: sys.stdin.readline()[:-1][:1]
1383
1384 try:
1385 r = inc = os.environ.get('LINES', 25) - 1
Neal Norwitz9d72bb42007-04-17 08:48:32 +00001386 sys.stdout.write('\n'.join(lines[:inc]) + '\n')
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001387 while lines[r:]:
1388 sys.stdout.write('-- more --')
1389 sys.stdout.flush()
1390 c = getchar()
1391
Raymond Hettingerdbecd932005-02-06 06:57:08 +00001392 if c in ('q', 'Q'):
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001393 sys.stdout.write('\r \r')
1394 break
Raymond Hettingerdbecd932005-02-06 06:57:08 +00001395 elif c in ('\r', '\n'):
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001396 sys.stdout.write('\r \r' + lines[r] + '\n')
1397 r = r + 1
1398 continue
Raymond Hettingerdbecd932005-02-06 06:57:08 +00001399 if c in ('b', 'B', '\x1b'):
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001400 r = r - inc - inc
1401 if r < 0: r = 0
Neal Norwitz9d72bb42007-04-17 08:48:32 +00001402 sys.stdout.write('\n' + '\n'.join(lines[r:r+inc]) + '\n')
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001403 r = r + inc
1404
1405 finally:
1406 if tty:
1407 tty.tcsetattr(fd, tty.TCSAFLUSH, old)
1408
1409def plainpager(text):
1410 """Simply print unformatted text. This is the ultimate fallback."""
1411 sys.stdout.write(plain(text))
1412
1413def describe(thing):
Ka-Ping Yee3bda8792001-03-23 13:17:50 +00001414 """Produce a short description of the given thing."""
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001415 if inspect.ismodule(thing):
1416 if thing.__name__ in sys.builtin_module_names:
1417 return 'built-in module ' + thing.__name__
1418 if hasattr(thing, '__path__'):
1419 return 'package ' + thing.__name__
1420 else:
1421 return 'module ' + thing.__name__
1422 if inspect.isbuiltin(thing):
1423 return 'built-in function ' + thing.__name__
Thomas Wouters0e3f5912006-08-11 14:57:12 +00001424 if inspect.isgetsetdescriptor(thing):
1425 return 'getset descriptor %s.%s.%s' % (
1426 thing.__objclass__.__module__, thing.__objclass__.__name__,
1427 thing.__name__)
1428 if inspect.ismemberdescriptor(thing):
1429 return 'member descriptor %s.%s.%s' % (
1430 thing.__objclass__.__module__, thing.__objclass__.__name__,
1431 thing.__name__)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001432 if inspect.isclass(thing):
1433 return 'class ' + thing.__name__
1434 if inspect.isfunction(thing):
1435 return 'function ' + thing.__name__
1436 if inspect.ismethod(thing):
1437 return 'method ' + thing.__name__
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001438 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."""
Neal Norwitz9d72bb42007-04-17 08:48:32 +00001442 parts = [part for part in path.split('.') if part]
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001443 module, n = None, 0
1444 while n < len(parts):
Neal Norwitz9d72bb42007-04-17 08:48:32 +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
Guido van Rossumd8faa362007-04-27 19:54:29 +00001473def render_doc(thing, title='Python Library Documentation: %s', forceload=0):
1474 """Render text documentation, given an object or a path to an object."""
1475 object, name = resolve(thing, forceload)
1476 desc = describe(object)
1477 module = inspect.getmodule(object)
1478 if name and '.' in name:
1479 desc += ' in ' + name[:name.rfind('.')]
1480 elif module and module is not object:
1481 desc += ' in module ' + module.__name__
1482 elif not (inspect.ismodule(object) or
1483 inspect.isclass(object) or
1484 inspect.isroutine(object) or
1485 inspect.isgetsetdescriptor(object) or
1486 inspect.ismemberdescriptor(object) or
1487 isinstance(object, property)):
1488 # If the passed object is a piece of data or an instance,
1489 # document its available methods instead of its value.
1490 object = type(object)
1491 desc += ' object'
1492 return title % desc + '\n\n' + text.document(object, name)
1493
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001494def doc(thing, title='Python Library Documentation: %s', forceload=0):
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001495 """Display text documentation, given an object or a path to an object."""
Ka-Ping Yee45daeb02002-08-11 15:11:33 +00001496 try:
1497 object, name = resolve(thing, forceload)
1498 desc = describe(object)
1499 module = inspect.getmodule(object)
1500 if name and '.' in name:
1501 desc += ' in ' + name[:name.rfind('.')]
1502 elif module and module is not object:
1503 desc += ' in module ' + module.__name__
Thomas Woutersfc7bb8c2007-01-15 15:49:28 +00001504 elif not (inspect.ismodule(object) or
1505 inspect.isclass(object) or
1506 inspect.isroutine(object) or
1507 inspect.isgetsetdescriptor(object) or
1508 inspect.ismemberdescriptor(object) or
1509 isinstance(object, property)):
Ka-Ping Yeebba6acc2005-02-19 22:58:26 +00001510 # If the passed object is a piece of data or an instance,
1511 # document its available methods instead of its value.
1512 object = type(object)
1513 desc += ' object'
Guido van Rossumd8faa362007-04-27 19:54:29 +00001514 pager(render_doc(thing, title, forceload))
Guido van Rossumb940e112007-01-10 16:19:56 +00001515 except (ImportError, ErrorDuringImport) as value:
Guido van Rossumbe19ed72007-02-09 05:37:30 +00001516 print(value)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001517
Ka-Ping Yee45daeb02002-08-11 15:11:33 +00001518def writedoc(thing, forceload=0):
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001519 """Write HTML documentation to a file in the current directory."""
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001520 try:
Ka-Ping Yee45daeb02002-08-11 15:11:33 +00001521 object, name = resolve(thing, forceload)
1522 page = html.page(describe(object), html.document(object, name))
1523 file = open(name + '.html', 'w')
1524 file.write(page)
1525 file.close()
Guido van Rossumbe19ed72007-02-09 05:37:30 +00001526 print('wrote', name + '.html')
Guido van Rossumb940e112007-01-10 16:19:56 +00001527 except (ImportError, ErrorDuringImport) as value:
Guido van Rossumbe19ed72007-02-09 05:37:30 +00001528 print(value)
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001529
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001530def writedocs(dir, pkgpath='', done=None):
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001531 """Write out HTML documentation for all modules in a directory tree."""
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001532 if done is None: done = {}
Thomas Wouters49fd7fa2006-04-21 10:40:58 +00001533 for importer, modname, ispkg in pkgutil.walk_packages([dir], pkgpath):
1534 writedoc(modname)
1535 return
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001536
Neal Norwitzce96f692006-03-17 06:49:51 +00001537def raw_input(prompt):
1538 sys.stdout.write(prompt)
1539 sys.stdout.flush()
1540 return sys.stdin.readline()
1541
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001542class Helper:
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001543 keywords = {
1544 'and': 'BOOLEAN',
Guido van Rossumd8faa362007-04-27 19:54:29 +00001545 'as': 'with',
Ka-Ping Yee79c009d2001-04-13 10:53:25 +00001546 'assert': ('ref/assert', ''),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001547 'break': ('ref/break', 'while for'),
1548 'class': ('ref/class', 'CLASSES SPECIALMETHODS'),
1549 'continue': ('ref/continue', 'while for'),
1550 'def': ('ref/function', ''),
1551 'del': ('ref/del', 'BASICMETHODS'),
1552 'elif': 'if',
1553 'else': ('ref/if', 'while for'),
1554 'except': 'try',
1555 'exec': ('ref/exec', ''),
1556 'finally': 'try',
1557 'for': ('ref/for', 'break continue while'),
1558 'from': 'import',
1559 'global': ('ref/global', 'NAMESPACES'),
1560 'if': ('ref/if', 'TRUTHVALUE'),
1561 'import': ('ref/import', 'MODULES'),
1562 'in': ('ref/comparisons', 'SEQUENCEMETHODS2'),
1563 'is': 'COMPARISON',
Neal Norwitz742dde42003-03-30 20:31:34 +00001564 'lambda': ('ref/lambdas', 'FUNCTIONS'),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001565 'not': 'BOOLEAN',
1566 'or': 'BOOLEAN',
Neal Norwitz742dde42003-03-30 20:31:34 +00001567 'pass': ('ref/pass', ''),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001568 'print': ('ref/print', ''),
1569 'raise': ('ref/raise', 'EXCEPTIONS'),
Ka-Ping Yee79c009d2001-04-13 10:53:25 +00001570 'return': ('ref/return', 'FUNCTIONS'),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001571 'try': ('ref/try', 'EXCEPTIONS'),
1572 'while': ('ref/while', 'break continue if TRUTHVALUE'),
Guido van Rossumd8faa362007-04-27 19:54:29 +00001573 'with': ('ref/with', 'CONTEXTMANAGERS EXCEPTIONS yield'),
Tim Petersfb05c4e2002-10-30 05:21:00 +00001574 'yield': ('ref/yield', ''),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001575 }
1576
1577 topics = {
1578 'TYPES': ('ref/types', 'STRINGS UNICODE NUMBERS SEQUENCES MAPPINGS FUNCTIONS CLASSES MODULES FILES inspect'),
Ka-Ping Yee79c009d2001-04-13 10:53:25 +00001579 'STRINGS': ('ref/strings', 'str UNICODE SEQUENCES STRINGMETHODS FORMATTING TYPES'),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001580 'STRINGMETHODS': ('lib/string-methods', 'STRINGS FORMATTING'),
1581 'FORMATTING': ('lib/typesseq-strings', 'OPERATORS'),
Neal Norwitz54f871e2003-05-26 13:49:54 +00001582 'UNICODE': ('ref/strings', 'encodings unicode SEQUENCES STRINGMETHODS FORMATTING TYPES'),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001583 'NUMBERS': ('ref/numbers', 'INTEGER FLOAT COMPLEX TYPES'),
1584 'INTEGER': ('ref/integers', 'int range'),
1585 'FLOAT': ('ref/floating', 'float math'),
1586 'COMPLEX': ('ref/imaginary', 'complex cmath'),
Guido van Rossum805365e2007-05-07 22:24:25 +00001587 'SEQUENCES': ('lib/typesseq', 'STRINGMETHODS FORMATTING range LISTS'),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001588 'MAPPINGS': 'DICTIONARIES',
1589 'FUNCTIONS': ('lib/typesfunctions', 'def TYPES'),
1590 'METHODS': ('lib/typesmethods', 'class def CLASSES TYPES'),
1591 'CODEOBJECTS': ('lib/bltin-code-objects', 'compile FUNCTIONS TYPES'),
Ka-Ping Yee79c009d2001-04-13 10:53:25 +00001592 'TYPEOBJECTS': ('lib/bltin-type-objects', 'types TYPES'),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001593 'FRAMEOBJECTS': 'TYPES',
1594 'TRACEBACKS': 'TYPES',
1595 'NONE': ('lib/bltin-null-object', ''),
1596 'ELLIPSIS': ('lib/bltin-ellipsis-object', 'SLICINGS'),
1597 'FILES': ('lib/bltin-file-objects', ''),
1598 'SPECIALATTRIBUTES': ('lib/specialattrs', ''),
1599 'CLASSES': ('ref/types', 'class SPECIALMETHODS PRIVATENAMES'),
1600 'MODULES': ('lib/typesmodules', 'import'),
1601 'PACKAGES': 'import',
Neal Norwitz2eca4402006-08-29 04:40:24 +00001602 '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 +00001603 'OPERATORS': 'EXPRESSIONS',
1604 'PRECEDENCE': 'EXPRESSIONS',
1605 'OBJECTS': ('ref/objects', 'TYPES'),
1606 'SPECIALMETHODS': ('ref/specialnames', 'BASICMETHODS ATTRIBUTEMETHODS CALLABLEMETHODS SEQUENCEMETHODS1 MAPPINGMETHODS SEQUENCEMETHODS2 NUMBERMETHODS CLASSES'),
Ka-Ping Yee79c009d2001-04-13 10:53:25 +00001607 'BASICMETHODS': ('ref/customization', 'cmp hash repr str SPECIALMETHODS'),
1608 'ATTRIBUTEMETHODS': ('ref/attribute-access', 'ATTRIBUTES SPECIALMETHODS'),
1609 'CALLABLEMETHODS': ('ref/callable-types', 'CALLS SPECIALMETHODS'),
1610 'SEQUENCEMETHODS1': ('ref/sequence-types', 'SEQUENCES SEQUENCEMETHODS2 SPECIALMETHODS'),
1611 'SEQUENCEMETHODS2': ('ref/sequence-methods', 'SEQUENCES SEQUENCEMETHODS1 SPECIALMETHODS'),
1612 'MAPPINGMETHODS': ('ref/sequence-types', 'MAPPINGS SPECIALMETHODS'),
1613 'NUMBERMETHODS': ('ref/numeric-types', 'NUMBERS AUGMENTEDASSIGNMENT SPECIALMETHODS'),
Neal Norwitz54f871e2003-05-26 13:49:54 +00001614 'EXECUTION': ('ref/execmodel', 'NAMESPACES DYNAMICFEATURES EXCEPTIONS'),
1615 'NAMESPACES': ('ref/naming', 'global ASSIGNMENT DELETION DYNAMICFEATURES'),
1616 'DYNAMICFEATURES': ('ref/dynamic-features', ''),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001617 'SCOPING': 'NAMESPACES',
1618 'FRAMES': 'NAMESPACES',
1619 'EXCEPTIONS': ('ref/exceptions', 'try except finally raise'),
Neal Norwitz54f871e2003-05-26 13:49:54 +00001620 'COERCIONS': ('ref/coercion-rules','CONVERSIONS'),
1621 'CONVERSIONS': ('ref/conversions', 'COERCIONS'),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001622 'IDENTIFIERS': ('ref/identifiers', 'keywords SPECIALIDENTIFIERS'),
1623 'SPECIALIDENTIFIERS': ('ref/id-classes', ''),
Ka-Ping Yee79c009d2001-04-13 10:53:25 +00001624 'PRIVATENAMES': ('ref/atom-identifiers', ''),
Neal Norwitz2eca4402006-08-29 04:40:24 +00001625 'LITERALS': ('ref/atom-literals', 'STRINGS NUMBERS TUPLELITERALS LISTLITERALS DICTIONARYLITERALS'),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001626 'TUPLES': 'SEQUENCES',
Ka-Ping Yee79c009d2001-04-13 10:53:25 +00001627 'TUPLELITERALS': ('ref/exprlists', 'TUPLES LITERALS'),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001628 'LISTS': ('lib/typesseq-mutable', 'LISTLITERALS'),
Ka-Ping Yee79c009d2001-04-13 10:53:25 +00001629 'LISTLITERALS': ('ref/lists', 'LISTS LITERALS'),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001630 'DICTIONARIES': ('lib/typesmapping', 'DICTIONARYLITERALS'),
Ka-Ping Yee79c009d2001-04-13 10:53:25 +00001631 'DICTIONARYLITERALS': ('ref/dict', 'DICTIONARIES LITERALS'),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001632 'ATTRIBUTES': ('ref/attribute-references', 'getattr hasattr setattr ATTRIBUTEMETHODS'),
1633 'SUBSCRIPTS': ('ref/subscriptions', 'SEQUENCEMETHODS1'),
1634 'SLICINGS': ('ref/slicings', 'SEQUENCEMETHODS2'),
1635 'CALLS': ('ref/calls', 'EXPRESSIONS'),
1636 'POWER': ('ref/power', 'EXPRESSIONS'),
1637 'UNARY': ('ref/unary', 'EXPRESSIONS'),
1638 'BINARY': ('ref/binary', 'EXPRESSIONS'),
1639 'SHIFTING': ('ref/shifting', 'EXPRESSIONS'),
1640 'BITWISE': ('ref/bitwise', 'EXPRESSIONS'),
1641 'COMPARISON': ('ref/comparisons', 'EXPRESSIONS BASICMETHODS'),
Neal Norwitzf98159c2003-02-07 20:49:40 +00001642 'BOOLEAN': ('ref/Booleans', 'EXPRESSIONS TRUTHVALUE'),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001643 'ASSERTION': 'assert',
1644 'ASSIGNMENT': ('ref/assignment', 'AUGMENTEDASSIGNMENT'),
Ka-Ping Yee79c009d2001-04-13 10:53:25 +00001645 'AUGMENTEDASSIGNMENT': ('ref/augassign', 'NUMBERMETHODS'),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001646 'DELETION': 'del',
1647 'PRINTING': 'print',
1648 'RETURNING': 'return',
1649 'IMPORTING': 'import',
1650 'CONDITIONAL': 'if',
1651 'LOOPING': ('ref/compound', 'for while break continue'),
1652 'TRUTHVALUE': ('lib/truth', 'if while and or not BASICMETHODS'),
Ka-Ping Yee79c009d2001-04-13 10:53:25 +00001653 'DEBUGGING': ('lib/module-pdb', 'pdb'),
Guido van Rossumd8faa362007-04-27 19:54:29 +00001654 'CONTEXTMANAGERS': ('ref/context-managers', 'with'),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001655 }
1656
1657 def __init__(self, input, output):
1658 self.input = input
1659 self.output = output
1660 self.docdir = None
1661 execdir = os.path.dirname(sys.executable)
1662 homedir = os.environ.get('PYTHONHOME')
Guido van Rossumd8faa362007-04-27 19:54:29 +00001663 join = os.path.join
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001664 for dir in [os.environ.get('PYTHONDOCS'),
1665 homedir and os.path.join(homedir, 'doc'),
Guido van Rossumd8faa362007-04-27 19:54:29 +00001666 join(execdir, 'doc'), # for Windows
1667 join(sys.prefix, 'doc/python-docs-' + sys.version.split()[0]),
1668 join(sys.prefix, 'doc/python-' + sys.version.split()[0]),
1669 join(sys.prefix, 'doc/python-docs-' + sys.version[:3]),
1670 join(sys.prefix, 'doc/python-' + sys.version[:3]),
1671 join(sys.prefix, 'Resources/English.lproj/Documentation')]:
1672 if dir and os.path.isdir(join(dir, 'lib')):
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001673 self.docdir = dir
Guido van Rossumd8faa362007-04-27 19:54:29 +00001674 break
1675 if dir and os.path.isdir(join(dir, 'html', 'lib')):
1676 self.docdir = join(dir, 'html')
1677 break
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001678
Ka-Ping Yee79c009d2001-04-13 10:53:25 +00001679 def __repr__(self):
Ka-Ping Yee9bc576b2001-04-13 13:57:31 +00001680 if inspect.stack()[1][3] == '?':
Ka-Ping Yee79c009d2001-04-13 10:53:25 +00001681 self()
1682 return ''
Ka-Ping Yee9bc576b2001-04-13 13:57:31 +00001683 return '<pydoc.Helper instance>'
Ka-Ping Yee79c009d2001-04-13 10:53:25 +00001684
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001685 def __call__(self, request=None):
1686 if request is not None:
1687 self.help(request)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001688 else:
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001689 self.intro()
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001690 self.interact()
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001691 self.output.write('''
Fred Drakee61967f2001-05-10 18:41:02 +00001692You are now leaving help and returning to the Python interpreter.
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001693If you want to ask for help on a particular object directly from the
1694interpreter, you can type "help(object)". Executing "help('string')"
1695has the same effect as typing a particular string at the help> prompt.
1696''')
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001697
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001698 def interact(self):
1699 self.output.write('\n')
Guido van Rossum8ca162f2002-04-07 06:36:23 +00001700 while True:
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001701 try:
Johannes Gijsberse7691d32004-08-17 13:21:53 +00001702 request = self.getline('help> ')
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001703 if not request: break
Johannes Gijsberse7691d32004-08-17 13:21:53 +00001704 except (KeyboardInterrupt, EOFError):
1705 break
Neal Norwitz9d72bb42007-04-17 08:48:32 +00001706 request = replace(request, '"', '', "'", '').strip()
1707 if request.lower() in ('q', 'quit'): break
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001708 self.help(request)
1709
Johannes Gijsberse7691d32004-08-17 13:21:53 +00001710 def getline(self, prompt):
1711 """Read one line, using raw_input when available."""
1712 if self.input is sys.stdin:
1713 return raw_input(prompt)
1714 else:
1715 self.output.write(prompt)
1716 self.output.flush()
1717 return self.input.readline()
1718
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001719 def help(self, request):
1720 if type(request) is type(''):
1721 if request == 'help': self.intro()
1722 elif request == 'keywords': self.listkeywords()
1723 elif request == 'topics': self.listtopics()
1724 elif request == 'modules': self.listmodules()
1725 elif request[:8] == 'modules ':
Neal Norwitz9d72bb42007-04-17 08:48:32 +00001726 self.listmodules(request.split()[1])
Raymond Hettinger54f02222002-06-01 14:18:47 +00001727 elif request in self.keywords: self.showtopic(request)
1728 elif request in self.topics: self.showtopic(request)
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001729 elif request: doc(request, 'Help on %s:')
1730 elif isinstance(request, Helper): self()
1731 else: doc(request, 'Help on %s:')
1732 self.output.write('\n')
1733
1734 def intro(self):
1735 self.output.write('''
1736Welcome to Python %s! This is the online help utility.
1737
1738If this is your first time using Python, you should definitely check out
1739the tutorial on the Internet at http://www.python.org/doc/tut/.
1740
1741Enter the name of any module, keyword, or topic to get help on writing
1742Python programs and using Python modules. To quit this help utility and
1743return to the interpreter, just type "quit".
1744
1745To get a list of available modules, keywords, or topics, type "modules",
1746"keywords", or "topics". Each module also comes with a one-line summary
1747of what it does; to list the modules whose summaries contain a given word
1748such as "spam", type "modules spam".
1749''' % sys.version[:3])
1750
1751 def list(self, items, columns=4, width=80):
1752 items = items[:]
1753 items.sort()
1754 colw = width / columns
1755 rows = (len(items) + columns - 1) / columns
1756 for row in range(rows):
1757 for col in range(columns):
1758 i = col * rows + row
1759 if i < len(items):
1760 self.output.write(items[i])
1761 if col < columns - 1:
1762 self.output.write(' ' + ' ' * (colw-1 - len(items[i])))
1763 self.output.write('\n')
1764
1765 def listkeywords(self):
1766 self.output.write('''
1767Here is a list of the Python keywords. Enter any keyword to get more help.
1768
1769''')
1770 self.list(self.keywords.keys())
1771
1772 def listtopics(self):
1773 self.output.write('''
1774Here is a list of available topics. Enter any topic name to get more help.
1775
1776''')
1777 self.list(self.topics.keys())
1778
1779 def showtopic(self, topic):
1780 if not self.docdir:
1781 self.output.write('''
1782Sorry, topic and keyword documentation is not available because the Python
1783HTML documentation files could not be found. If you have installed them,
1784please set the environment variable PYTHONDOCS to indicate their location.
Thomas Wouters902d6eb2007-01-09 23:18:33 +00001785
1786On the Microsoft Windows operating system, the files can be built by
1787running "hh -decompile . PythonNN.chm" in the C:\PythonNN\Doc> directory.
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001788''')
1789 return
1790 target = self.topics.get(topic, self.keywords.get(topic))
1791 if not target:
1792 self.output.write('no documentation found for %s\n' % repr(topic))
1793 return
1794 if type(target) is type(''):
1795 return self.showtopic(target)
1796
1797 filename, xrefs = target
1798 filename = self.docdir + '/' + filename + '.html'
1799 try:
1800 file = open(filename)
1801 except:
1802 self.output.write('could not read docs from %s\n' % filename)
1803 return
1804
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001805 divpat = re.compile('<div[^>]*navigat.*?</div.*?>', re.I | re.S)
1806 addrpat = re.compile('<address.*?>.*?</address.*?>', re.I | re.S)
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001807 document = re.sub(addrpat, '', re.sub(divpat, '', file.read()))
1808 file.close()
1809
1810 import htmllib, formatter, StringIO
1811 buffer = StringIO.StringIO()
1812 parser = htmllib.HTMLParser(
1813 formatter.AbstractFormatter(formatter.DumbWriter(buffer)))
1814 parser.start_table = parser.do_p
1815 parser.end_table = lambda parser=parser: parser.do_p({})
1816 parser.start_tr = parser.do_br
1817 parser.start_td = parser.start_th = lambda a, b=buffer: b.write('\t')
1818 parser.feed(document)
1819 buffer = replace(buffer.getvalue(), '\xa0', ' ', '\n', '\n ')
Neal Norwitz9d72bb42007-04-17 08:48:32 +00001820 pager(' ' + buffer.strip() + '\n')
Ka-Ping Yeeda793892001-04-13 11:02:51 +00001821 if xrefs:
1822 buffer = StringIO.StringIO()
1823 formatter.DumbWriter(buffer).send_flowing_data(
Neal Norwitz9d72bb42007-04-17 08:48:32 +00001824 'Related help topics: ' + ', '.join(xrefs.split()) + '\n')
Ka-Ping Yeeda793892001-04-13 11:02:51 +00001825 self.output.write('\n%s\n' % buffer.getvalue())
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001826
1827 def listmodules(self, key=''):
1828 if key:
1829 self.output.write('''
1830Here is a list of matching modules. Enter any module name to get more help.
1831
1832''')
1833 apropos(key)
1834 else:
1835 self.output.write('''
1836Please wait a moment while I gather a list of all available modules...
1837
1838''')
1839 modules = {}
1840 def callback(path, modname, desc, modules=modules):
1841 if modname and modname[-9:] == '.__init__':
1842 modname = modname[:-9] + ' (package)'
Neal Norwitz9d72bb42007-04-17 08:48:32 +00001843 if modname.find('.') < 0:
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001844 modules[modname] = 1
1845 ModuleScanner().run(callback)
1846 self.list(modules.keys())
1847 self.output.write('''
1848Enter any module name to get more help. Or, type "modules spam" to search
1849for modules whose descriptions contain the word "spam".
1850''')
1851
1852help = Helper(sys.stdin, sys.stdout)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001853
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001854class Scanner:
1855 """A generic tree iterator."""
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001856 def __init__(self, roots, children, descendp):
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001857 self.roots = roots[:]
1858 self.state = []
1859 self.children = children
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001860 self.descendp = descendp
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001861
1862 def next(self):
1863 if not self.state:
1864 if not self.roots:
1865 return None
1866 root = self.roots.pop(0)
1867 self.state = [(root, self.children(root))]
1868 node, children = self.state[-1]
1869 if not children:
1870 self.state.pop()
1871 return self.next()
1872 child = children.pop(0)
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001873 if self.descendp(child):
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001874 self.state.append((child, self.children(child)))
1875 return child
1876
Thomas Wouters49fd7fa2006-04-21 10:40:58 +00001877
1878class ModuleScanner:
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001879 """An interruptible scanner that searches module synopses."""
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001880
Ka-Ping Yee66246962001-04-12 11:59:50 +00001881 def run(self, callback, key=None, completer=None):
Neal Norwitz9d72bb42007-04-17 08:48:32 +00001882 if key: key = key.lower()
Guido van Rossum8ca162f2002-04-07 06:36:23 +00001883 self.quit = False
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001884 seen = {}
1885
1886 for modname in sys.builtin_module_names:
Ka-Ping Yee239432a2001-03-02 02:45:08 +00001887 if modname != '__main__':
1888 seen[modname] = 1
Ka-Ping Yee66246962001-04-12 11:59:50 +00001889 if key is None:
1890 callback(None, modname, '')
1891 else:
Neal Norwitz9d72bb42007-04-17 08:48:32 +00001892 name = __import__(modname).__doc__ or ''
1893 desc = name.split('\n')[0]
1894 name = modname + ' - ' + desc
1895 if name.lower().find(key) >= 0:
Ka-Ping Yee66246962001-04-12 11:59:50 +00001896 callback(None, modname, desc)
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001897
Thomas Wouters49fd7fa2006-04-21 10:40:58 +00001898 for importer, modname, ispkg in pkgutil.walk_packages():
1899 if self.quit:
1900 break
1901 if key is None:
1902 callback(None, modname, '')
1903 else:
1904 loader = importer.find_module(modname)
1905 if hasattr(loader,'get_source'):
1906 import StringIO
1907 desc = source_synopsis(
1908 StringIO.StringIO(loader.get_source(modname))
1909 ) or ''
1910 if hasattr(loader,'get_filename'):
1911 path = loader.get_filename(modname)
Ka-Ping Yee66246962001-04-12 11:59:50 +00001912 else:
Thomas Wouters49fd7fa2006-04-21 10:40:58 +00001913 path = None
1914 else:
1915 module = loader.load_module(modname)
1916 desc = (module.__doc__ or '').splitlines()[0]
1917 path = getattr(module,'__file__',None)
Neal Norwitz9d72bb42007-04-17 08:48:32 +00001918 name = modname + ' - ' + desc
1919 if name.lower().find(key) >= 0:
Thomas Wouters49fd7fa2006-04-21 10:40:58 +00001920 callback(path, modname, desc)
1921
1922 if completer:
1923 completer()
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001924
1925def apropos(key):
1926 """Print all the one-line module summaries that contain a substring."""
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001927 def callback(path, modname, desc):
1928 if modname[-9:] == '.__init__':
1929 modname = modname[:-9] + ' (package)'
Guido van Rossumbe19ed72007-02-09 05:37:30 +00001930 print(modname, desc and '- ' + desc)
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +00001931 try: import warnings
1932 except ImportError: pass
1933 else: warnings.filterwarnings('ignore') # ignore problems during import
Ka-Ping Yee66246962001-04-12 11:59:50 +00001934 ModuleScanner().run(callback, key)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001935
1936# --------------------------------------------------- web browser interface
1937
Ka-Ping Yee66246962001-04-12 11:59:50 +00001938def serve(port, callback=None, completer=None):
Ka-Ping Yeefd540692001-04-12 12:54:36 +00001939 import BaseHTTPServer, mimetools, select
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001940
1941 # Patch up mimetools.Message so it doesn't break if rfc822 is reloaded.
1942 class Message(mimetools.Message):
1943 def __init__(self, fp, seekable=1):
1944 Message = self.__class__
1945 Message.__bases__[0].__bases__[0].__init__(self, fp, seekable)
1946 self.encodingheader = self.getheader('content-transfer-encoding')
1947 self.typeheader = self.getheader('content-type')
1948 self.parsetype()
1949 self.parseplist()
1950
1951 class DocHandler(BaseHTTPServer.BaseHTTPRequestHandler):
1952 def send_document(self, title, contents):
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001953 try:
1954 self.send_response(200)
1955 self.send_header('Content-Type', 'text/html')
1956 self.end_headers()
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001957 self.wfile.write(html.page(title, contents))
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001958 except IOError: pass
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001959
1960 def do_GET(self):
1961 path = self.path
1962 if path[-5:] == '.html': path = path[:-5]
1963 if path[:1] == '/': path = path[1:]
1964 if path and path != '.':
1965 try:
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001966 obj = locate(path, forceload=1)
Guido van Rossumb940e112007-01-10 16:19:56 +00001967 except ErrorDuringImport as value:
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001968 self.send_document(path, html.escape(str(value)))
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001969 return
Ka-Ping Yee3bda8792001-03-23 13:17:50 +00001970 if obj:
1971 self.send_document(describe(obj), html.document(obj, path))
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001972 else:
1973 self.send_document(path,
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001974'no Python documentation found for %s' % repr(path))
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001975 else:
1976 heading = html.heading(
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001977'<big><big><strong>Python: Index of Modules</strong></big></big>',
1978'#ffffff', '#7799ee')
Ka-Ping Yee239432a2001-03-02 02:45:08 +00001979 def bltinlink(name):
1980 return '<a href="%s.html">%s</a>' % (name, name)
Ka-Ping Yee3bda8792001-03-23 13:17:50 +00001981 names = filter(lambda x: x != '__main__',
1982 sys.builtin_module_names)
Ka-Ping Yee239432a2001-03-02 02:45:08 +00001983 contents = html.multicolumn(names, bltinlink)
1984 indices = ['<p>' + html.bigsection(
1985 'Built-in Modules', '#ffffff', '#ee77aa', contents)]
1986
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001987 seen = {}
Thomas Wouters49fd7fa2006-04-21 10:40:58 +00001988 for dir in sys.path:
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001989 indices.append(html.index(dir, seen))
Neal Norwitz9d72bb42007-04-17 08:48:32 +00001990 contents = heading + ' '.join(indices) + '''<p align=right>
Tim Peters2306d242001-09-25 03:18:32 +00001991<font color="#909090" face="helvetica, arial"><strong>
1992pydoc</strong> by Ka-Ping Yee &lt;ping@lfw.org&gt;</font>'''
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001993 self.send_document('Index of Modules', contents)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001994
1995 def log_message(self, *args): pass
1996
Ka-Ping Yeefd540692001-04-12 12:54:36 +00001997 class DocServer(BaseHTTPServer.HTTPServer):
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001998 def __init__(self, port, callback):
Ka-Ping Yeedb8ed152001-03-02 05:58:17 +00001999 host = (sys.platform == 'mac') and '127.0.0.1' or 'localhost'
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00002000 self.address = ('', port)
Ka-Ping Yeedb8ed152001-03-02 05:58:17 +00002001 self.url = 'http://%s:%d/' % (host, port)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002002 self.callback = callback
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002003 self.base.__init__(self, self.address, self.handler)
2004
2005 def serve_until_quit(self):
2006 import select
Guido van Rossum8ca162f2002-04-07 06:36:23 +00002007 self.quit = False
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002008 while not self.quit:
2009 rd, wr, ex = select.select([self.socket.fileno()], [], [], 1)
2010 if rd: self.handle_request()
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002011
2012 def server_activate(self):
2013 self.base.server_activate(self)
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002014 if self.callback: self.callback(self)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002015
2016 DocServer.base = BaseHTTPServer.HTTPServer
2017 DocServer.handler = DocHandler
2018 DocHandler.MessageClass = Message
2019 try:
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00002020 try:
2021 DocServer(port, callback).serve_until_quit()
2022 except (KeyboardInterrupt, select.error):
2023 pass
2024 finally:
Ka-Ping Yee66246962001-04-12 11:59:50 +00002025 if completer: completer()
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002026
2027# ----------------------------------------------------- graphical interface
2028
2029def gui():
2030 """Graphical interface (starts web server and pops up a control window)."""
2031 class GUI:
2032 def __init__(self, window, port=7464):
2033 self.window = window
2034 self.server = None
2035 self.scanner = None
2036
2037 import Tkinter
2038 self.server_frm = Tkinter.Frame(window)
2039 self.title_lbl = Tkinter.Label(self.server_frm,
2040 text='Starting server...\n ')
2041 self.open_btn = Tkinter.Button(self.server_frm,
2042 text='open browser', command=self.open, state='disabled')
2043 self.quit_btn = Tkinter.Button(self.server_frm,
2044 text='quit serving', command=self.quit, state='disabled')
2045
2046 self.search_frm = Tkinter.Frame(window)
2047 self.search_lbl = Tkinter.Label(self.search_frm, text='Search for')
2048 self.search_ent = Tkinter.Entry(self.search_frm)
2049 self.search_ent.bind('<Return>', self.search)
2050 self.stop_btn = Tkinter.Button(self.search_frm,
2051 text='stop', pady=0, command=self.stop, state='disabled')
2052 if sys.platform == 'win32':
Ka-Ping Yeec92cdf72001-03-02 05:54:35 +00002053 # Trying to hide and show this button crashes under Windows.
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002054 self.stop_btn.pack(side='right')
2055
2056 self.window.title('pydoc')
2057 self.window.protocol('WM_DELETE_WINDOW', self.quit)
2058 self.title_lbl.pack(side='top', fill='x')
2059 self.open_btn.pack(side='left', fill='x', expand=1)
2060 self.quit_btn.pack(side='right', fill='x', expand=1)
2061 self.server_frm.pack(side='top', fill='x')
2062
2063 self.search_lbl.pack(side='left')
2064 self.search_ent.pack(side='right', fill='x', expand=1)
2065 self.search_frm.pack(side='top', fill='x')
2066 self.search_ent.focus_set()
2067
Ka-Ping Yeec92cdf72001-03-02 05:54:35 +00002068 font = ('helvetica', sys.platform == 'win32' and 8 or 10)
Ka-Ping Yee239432a2001-03-02 02:45:08 +00002069 self.result_lst = Tkinter.Listbox(window, font=font, height=6)
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002070 self.result_lst.bind('<Button-1>', self.select)
2071 self.result_lst.bind('<Double-Button-1>', self.goto)
2072 self.result_scr = Tkinter.Scrollbar(window,
2073 orient='vertical', command=self.result_lst.yview)
2074 self.result_lst.config(yscrollcommand=self.result_scr.set)
2075
2076 self.result_frm = Tkinter.Frame(window)
2077 self.goto_btn = Tkinter.Button(self.result_frm,
2078 text='go to selected', command=self.goto)
2079 self.hide_btn = Tkinter.Button(self.result_frm,
2080 text='hide results', command=self.hide)
2081 self.goto_btn.pack(side='left', fill='x', expand=1)
2082 self.hide_btn.pack(side='right', fill='x', expand=1)
2083
2084 self.window.update()
2085 self.minwidth = self.window.winfo_width()
2086 self.minheight = self.window.winfo_height()
2087 self.bigminheight = (self.server_frm.winfo_reqheight() +
2088 self.search_frm.winfo_reqheight() +
2089 self.result_lst.winfo_reqheight() +
2090 self.result_frm.winfo_reqheight())
2091 self.bigwidth, self.bigheight = self.minwidth, self.bigminheight
2092 self.expanded = 0
2093 self.window.wm_geometry('%dx%d' % (self.minwidth, self.minheight))
2094 self.window.wm_minsize(self.minwidth, self.minheight)
Martin v. Löwis5b26abb2002-12-28 09:23:09 +00002095 self.window.tk.willdispatch()
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002096
2097 import threading
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00002098 threading.Thread(
2099 target=serve, args=(port, self.ready, self.quit)).start()
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002100
2101 def ready(self, server):
2102 self.server = server
2103 self.title_lbl.config(
2104 text='Python documentation server at\n' + server.url)
2105 self.open_btn.config(state='normal')
2106 self.quit_btn.config(state='normal')
2107
Ka-Ping Yee239432a2001-03-02 02:45:08 +00002108 def open(self, event=None, url=None):
2109 url = url or self.server.url
2110 try:
2111 import webbrowser
2112 webbrowser.open(url)
2113 except ImportError: # pre-webbrowser.py compatibility
Ka-Ping Yeec92cdf72001-03-02 05:54:35 +00002114 if sys.platform == 'win32':
Ka-Ping Yee239432a2001-03-02 02:45:08 +00002115 os.system('start "%s"' % url)
2116 elif sys.platform == 'mac':
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +00002117 try: import ic
Ka-Ping Yee239432a2001-03-02 02:45:08 +00002118 except ImportError: pass
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +00002119 else: ic.launchurl(url)
Ka-Ping Yee239432a2001-03-02 02:45:08 +00002120 else:
2121 rc = os.system('netscape -remote "openURL(%s)" &' % url)
2122 if rc: os.system('netscape "%s" &' % url)
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002123
2124 def quit(self, event=None):
2125 if self.server:
2126 self.server.quit = 1
2127 self.window.quit()
2128
2129 def search(self, event=None):
2130 key = self.search_ent.get()
2131 self.stop_btn.pack(side='right')
2132 self.stop_btn.config(state='normal')
2133 self.search_lbl.config(text='Searching for "%s"...' % key)
2134 self.search_ent.forget()
2135 self.search_lbl.pack(side='left')
2136 self.result_lst.delete(0, 'end')
2137 self.goto_btn.config(state='disabled')
2138 self.expand()
2139
2140 import threading
2141 if self.scanner:
2142 self.scanner.quit = 1
2143 self.scanner = ModuleScanner()
2144 threading.Thread(target=self.scanner.run,
Ka-Ping Yee6dcfa382001-04-12 20:27:31 +00002145 args=(self.update, key, self.done)).start()
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002146
2147 def update(self, path, modname, desc):
2148 if modname[-9:] == '.__init__':
2149 modname = modname[:-9] + ' (package)'
2150 self.result_lst.insert('end',
2151 modname + ' - ' + (desc or '(no description)'))
2152
2153 def stop(self, event=None):
2154 if self.scanner:
2155 self.scanner.quit = 1
2156 self.scanner = None
2157
2158 def done(self):
2159 self.scanner = None
2160 self.search_lbl.config(text='Search for')
2161 self.search_lbl.pack(side='left')
2162 self.search_ent.pack(side='right', fill='x', expand=1)
2163 if sys.platform != 'win32': self.stop_btn.forget()
2164 self.stop_btn.config(state='disabled')
2165
2166 def select(self, event=None):
2167 self.goto_btn.config(state='normal')
2168
2169 def goto(self, event=None):
2170 selection = self.result_lst.curselection()
2171 if selection:
Neal Norwitz9d72bb42007-04-17 08:48:32 +00002172 modname = self.result_lst.get(selection[0]).split()[0]
Ka-Ping Yee239432a2001-03-02 02:45:08 +00002173 self.open(url=self.server.url + modname + '.html')
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002174
2175 def collapse(self):
2176 if not self.expanded: return
2177 self.result_frm.forget()
2178 self.result_scr.forget()
2179 self.result_lst.forget()
2180 self.bigwidth = self.window.winfo_width()
2181 self.bigheight = self.window.winfo_height()
2182 self.window.wm_geometry('%dx%d' % (self.minwidth, self.minheight))
2183 self.window.wm_minsize(self.minwidth, self.minheight)
2184 self.expanded = 0
2185
2186 def expand(self):
2187 if self.expanded: return
2188 self.result_frm.pack(side='bottom', fill='x')
2189 self.result_scr.pack(side='right', fill='y')
2190 self.result_lst.pack(side='top', fill='both', expand=1)
2191 self.window.wm_geometry('%dx%d' % (self.bigwidth, self.bigheight))
2192 self.window.wm_minsize(self.minwidth, self.bigminheight)
2193 self.expanded = 1
2194
2195 def hide(self, event=None):
2196 self.stop()
2197 self.collapse()
2198
2199 import Tkinter
2200 try:
Martin v. Löwise09bd932004-08-22 16:13:26 +00002201 root = Tkinter.Tk()
2202 # Tk will crash if pythonw.exe has an XP .manifest
2203 # file and the root has is not destroyed explicitly.
2204 # If the problem is ever fixed in Tk, the explicit
2205 # destroy can go.
2206 try:
2207 gui = GUI(root)
2208 root.mainloop()
2209 finally:
2210 root.destroy()
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002211 except KeyboardInterrupt:
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002212 pass
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002213
2214# -------------------------------------------------- command-line interface
2215
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00002216def ispath(x):
Neal Norwitz9d72bb42007-04-17 08:48:32 +00002217 return isinstance(x, str) and x.find(os.sep) >= 0
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00002218
Ka-Ping Yee1d384632001-03-01 00:24:32 +00002219def cli():
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002220 """Command-line interface (looks at sys.argv to decide what to do)."""
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002221 import getopt
Guido van Rossum756aa932007-04-07 03:04:01 +00002222 class BadUsage(Exception): pass
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002223
Ka-Ping Yee3bda8792001-03-23 13:17:50 +00002224 # Scripts don't get the current directory in their path by default.
Ka-Ping Yeef78a81b2001-03-27 08:13:42 +00002225 scriptdir = os.path.dirname(sys.argv[0])
2226 if scriptdir in sys.path:
2227 sys.path.remove(scriptdir)
Ka-Ping Yee3bda8792001-03-23 13:17:50 +00002228 sys.path.insert(0, '.')
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002229
Ka-Ping Yee3bda8792001-03-23 13:17:50 +00002230 try:
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002231 opts, args = getopt.getopt(sys.argv[1:], 'gk:p:w')
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002232 writing = 0
2233
2234 for opt, val in opts:
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002235 if opt == '-g':
2236 gui()
2237 return
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002238 if opt == '-k':
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002239 apropos(val)
2240 return
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002241 if opt == '-p':
2242 try:
2243 port = int(val)
2244 except ValueError:
2245 raise BadUsage
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002246 def ready(server):
Guido van Rossumbe19ed72007-02-09 05:37:30 +00002247 print('pydoc server ready at %s' % server.url)
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00002248 def stopped():
Guido van Rossumbe19ed72007-02-09 05:37:30 +00002249 print('pydoc server stopped')
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00002250 serve(port, ready, stopped)
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002251 return
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002252 if opt == '-w':
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002253 writing = 1
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002254
2255 if not args: raise BadUsage
2256 for arg in args:
Ka-Ping Yee45daeb02002-08-11 15:11:33 +00002257 if ispath(arg) and not os.path.exists(arg):
Guido van Rossumbe19ed72007-02-09 05:37:30 +00002258 print('file %r does not exist' % arg)
Ka-Ping Yee45daeb02002-08-11 15:11:33 +00002259 break
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002260 try:
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00002261 if ispath(arg) and os.path.isfile(arg):
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002262 arg = importfile(arg)
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00002263 if writing:
2264 if ispath(arg) and os.path.isdir(arg):
2265 writedocs(arg)
2266 else:
2267 writedoc(arg)
2268 else:
Martin v. Löwisb8c084e2003-06-14 09:03:46 +00002269 help.help(arg)
Guido van Rossumb940e112007-01-10 16:19:56 +00002270 except ErrorDuringImport as value:
Guido van Rossumbe19ed72007-02-09 05:37:30 +00002271 print(value)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002272
2273 except (getopt.error, BadUsage):
Martin v. Löwisf9b08b82003-10-31 13:05:21 +00002274 cmd = os.path.basename(sys.argv[0])
Guido van Rossumbe19ed72007-02-09 05:37:30 +00002275 print("""pydoc - the Python documentation tool
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002276
2277%s <name> ...
2278 Show text documentation on something. <name> may be the name of a
Martin v. Löwisb8c084e2003-06-14 09:03:46 +00002279 Python keyword, topic, function, module, or package, or a dotted
2280 reference to a class or function within a module or module in a
2281 package. If <name> contains a '%s', it is used as the path to a
2282 Python source file to document. If name is 'keywords', 'topics',
2283 or 'modules', a listing of these things is displayed.
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002284
2285%s -k <keyword>
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002286 Search for a keyword in the synopsis lines of all available modules.
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002287
2288%s -p <port>
2289 Start an HTTP server on the given port on the local machine.
2290
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002291%s -g
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +00002292 Pop up a graphical interface for finding and serving documentation.
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002293
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002294%s -w <name> ...
2295 Write out the HTML documentation for a module to a file in the current
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +00002296 directory. If <name> contains a '%s', it is treated as a filename; if
2297 it names a directory, documentation is written for all the contents.
Guido van Rossumbe19ed72007-02-09 05:37:30 +00002298""" % (cmd, os.sep, cmd, cmd, cmd, cmd, os.sep))
Ka-Ping Yee1d384632001-03-01 00:24:32 +00002299
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002300if __name__ == '__main__': cli()