blob: 5d764eb3a78983d485d5e4c61338eef18bac4e7b [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
Georg Brandl1a3284e2007-12-02 09:40:06 +000055import sys, imp, os, re, inspect, builtins, 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:
Guido van Rossume7ba4952007-06-06 23:52:48 +0000274 # Remove the module from sys.modules and re-import to try
275 # and avoid problems with partially loaded modules.
276 # Also remove any submodules because they won't appear
277 # in the newly loaded module's namespace if they're already
278 # in sys.modules.
Ka-Ping Yee9a2dcf82005-11-05 05:04:41 +0000279 subs = [m for m in sys.modules if m.startswith(path + '.')]
280 for key in [path] + subs:
281 # Prevent garbage collection.
282 cache[key] = sys.modules[key]
283 del sys.modules[key]
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000284 module = __import__(path)
285 except:
286 # Did the error occur before or after the module was found?
287 (exc, value, tb) = info = sys.exc_info()
Raymond Hettinger54f02222002-06-01 14:18:47 +0000288 if path in sys.modules:
Fred Drakedb390c12005-10-28 14:39:47 +0000289 # An error occurred while executing the imported module.
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000290 raise ErrorDuringImport(sys.modules[path].__file__, info)
291 elif exc is SyntaxError:
292 # A SyntaxError occurred before we could execute the module.
293 raise ErrorDuringImport(value.filename, info)
294 elif exc is ImportError and \
Neal Norwitz9d72bb42007-04-17 08:48:32 +0000295 str(value).lower().split()[:2] == ['no', 'module']:
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000296 # The module was not found.
297 return None
298 else:
299 # Some other error occurred during the importing process.
300 raise ErrorDuringImport(path, sys.exc_info())
Neal Norwitz9d72bb42007-04-17 08:48:32 +0000301 for part in path.split('.')[1:]:
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000302 try: module = getattr(module, part)
303 except AttributeError: return None
304 return module
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000305
306# ---------------------------------------------------- formatter base class
307
308class Doc:
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000309 def document(self, object, name=None, *args):
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000310 """Generate documentation for an object."""
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000311 args = (object, name) + args
Brett Cannon28a4f0f2003-06-11 23:38:55 +0000312 # 'try' clause is to attempt to handle the possibility that inspect
313 # identifies something in a way that pydoc itself has issues handling;
314 # think 'super' and how it is a descriptor (which raises the exception
315 # by lacking a __name__ attribute) and an instance.
Thomas Wouters0e3f5912006-08-11 14:57:12 +0000316 if inspect.isgetsetdescriptor(object): return self.docdata(*args)
317 if inspect.ismemberdescriptor(object): return self.docdata(*args)
Brett Cannon28a4f0f2003-06-11 23:38:55 +0000318 try:
319 if inspect.ismodule(object): return self.docmodule(*args)
320 if inspect.isclass(object): return self.docclass(*args)
321 if inspect.isroutine(object): return self.docroutine(*args)
322 except AttributeError:
323 pass
Johannes Gijsbers8de645a2004-11-07 19:16:05 +0000324 if isinstance(object, property): return self.docproperty(*args)
Guido van Rossum68468eb2003-02-27 20:14:51 +0000325 return self.docother(*args)
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000326
327 def fail(self, object, name=None, *args):
328 """Raise an exception for unimplemented types."""
329 message = "don't know how to document object%s of type %s" % (
330 name and ' ' + repr(name), type(object).__name__)
Collin Winterce36ad82007-08-30 01:19:48 +0000331 raise TypeError(message)
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000332
Thomas Wouters0e3f5912006-08-11 14:57:12 +0000333 docmodule = docclass = docroutine = docother = docproperty = docdata = fail
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000334
Skip Montanaro4997a692003-09-10 16:47:51 +0000335 def getdocloc(self, object):
336 """Return the location of module docs or None"""
337
338 try:
339 file = inspect.getabsfile(object)
340 except TypeError:
341 file = '(built-in)'
342
343 docloc = os.environ.get("PYTHONDOCS",
344 "http://www.python.org/doc/current/lib")
345 basedir = os.path.join(sys.exec_prefix, "lib",
346 "python"+sys.version[0:3])
347 if (isinstance(object, type(os)) and
348 (object.__name__ in ('errno', 'exceptions', 'gc', 'imp',
349 'marshal', 'posix', 'signal', 'sys',
350 'thread', 'zipimport') or
351 (file.startswith(basedir) and
352 not file.startswith(os.path.join(basedir, 'site-packages'))))):
Skip Montanarof2134842004-06-07 02:40:05 +0000353 htmlfile = "module-%s.html" % object.__name__
Skip Montanaro4997a692003-09-10 16:47:51 +0000354 if docloc.startswith("http://"):
Skip Montanarof2134842004-06-07 02:40:05 +0000355 docloc = "%s/%s" % (docloc.rstrip("/"), htmlfile)
Skip Montanaro4997a692003-09-10 16:47:51 +0000356 else:
Skip Montanarof2134842004-06-07 02:40:05 +0000357 docloc = os.path.join(docloc, htmlfile)
Skip Montanaro4997a692003-09-10 16:47:51 +0000358 else:
359 docloc = None
360 return docloc
361
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000362# -------------------------------------------- HTML documentation generator
363
364class HTMLRepr(Repr):
365 """Class for safely making an HTML representation of a Python object."""
366 def __init__(self):
367 Repr.__init__(self)
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000368 self.maxlist = self.maxtuple = 20
369 self.maxdict = 10
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000370 self.maxstring = self.maxother = 100
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000371
372 def escape(self, text):
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +0000373 return replace(text, '&', '&amp;', '<', '&lt;', '>', '&gt;')
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000374
375 def repr(self, object):
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000376 return Repr.repr(self, object)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000377
378 def repr1(self, x, level):
Skip Montanaro0fe8fce2003-06-27 15:45:41 +0000379 if hasattr(type(x), '__name__'):
Neal Norwitz9d72bb42007-04-17 08:48:32 +0000380 methodname = 'repr_' + '_'.join(type(x).__name__.split())
Skip Montanaro0fe8fce2003-06-27 15:45:41 +0000381 if hasattr(self, methodname):
382 return getattr(self, methodname)(x, level)
383 return self.escape(cram(stripid(repr(x)), self.maxother))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000384
385 def repr_string(self, x, level):
Ka-Ping Yeea2fe1032001-03-02 01:19:14 +0000386 test = cram(x, self.maxstring)
387 testrepr = repr(test)
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +0000388 if '\\' in test and '\\' not in replace(testrepr, r'\\', ''):
Ka-Ping Yeea2fe1032001-03-02 01:19:14 +0000389 # Backslashes are only literal in the string and are never
390 # needed to make any special characters, so show a raw string.
391 return 'r' + testrepr[0] + self.escape(test) + testrepr[0]
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +0000392 return re.sub(r'((\\[\\abfnrtv\'"]|\\[0-9]..|\\x..|\\u....)+)',
Ka-Ping Yeea2fe1032001-03-02 01:19:14 +0000393 r'<font color="#c040c0">\1</font>',
394 self.escape(testrepr))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000395
Skip Montanarodf708782002-03-07 22:58:02 +0000396 repr_str = repr_string
397
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000398 def repr_instance(self, x, level):
399 try:
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000400 return self.escape(cram(stripid(repr(x)), self.maxstring))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000401 except:
402 return self.escape('<%s instance>' % x.__class__.__name__)
403
404 repr_unicode = repr_string
405
406class HTMLDoc(Doc):
407 """Formatter class for HTML documentation."""
408
409 # ------------------------------------------- HTML formatting utilities
410
411 _repr_instance = HTMLRepr()
412 repr = _repr_instance.repr
413 escape = _repr_instance.escape
414
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000415 def page(self, title, contents):
416 """Format an HTML page."""
417 return '''
Tim Peters59ed4482001-10-31 04:20:26 +0000418<!doctype html PUBLIC "-//W3C//DTD HTML 4.0 Transitional//EN">
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000419<html><head><title>Python: %s</title>
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000420</head><body bgcolor="#f0f0f8">
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000421%s
422</body></html>''' % (title, contents)
423
424 def heading(self, title, fgcol, bgcol, extras=''):
425 """Format a page heading."""
426 return '''
Tim Peters59ed4482001-10-31 04:20:26 +0000427<table width="100%%" cellspacing=0 cellpadding=2 border=0 summary="heading">
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000428<tr bgcolor="%s">
Tim Peters2306d242001-09-25 03:18:32 +0000429<td valign=bottom>&nbsp;<br>
430<font color="%s" face="helvetica, arial">&nbsp;<br>%s</font></td
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000431><td align=right valign=bottom
Ka-Ping Yee987ec902001-03-23 13:35:45 +0000432><font color="%s" face="helvetica, arial">%s</font></td></tr></table>
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000433 ''' % (bgcol, fgcol, title, fgcol, extras or '&nbsp;')
434
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000435 def section(self, title, fgcol, bgcol, contents, width=6,
436 prelude='', marginalia=None, gap='&nbsp;'):
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000437 """Format a section with a heading."""
438 if marginalia is None:
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +0000439 marginalia = '<tt>' + '&nbsp;' * width + '</tt>'
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000440 result = '''<p>
Tim Peters59ed4482001-10-31 04:20:26 +0000441<table width="100%%" cellspacing=0 cellpadding=2 border=0 summary="section">
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000442<tr bgcolor="%s">
Tim Peters2306d242001-09-25 03:18:32 +0000443<td colspan=3 valign=bottom>&nbsp;<br>
444<font color="%s" face="helvetica, arial">%s</font></td></tr>
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000445 ''' % (bgcol, fgcol, title)
446 if prelude:
447 result = result + '''
Ka-Ping Yee987ec902001-03-23 13:35:45 +0000448<tr bgcolor="%s"><td rowspan=2>%s</td>
449<td colspan=2>%s</td></tr>
450<tr><td>%s</td>''' % (bgcol, marginalia, prelude, gap)
451 else:
452 result = result + '''
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000453<tr><td bgcolor="%s">%s</td><td>%s</td>''' % (bgcol, marginalia, gap)
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000454
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +0000455 return result + '\n<td width="100%%">%s</td></tr></table>' % contents
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000456
457 def bigsection(self, title, *args):
458 """Format a section with a big heading."""
459 title = '<big><strong>%s</strong></big>' % title
Guido van Rossum68468eb2003-02-27 20:14:51 +0000460 return self.section(title, *args)
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000461
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000462 def preformat(self, text):
463 """Format literal preformatted text."""
Neal Norwitz9d72bb42007-04-17 08:48:32 +0000464 text = self.escape(text.expandtabs())
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +0000465 return replace(text, '\n\n', '\n \n', '\n\n', '\n \n',
466 ' ', '&nbsp;', '\n', '<br>\n')
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000467
468 def multicolumn(self, list, format, cols=4):
469 """Format a list of items into a multi-column list."""
470 result = ''
Guido van Rossumc1f779c2007-07-03 08:25:58 +0000471 rows = (len(list)+cols-1)//cols
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000472 for col in range(cols):
Guido van Rossumc1f779c2007-07-03 08:25:58 +0000473 result = result + '<td width="%d%%" valign=top>' % (100//cols)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000474 for i in range(rows*col, rows*col+rows):
475 if i < len(list):
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000476 result = result + format(list[i]) + '<br>\n'
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000477 result = result + '</td>'
Tim Peters59ed4482001-10-31 04:20:26 +0000478 return '<table width="100%%" summary="list"><tr>%s</tr></table>' % result
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000479
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000480 def grey(self, text): return '<font color="#909090">%s</font>' % text
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000481
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000482 def namelink(self, name, *dicts):
483 """Make a link for an identifier, given name-to-URL mappings."""
484 for dict in dicts:
Raymond Hettinger54f02222002-06-01 14:18:47 +0000485 if name in dict:
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000486 return '<a href="%s">%s</a>' % (dict[name], name)
487 return name
488
Ka-Ping Yeeb7a48302001-04-12 20:39:14 +0000489 def classlink(self, object, modname):
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000490 """Make a link for a class."""
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000491 name, module = object.__name__, sys.modules.get(object.__module__)
492 if hasattr(module, name) and getattr(module, name) is object:
Ka-Ping Yeeb7a48302001-04-12 20:39:14 +0000493 return '<a href="%s.html#%s">%s</a>' % (
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000494 module.__name__, name, classname(object, modname))
495 return classname(object, modname)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000496
497 def modulelink(self, object):
498 """Make a link for a module."""
499 return '<a href="%s.html">%s</a>' % (object.__name__, object.__name__)
500
Guido van Rossum1bc535d2007-05-15 18:46:22 +0000501 def modpkglink(self, modpkginfo):
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000502 """Make a link for a module or package to display in an index."""
Guido van Rossum1bc535d2007-05-15 18:46:22 +0000503 name, path, ispackage, shadowed = modpkginfo
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000504 if shadowed:
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000505 return self.grey(name)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000506 if path:
507 url = '%s.%s.html' % (path, name)
508 else:
509 url = '%s.html' % name
510 if ispackage:
511 text = '<strong>%s</strong>&nbsp;(package)' % name
512 else:
513 text = name
514 return '<a href="%s">%s</a>' % (url, text)
515
516 def markup(self, text, escape=None, funcs={}, classes={}, methods={}):
517 """Mark up some plain text, given a context of symbols to look for.
518 Each context dictionary maps object names to anchor names."""
519 escape = escape or self.escape
520 results = []
521 here = 0
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000522 pattern = re.compile(r'\b((http|ftp)://\S+[\w/]|'
523 r'RFC[- ]?(\d+)|'
Ka-Ping Yeef78a81b2001-03-27 08:13:42 +0000524 r'PEP[- ]?(\d+)|'
Neil Schemenauerd69711c2002-03-24 23:02:07 +0000525 r'(self\.)?(\w+))')
Guido van Rossum8ca162f2002-04-07 06:36:23 +0000526 while True:
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000527 match = pattern.search(text, here)
528 if not match: break
529 start, end = match.span()
530 results.append(escape(text[here:start]))
531
Ka-Ping Yeef78a81b2001-03-27 08:13:42 +0000532 all, scheme, rfc, pep, selfdot, name = match.groups()
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000533 if scheme:
Neil Schemenauercddc1a02002-03-24 23:11:21 +0000534 url = escape(all).replace('"', '&quot;')
535 results.append('<a href="%s">%s</a>' % (url, url))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000536 elif rfc:
Ka-Ping Yeef78a81b2001-03-27 08:13:42 +0000537 url = 'http://www.rfc-editor.org/rfc/rfc%d.txt' % int(rfc)
538 results.append('<a href="%s">%s</a>' % (url, escape(all)))
539 elif pep:
540 url = 'http://www.python.org/peps/pep-%04d.html' % int(pep)
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000541 results.append('<a href="%s">%s</a>' % (url, escape(all)))
542 elif text[end:end+1] == '(':
543 results.append(self.namelink(name, methods, funcs, classes))
544 elif selfdot:
545 results.append('self.<strong>%s</strong>' % name)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000546 else:
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000547 results.append(self.namelink(name, classes))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000548 here = end
549 results.append(escape(text[here:]))
Neal Norwitz9d72bb42007-04-17 08:48:32 +0000550 return ''.join(results)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000551
552 # ---------------------------------------------- type-specific routines
553
Ka-Ping Yeeb7a48302001-04-12 20:39:14 +0000554 def formattree(self, tree, modname, parent=None):
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000555 """Produce HTML for a class tree as given by inspect.getclasstree()."""
556 result = ''
557 for entry in tree:
558 if type(entry) is type(()):
559 c, bases = entry
Tim Peters2306d242001-09-25 03:18:32 +0000560 result = result + '<dt><font face="helvetica, arial">'
Ka-Ping Yeeb7a48302001-04-12 20:39:14 +0000561 result = result + self.classlink(c, modname)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000562 if bases and bases != (parent,):
563 parents = []
564 for base in bases:
Ka-Ping Yeeb7a48302001-04-12 20:39:14 +0000565 parents.append(self.classlink(base, modname))
Neal Norwitz9d72bb42007-04-17 08:48:32 +0000566 result = result + '(' + ', '.join(parents) + ')'
Tim Peters2306d242001-09-25 03:18:32 +0000567 result = result + '\n</font></dt>'
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000568 elif type(entry) is type([]):
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000569 result = result + '<dd>\n%s</dd>\n' % self.formattree(
Ka-Ping Yeeb7a48302001-04-12 20:39:14 +0000570 entry, modname, c)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000571 return '<dl>\n%s</dl>\n' % result
572
Tim Peters8dd7ade2001-10-18 19:56:17 +0000573 def docmodule(self, object, name=None, mod=None, *ignored):
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000574 """Produce HTML documentation for a module object."""
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000575 name = object.__name__ # ignore the passed-in name
Skip Montanaroa5616d22004-06-11 04:46:12 +0000576 try:
577 all = object.__all__
578 except AttributeError:
579 all = None
Neal Norwitz9d72bb42007-04-17 08:48:32 +0000580 parts = name.split('.')
Ka-Ping Yee66efbc72001-03-01 13:55:20 +0000581 links = []
582 for i in range(len(parts)-1):
583 links.append(
584 '<a href="%s.html"><font color="#ffffff">%s</font></a>' %
Neal Norwitz9d72bb42007-04-17 08:48:32 +0000585 ('.'.join(parts[:i+1]), parts[i]))
586 linkedname = '.'.join(links + parts[-1:])
Ka-Ping Yee66efbc72001-03-01 13:55:20 +0000587 head = '<big><big><strong>%s</strong></big></big>' % linkedname
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000588 try:
Ka-Ping Yee239432a2001-03-02 02:45:08 +0000589 path = inspect.getabsfile(object)
Ka-Ping Yee6191a232001-04-13 15:00:27 +0000590 url = path
591 if sys.platform == 'win32':
592 import nturl2path
593 url = nturl2path.pathname2url(path)
594 filelink = '<a href="file:%s">%s</a>' % (url, path)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000595 except TypeError:
596 filelink = '(built-in)'
Ka-Ping Yee6f3f9a42001-02-27 22:42:36 +0000597 info = []
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000598 if hasattr(object, '__version__'):
Ka-Ping Yee6f3f9a42001-02-27 22:42:36 +0000599 version = str(object.__version__)
Ka-Ping Yee40c49912001-02-27 22:46:01 +0000600 if version[:11] == '$' + 'Revision: ' and version[-1:] == '$':
Neal Norwitz9d72bb42007-04-17 08:48:32 +0000601 version = version[11:-1].strip()
Ka-Ping Yee1d384632001-03-01 00:24:32 +0000602 info.append('version %s' % self.escape(version))
Ka-Ping Yee6f3f9a42001-02-27 22:42:36 +0000603 if hasattr(object, '__date__'):
604 info.append(self.escape(str(object.__date__)))
605 if info:
Neal Norwitz9d72bb42007-04-17 08:48:32 +0000606 head = head + ' (%s)' % ', '.join(info)
Skip Montanaro4997a692003-09-10 16:47:51 +0000607 docloc = self.getdocloc(object)
608 if docloc is not None:
609 docloc = '<br><a href="%(docloc)s">Module Docs</a>' % locals()
610 else:
611 docloc = ''
Ka-Ping Yee66efbc72001-03-01 13:55:20 +0000612 result = self.heading(
Skip Montanaro4997a692003-09-10 16:47:51 +0000613 head, '#ffffff', '#7799ee',
614 '<a href=".">index</a><br>' + filelink + docloc)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000615
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000616 modules = inspect.getmembers(object, inspect.ismodule)
617
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000618 classes, cdict = [], {}
619 for key, value in inspect.getmembers(object, inspect.isclass):
Johannes Gijsbers4c11f602004-08-30 14:13:04 +0000620 # if __all__ exists, believe it. Otherwise use old heuristic.
621 if (all is not None or
622 (inspect.getmodule(value) or object) is object):
Skip Montanaroa5616d22004-06-11 04:46:12 +0000623 if visiblename(key, all):
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000624 classes.append((key, value))
625 cdict[key] = cdict[value] = '#' + key
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000626 for key, value in classes:
627 for base in value.__bases__:
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000628 key, modname = base.__name__, base.__module__
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000629 module = sys.modules.get(modname)
630 if modname != name and module and hasattr(module, key):
631 if getattr(module, key) is base:
Raymond Hettinger54f02222002-06-01 14:18:47 +0000632 if not key in cdict:
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000633 cdict[key] = cdict[base] = modname + '.html#' + key
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000634 funcs, fdict = [], {}
635 for key, value in inspect.getmembers(object, inspect.isroutine):
Johannes Gijsbers4c11f602004-08-30 14:13:04 +0000636 # if __all__ exists, believe it. Otherwise use old heuristic.
637 if (all is not None or
638 inspect.isbuiltin(value) or inspect.getmodule(value) is object):
Skip Montanaroa5616d22004-06-11 04:46:12 +0000639 if visiblename(key, all):
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000640 funcs.append((key, value))
641 fdict[key] = '#-' + key
642 if inspect.isfunction(value): fdict[value] = fdict[key]
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000643 data = []
644 for key, value in inspect.getmembers(object, isdata):
Skip Montanaroa5616d22004-06-11 04:46:12 +0000645 if visiblename(key, all):
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000646 data.append((key, value))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000647
648 doc = self.markup(getdoc(object), self.preformat, fdict, cdict)
649 doc = doc and '<tt>%s</tt>' % doc
Tim Peters2306d242001-09-25 03:18:32 +0000650 result = result + '<p>%s</p>\n' % doc
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000651
652 if hasattr(object, '__path__'):
653 modpkgs = []
Thomas Wouters49fd7fa2006-04-21 10:40:58 +0000654 for importer, modname, ispkg in pkgutil.iter_modules(object.__path__):
655 modpkgs.append((modname, name, ispkg, 0))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000656 modpkgs.sort()
657 contents = self.multicolumn(modpkgs, self.modpkglink)
658 result = result + self.bigsection(
659 'Package Contents', '#ffffff', '#aa55cc', contents)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000660 elif modules:
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000661 contents = self.multicolumn(
Guido van Rossum1bc535d2007-05-15 18:46:22 +0000662 modules, lambda t: self.modulelink(t[1]))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000663 result = result + self.bigsection(
664 'Modules', '#fffff', '#aa55cc', contents)
665
666 if classes:
Guido van Rossum1bc535d2007-05-15 18:46:22 +0000667 classlist = [value for (key, value) in classes]
Ka-Ping Yeeb7a48302001-04-12 20:39:14 +0000668 contents = [
669 self.formattree(inspect.getclasstree(classlist, 1), name)]
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000670 for key, value in classes:
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +0000671 contents.append(self.document(value, key, name, fdict, cdict))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000672 result = result + self.bigsection(
Neal Norwitz9d72bb42007-04-17 08:48:32 +0000673 'Classes', '#ffffff', '#ee77aa', ' '.join(contents))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000674 if funcs:
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000675 contents = []
676 for key, value in funcs:
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +0000677 contents.append(self.document(value, key, name, fdict, cdict))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000678 result = result + self.bigsection(
Neal Norwitz9d72bb42007-04-17 08:48:32 +0000679 'Functions', '#ffffff', '#eeaa77', ' '.join(contents))
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000680 if data:
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000681 contents = []
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000682 for key, value in data:
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000683 contents.append(self.document(value, key))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000684 result = result + self.bigsection(
Neal Norwitz9d72bb42007-04-17 08:48:32 +0000685 'Data', '#ffffff', '#55aa55', '<br>\n'.join(contents))
Ka-Ping Yee6f3f9a42001-02-27 22:42:36 +0000686 if hasattr(object, '__author__'):
687 contents = self.markup(str(object.__author__), self.preformat)
688 result = result + self.bigsection(
689 'Author', '#ffffff', '#7799ee', contents)
Ka-Ping Yee6f3f9a42001-02-27 22:42:36 +0000690 if hasattr(object, '__credits__'):
691 contents = self.markup(str(object.__credits__), self.preformat)
692 result = result + self.bigsection(
693 'Credits', '#ffffff', '#7799ee', contents)
694
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000695 return result
696
Tim Peters8dd7ade2001-10-18 19:56:17 +0000697 def docclass(self, object, name=None, mod=None, funcs={}, classes={},
698 *ignored):
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000699 """Produce HTML documentation for a class object."""
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000700 realname = object.__name__
701 name = name or realname
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000702 bases = object.__bases__
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000703
Tim Petersb47879b2001-09-24 04:47:19 +0000704 contents = []
705 push = contents.append
706
Tim Petersfa26f7c2001-09-24 08:05:11 +0000707 # Cute little class to pump out a horizontal rule between sections.
708 class HorizontalRule:
709 def __init__(self):
710 self.needone = 0
711 def maybe(self):
712 if self.needone:
713 push('<hr>\n')
714 self.needone = 1
715 hr = HorizontalRule()
716
Tim Petersc86f6ca2001-09-26 21:31:51 +0000717 # List the mro, if non-trivial.
Raymond Hettinger756b3f32004-01-29 06:37:52 +0000718 mro = deque(inspect.getmro(object))
Tim Petersc86f6ca2001-09-26 21:31:51 +0000719 if len(mro) > 2:
720 hr.maybe()
721 push('<dl><dt>Method resolution order:</dt>\n')
722 for base in mro:
723 push('<dd>%s</dd>\n' % self.classlink(base,
724 object.__module__))
725 push('</dl>\n')
726
Tim Petersb47879b2001-09-24 04:47:19 +0000727 def spill(msg, attrs, predicate):
Tim Petersfa26f7c2001-09-24 08:05:11 +0000728 ok, attrs = _split_list(attrs, predicate)
Tim Petersb47879b2001-09-24 04:47:19 +0000729 if ok:
Tim Petersfa26f7c2001-09-24 08:05:11 +0000730 hr.maybe()
Tim Petersb47879b2001-09-24 04:47:19 +0000731 push(msg)
732 for name, kind, homecls, value in ok:
733 push(self.document(getattr(object, name), name, mod,
734 funcs, classes, mdict, object))
735 push('\n')
736 return attrs
737
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +0000738 def spilldescriptors(msg, attrs, predicate):
Tim Petersfa26f7c2001-09-24 08:05:11 +0000739 ok, attrs = _split_list(attrs, predicate)
Tim Petersb47879b2001-09-24 04:47:19 +0000740 if ok:
Tim Petersfa26f7c2001-09-24 08:05:11 +0000741 hr.maybe()
Tim Petersb47879b2001-09-24 04:47:19 +0000742 push(msg)
743 for name, kind, homecls, value in ok:
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +0000744 push(self._docdescriptor(name, value, mod))
Tim Petersb47879b2001-09-24 04:47:19 +0000745 return attrs
746
Tim Petersfa26f7c2001-09-24 08:05:11 +0000747 def spilldata(msg, attrs, predicate):
748 ok, attrs = _split_list(attrs, predicate)
Tim Petersb47879b2001-09-24 04:47:19 +0000749 if ok:
Tim Petersfa26f7c2001-09-24 08:05:11 +0000750 hr.maybe()
Tim Petersb47879b2001-09-24 04:47:19 +0000751 push(msg)
752 for name, kind, homecls, value in ok:
753 base = self.docother(getattr(object, name), name, mod)
Guido van Rossumd59da4b2007-05-22 18:11:13 +0000754 if hasattr(value, '__call__') or inspect.isdatadescriptor(value):
Guido van Rossum5e355b22002-05-21 20:56:15 +0000755 doc = getattr(value, "__doc__", None)
756 else:
757 doc = None
Tim Petersb47879b2001-09-24 04:47:19 +0000758 if doc is None:
759 push('<dl><dt>%s</dl>\n' % base)
760 else:
761 doc = self.markup(getdoc(value), self.preformat,
762 funcs, classes, mdict)
Tim Peters2306d242001-09-25 03:18:32 +0000763 doc = '<dd><tt>%s</tt>' % doc
Tim Petersb47879b2001-09-24 04:47:19 +0000764 push('<dl><dt>%s%s</dl>\n' % (base, doc))
765 push('\n')
766 return attrs
767
Guido van Rossum1bc535d2007-05-15 18:46:22 +0000768 attrs = [(name, kind, cls, value)
769 for name, kind, cls, value in classify_class_attrs(object)
770 if visiblename(name)]
771
Tim Petersb47879b2001-09-24 04:47:19 +0000772 mdict = {}
773 for key, kind, homecls, value in attrs:
774 mdict[key] = anchor = '#' + name + '-' + key
775 value = getattr(object, key)
776 try:
777 # The value may not be hashable (e.g., a data attr with
778 # a dict or list value).
779 mdict[value] = anchor
780 except TypeError:
781 pass
782
Tim Petersfa26f7c2001-09-24 08:05:11 +0000783 while attrs:
Tim Peters351e3622001-09-27 03:29:51 +0000784 if mro:
Raymond Hettinger756b3f32004-01-29 06:37:52 +0000785 thisclass = mro.popleft()
Tim Peters351e3622001-09-27 03:29:51 +0000786 else:
787 thisclass = attrs[0][2]
Tim Petersfa26f7c2001-09-24 08:05:11 +0000788 attrs, inherited = _split_list(attrs, lambda t: t[2] is thisclass)
789
Georg Brandl1a3284e2007-12-02 09:40:06 +0000790 if thisclass is builtins.object:
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000791 attrs = inherited
792 continue
793 elif thisclass is object:
794 tag = 'defined here'
Tim Petersb47879b2001-09-24 04:47:19 +0000795 else:
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000796 tag = 'inherited from %s' % self.classlink(thisclass,
797 object.__module__)
Tim Petersb47879b2001-09-24 04:47:19 +0000798 tag += ':<br>\n'
799
800 # Sort attrs by name.
Thomas Wouters49fd7fa2006-04-21 10:40:58 +0000801 try:
802 attrs.sort(key=lambda t: t[0])
803 except TypeError:
804 attrs.sort(lambda t1, t2: cmp(t1[0], t2[0])) # 2.3 compat
Tim Petersb47879b2001-09-24 04:47:19 +0000805
806 # Pump out the attrs, segregated by kind.
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000807 attrs = spill('Methods %s' % tag, attrs,
Tim Petersb47879b2001-09-24 04:47:19 +0000808 lambda t: t[1] == 'method')
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000809 attrs = spill('Class methods %s' % tag, attrs,
Tim Petersb47879b2001-09-24 04:47:19 +0000810 lambda t: t[1] == 'class method')
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000811 attrs = spill('Static methods %s' % tag, attrs,
Tim Petersb47879b2001-09-24 04:47:19 +0000812 lambda t: t[1] == 'static method')
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +0000813 attrs = spilldescriptors('Data descriptors %s' % tag, attrs,
814 lambda t: t[1] == 'data descriptor')
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000815 attrs = spilldata('Data and other attributes %s' % tag, attrs,
Tim Petersfa26f7c2001-09-24 08:05:11 +0000816 lambda t: t[1] == 'data')
Tim Petersb47879b2001-09-24 04:47:19 +0000817 assert attrs == []
Tim Peters351e3622001-09-27 03:29:51 +0000818 attrs = inherited
Tim Petersb47879b2001-09-24 04:47:19 +0000819
820 contents = ''.join(contents)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000821
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000822 if name == realname:
823 title = '<a name="%s">class <strong>%s</strong></a>' % (
824 name, realname)
825 else:
826 title = '<strong>%s</strong> = <a name="%s">class %s</a>' % (
827 name, name, realname)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000828 if bases:
829 parents = []
830 for base in bases:
Ka-Ping Yeeb7a48302001-04-12 20:39:14 +0000831 parents.append(self.classlink(base, object.__module__))
Neal Norwitz9d72bb42007-04-17 08:48:32 +0000832 title = title + '(%s)' % ', '.join(parents)
Tim Peters2306d242001-09-25 03:18:32 +0000833 doc = self.markup(getdoc(object), self.preformat, funcs, classes, mdict)
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000834 doc = doc and '<tt>%s<br>&nbsp;</tt>' % doc
Tim Petersc86f6ca2001-09-26 21:31:51 +0000835
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000836 return self.section(title, '#000000', '#ffc8d8', contents, 3, doc)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000837
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000838 def formatvalue(self, object):
839 """Format an argument default value as text."""
Tim Peters2306d242001-09-25 03:18:32 +0000840 return self.grey('=' + self.repr(object))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000841
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +0000842 def docroutine(self, object, name=None, mod=None,
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000843 funcs={}, classes={}, methods={}, cl=None):
Ka-Ping Yee66efbc72001-03-01 13:55:20 +0000844 """Produce HTML documentation for a function or method object."""
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000845 realname = object.__name__
846 name = name or realname
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000847 anchor = (cl and cl.__name__ or '') + '-' + name
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000848 note = ''
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000849 skipdocs = 0
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000850 if inspect.ismethod(object):
Christian Heimesff737952007-11-27 10:40:20 +0000851 imclass = object.__self__.__class__
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000852 if cl:
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +0000853 if imclass is not cl:
Ka-Ping Yeeb7a48302001-04-12 20:39:14 +0000854 note = ' from ' + self.classlink(imclass, mod)
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000855 else:
Christian Heimesff737952007-11-27 10:40:20 +0000856 if object.__self__ is not None:
Ka-Ping Yeeb7a48302001-04-12 20:39:14 +0000857 note = ' method of %s instance' % self.classlink(
Christian Heimesff737952007-11-27 10:40:20 +0000858 object.__self__.__class__, mod)
Ka-Ping Yeeb7a48302001-04-12 20:39:14 +0000859 else:
860 note = ' unbound %s method' % self.classlink(imclass,mod)
Christian Heimesff737952007-11-27 10:40:20 +0000861 object = object.__func__
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000862
863 if name == realname:
864 title = '<a name="%s"><strong>%s</strong></a>' % (anchor, realname)
865 else:
Raymond Hettinger54f02222002-06-01 14:18:47 +0000866 if (cl and realname in cl.__dict__ and
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000867 cl.__dict__[realname] is object):
Ka-Ping Yeee280c062001-03-23 14:05:53 +0000868 reallink = '<a href="#%s">%s</a>' % (
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000869 cl.__name__ + '-' + realname, realname)
870 skipdocs = 1
871 else:
872 reallink = realname
873 title = '<a name="%s"><strong>%s</strong></a> = %s' % (
874 anchor, name, reallink)
Tim Peters4bcfa312001-09-20 06:08:24 +0000875 if inspect.isfunction(object):
Guido van Rossum2e65f892007-02-28 22:03:49 +0000876 args, varargs, kwonlyargs, kwdefaults, varkw, defaults, ann = \
877 inspect.getfullargspec(object)
Ka-Ping Yee66efbc72001-03-01 13:55:20 +0000878 argspec = inspect.formatargspec(
Guido van Rossum2e65f892007-02-28 22:03:49 +0000879 args, varargs, kwonlyargs, kwdefaults, varkw, defaults, ann,
880 formatvalue=self.formatvalue,
881 formatannotation=inspect.formatannotationrelativeto(object))
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000882 if realname == '<lambda>':
Tim Peters59ed4482001-10-31 04:20:26 +0000883 title = '<strong>%s</strong> <em>lambda</em> ' % name
Guido van Rossum2e65f892007-02-28 22:03:49 +0000884 # XXX lambda's won't usually have func_annotations['return']
885 # since the syntax doesn't support but it is possible.
886 # So removing parentheses isn't truly safe.
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000887 argspec = argspec[1:-1] # remove parentheses
Tim Peters4bcfa312001-09-20 06:08:24 +0000888 else:
889 argspec = '(...)'
Ka-Ping Yee66efbc72001-03-01 13:55:20 +0000890
Tim Peters2306d242001-09-25 03:18:32 +0000891 decl = title + argspec + (note and self.grey(
892 '<font face="helvetica, arial">%s</font>' % note))
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000893
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000894 if skipdocs:
Tim Peters2306d242001-09-25 03:18:32 +0000895 return '<dl><dt>%s</dt></dl>\n' % decl
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000896 else:
897 doc = self.markup(
898 getdoc(object), self.preformat, funcs, classes, methods)
Tim Peters2306d242001-09-25 03:18:32 +0000899 doc = doc and '<dd><tt>%s</tt></dd>' % doc
900 return '<dl><dt>%s</dt>%s</dl>\n' % (decl, doc)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000901
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +0000902 def _docdescriptor(self, name, value, mod):
Johannes Gijsbers8de645a2004-11-07 19:16:05 +0000903 results = []
904 push = results.append
905
906 if name:
907 push('<dl><dt><strong>%s</strong></dt>\n' % name)
908 if value.__doc__ is not None:
Ka-Ping Yeebba6acc2005-02-19 22:58:26 +0000909 doc = self.markup(getdoc(value), self.preformat)
Johannes Gijsbers8de645a2004-11-07 19:16:05 +0000910 push('<dd><tt>%s</tt></dd>\n' % doc)
Johannes Gijsbers8de645a2004-11-07 19:16:05 +0000911 push('</dl>\n')
912
913 return ''.join(results)
914
915 def docproperty(self, object, name=None, mod=None, cl=None):
916 """Produce html documentation for a property."""
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +0000917 return self._docdescriptor(name, object, mod)
Johannes Gijsbers8de645a2004-11-07 19:16:05 +0000918
Tim Peters8dd7ade2001-10-18 19:56:17 +0000919 def docother(self, object, name=None, mod=None, *ignored):
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000920 """Produce HTML documentation for a data object."""
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +0000921 lhs = name and '<strong>%s</strong> = ' % name or ''
922 return lhs + self.repr(object)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000923
Thomas Wouters0e3f5912006-08-11 14:57:12 +0000924 def docdata(self, object, name=None, mod=None, cl=None):
925 """Produce html documentation for a data descriptor."""
926 return self._docdescriptor(name, object, mod)
927
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000928 def index(self, dir, shadowed=None):
929 """Generate an HTML index for a directory of modules."""
930 modpkgs = []
931 if shadowed is None: shadowed = {}
Thomas Wouters49fd7fa2006-04-21 10:40:58 +0000932 for importer, name, ispkg in pkgutil.iter_modules([dir]):
933 modpkgs.append((name, '', ispkg, name in shadowed))
934 shadowed[name] = 1
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000935
936 modpkgs.sort()
937 contents = self.multicolumn(modpkgs, self.modpkglink)
938 return self.bigsection(dir, '#ffffff', '#ee77aa', contents)
939
940# -------------------------------------------- text documentation generator
941
942class TextRepr(Repr):
943 """Class for safely making a text representation of a Python object."""
944 def __init__(self):
945 Repr.__init__(self)
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000946 self.maxlist = self.maxtuple = 20
947 self.maxdict = 10
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000948 self.maxstring = self.maxother = 100
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000949
950 def repr1(self, x, level):
Skip Montanaro0fe8fce2003-06-27 15:45:41 +0000951 if hasattr(type(x), '__name__'):
Neal Norwitz9d72bb42007-04-17 08:48:32 +0000952 methodname = 'repr_' + '_'.join(type(x).__name__.split())
Skip Montanaro0fe8fce2003-06-27 15:45:41 +0000953 if hasattr(self, methodname):
954 return getattr(self, methodname)(x, level)
955 return cram(stripid(repr(x)), self.maxother)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000956
Ka-Ping Yeea2fe1032001-03-02 01:19:14 +0000957 def repr_string(self, x, level):
958 test = cram(x, self.maxstring)
959 testrepr = repr(test)
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +0000960 if '\\' in test and '\\' not in replace(testrepr, r'\\', ''):
Ka-Ping Yeea2fe1032001-03-02 01:19:14 +0000961 # Backslashes are only literal in the string and are never
962 # needed to make any special characters, so show a raw string.
963 return 'r' + testrepr[0] + test + testrepr[0]
964 return testrepr
965
Skip Montanarodf708782002-03-07 22:58:02 +0000966 repr_str = repr_string
967
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000968 def repr_instance(self, x, level):
969 try:
Ka-Ping Yee1d384632001-03-01 00:24:32 +0000970 return cram(stripid(repr(x)), self.maxstring)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000971 except:
972 return '<%s instance>' % x.__class__.__name__
973
974class TextDoc(Doc):
975 """Formatter class for text documentation."""
976
977 # ------------------------------------------- text formatting utilities
978
979 _repr_instance = TextRepr()
980 repr = _repr_instance.repr
981
982 def bold(self, text):
983 """Format a string in bold by overstriking."""
Neal Norwitz9d72bb42007-04-17 08:48:32 +0000984 return ''.join(map(lambda ch: ch + '\b' + ch, text))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000985
986 def indent(self, text, prefix=' '):
987 """Indent text by prepending a given prefix to each line."""
988 if not text: return ''
Collin Winter72e110c2007-07-17 00:27:30 +0000989 lines = [prefix + line for line in text.split('\n')]
Neal Norwitz9d72bb42007-04-17 08:48:32 +0000990 if lines: lines[-1] = lines[-1].rstrip()
991 return '\n'.join(lines)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000992
993 def section(self, title, contents):
994 """Format a section with a given heading."""
Neal Norwitz9d72bb42007-04-17 08:48:32 +0000995 clean_contents = self.indent(contents).rstrip()
996 return self.bold(title) + '\n' + clean_contents + '\n\n'
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000997
998 # ---------------------------------------------- type-specific routines
999
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001000 def formattree(self, tree, modname, parent=None, prefix=''):
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001001 """Render in text a class tree as returned by inspect.getclasstree()."""
1002 result = ''
1003 for entry in tree:
1004 if type(entry) is type(()):
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001005 c, bases = entry
1006 result = result + prefix + classname(c, modname)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001007 if bases and bases != (parent,):
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001008 parents = map(lambda c, m=modname: classname(c, m), bases)
Neal Norwitz9d72bb42007-04-17 08:48:32 +00001009 result = result + '(%s)' % ', '.join(parents)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001010 result = result + '\n'
1011 elif type(entry) is type([]):
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001012 result = result + self.formattree(
1013 entry, modname, c, prefix + ' ')
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001014 return result
1015
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001016 def docmodule(self, object, name=None, mod=None):
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001017 """Produce text documentation for a given module object."""
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001018 name = object.__name__ # ignore the passed-in name
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +00001019 synop, desc = splitdoc(getdoc(object))
1020 result = self.section('NAME', name + (synop and ' - ' + synop))
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001021
1022 try:
Skip Montanaroa5616d22004-06-11 04:46:12 +00001023 all = object.__all__
1024 except AttributeError:
1025 all = None
1026
1027 try:
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001028 file = inspect.getabsfile(object)
1029 except TypeError:
1030 file = '(built-in)'
Ka-Ping Yee239432a2001-03-02 02:45:08 +00001031 result = result + self.section('FILE', file)
Skip Montanaro4997a692003-09-10 16:47:51 +00001032
1033 docloc = self.getdocloc(object)
1034 if docloc is not None:
1035 result = result + self.section('MODULE DOCS', docloc)
1036
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +00001037 if desc:
1038 result = result + self.section('DESCRIPTION', desc)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001039
1040 classes = []
1041 for key, value in inspect.getmembers(object, inspect.isclass):
Johannes Gijsbers4c11f602004-08-30 14:13:04 +00001042 # if __all__ exists, believe it. Otherwise use old heuristic.
1043 if (all is not None
1044 or (inspect.getmodule(value) or object) is object):
Skip Montanaroa5616d22004-06-11 04:46:12 +00001045 if visiblename(key, all):
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +00001046 classes.append((key, value))
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001047 funcs = []
1048 for key, value in inspect.getmembers(object, inspect.isroutine):
Johannes Gijsbers4c11f602004-08-30 14:13:04 +00001049 # if __all__ exists, believe it. Otherwise use old heuristic.
1050 if (all is not None or
1051 inspect.isbuiltin(value) or inspect.getmodule(value) is object):
Skip Montanaroa5616d22004-06-11 04:46:12 +00001052 if visiblename(key, all):
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +00001053 funcs.append((key, value))
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001054 data = []
1055 for key, value in inspect.getmembers(object, isdata):
Skip Montanaroa5616d22004-06-11 04:46:12 +00001056 if visiblename(key, all):
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001057 data.append((key, value))
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001058
1059 if hasattr(object, '__path__'):
1060 modpkgs = []
Thomas Wouters49fd7fa2006-04-21 10:40:58 +00001061 for importer, modname, ispkg in pkgutil.iter_modules(object.__path__):
1062 if ispkg:
1063 modpkgs.append(modname + ' (package)')
1064 else:
1065 modpkgs.append(modname)
1066
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001067 modpkgs.sort()
1068 result = result + self.section(
Neal Norwitz9d72bb42007-04-17 08:48:32 +00001069 'PACKAGE CONTENTS', '\n'.join(modpkgs))
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001070
1071 if classes:
Guido van Rossum1bc535d2007-05-15 18:46:22 +00001072 classlist = [value for key, value in classes]
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001073 contents = [self.formattree(
1074 inspect.getclasstree(classlist, 1), name)]
1075 for key, value in classes:
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001076 contents.append(self.document(value, key, name))
Neal Norwitz9d72bb42007-04-17 08:48:32 +00001077 result = result + self.section('CLASSES', '\n'.join(contents))
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001078
1079 if funcs:
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001080 contents = []
1081 for key, value in funcs:
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001082 contents.append(self.document(value, key, name))
Neal Norwitz9d72bb42007-04-17 08:48:32 +00001083 result = result + self.section('FUNCTIONS', '\n'.join(contents))
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001084
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001085 if data:
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001086 contents = []
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001087 for key, value in data:
Georg Brandl8b813db2005-10-01 16:32:31 +00001088 contents.append(self.docother(value, key, name, maxlen=70))
Neal Norwitz9d72bb42007-04-17 08:48:32 +00001089 result = result + self.section('DATA', '\n'.join(contents))
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001090
1091 if hasattr(object, '__version__'):
1092 version = str(object.__version__)
Ka-Ping Yee1d384632001-03-01 00:24:32 +00001093 if version[:11] == '$' + 'Revision: ' and version[-1:] == '$':
Neal Norwitz9d72bb42007-04-17 08:48:32 +00001094 version = version[11:-1].strip()
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001095 result = result + self.section('VERSION', version)
Ka-Ping Yee6f3f9a42001-02-27 22:42:36 +00001096 if hasattr(object, '__date__'):
1097 result = result + self.section('DATE', str(object.__date__))
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001098 if hasattr(object, '__author__'):
Ka-Ping Yee6f3f9a42001-02-27 22:42:36 +00001099 result = result + self.section('AUTHOR', str(object.__author__))
1100 if hasattr(object, '__credits__'):
1101 result = result + self.section('CREDITS', str(object.__credits__))
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001102 return result
1103
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001104 def docclass(self, object, name=None, mod=None):
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001105 """Produce text documentation for a given class object."""
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001106 realname = object.__name__
1107 name = name or realname
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001108 bases = object.__bases__
1109
Tim Petersc86f6ca2001-09-26 21:31:51 +00001110 def makename(c, m=object.__module__):
1111 return classname(c, m)
1112
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001113 if name == realname:
1114 title = 'class ' + self.bold(realname)
1115 else:
1116 title = self.bold(name) + ' = class ' + realname
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001117 if bases:
Ka-Ping Yee3bda8792001-03-23 13:17:50 +00001118 parents = map(makename, bases)
Neal Norwitz9d72bb42007-04-17 08:48:32 +00001119 title = title + '(%s)' % ', '.join(parents)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001120
1121 doc = getdoc(object)
Tim Peters28355492001-09-23 21:29:55 +00001122 contents = doc and [doc + '\n'] or []
1123 push = contents.append
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001124
Tim Petersc86f6ca2001-09-26 21:31:51 +00001125 # List the mro, if non-trivial.
Raymond Hettinger756b3f32004-01-29 06:37:52 +00001126 mro = deque(inspect.getmro(object))
Tim Petersc86f6ca2001-09-26 21:31:51 +00001127 if len(mro) > 2:
1128 push("Method resolution order:")
1129 for base in mro:
1130 push(' ' + makename(base))
1131 push('')
1132
Tim Petersf4aad8e2001-09-24 22:40:47 +00001133 # Cute little class to pump out a horizontal rule between sections.
1134 class HorizontalRule:
1135 def __init__(self):
1136 self.needone = 0
1137 def maybe(self):
1138 if self.needone:
1139 push('-' * 70)
1140 self.needone = 1
1141 hr = HorizontalRule()
1142
Tim Peters28355492001-09-23 21:29:55 +00001143 def spill(msg, attrs, predicate):
Tim Petersfa26f7c2001-09-24 08:05:11 +00001144 ok, attrs = _split_list(attrs, predicate)
Tim Peters28355492001-09-23 21:29:55 +00001145 if ok:
Tim Petersf4aad8e2001-09-24 22:40:47 +00001146 hr.maybe()
Tim Peters28355492001-09-23 21:29:55 +00001147 push(msg)
1148 for name, kind, homecls, value in ok:
1149 push(self.document(getattr(object, name),
1150 name, mod, object))
1151 return attrs
1152
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +00001153 def spilldescriptors(msg, attrs, predicate):
Tim Petersfa26f7c2001-09-24 08:05:11 +00001154 ok, attrs = _split_list(attrs, predicate)
Tim Peters28355492001-09-23 21:29:55 +00001155 if ok:
Tim Petersf4aad8e2001-09-24 22:40:47 +00001156 hr.maybe()
Tim Peters28355492001-09-23 21:29:55 +00001157 push(msg)
1158 for name, kind, homecls, value in ok:
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +00001159 push(self._docdescriptor(name, value, mod))
Tim Peters28355492001-09-23 21:29:55 +00001160 return attrs
Tim Petersb47879b2001-09-24 04:47:19 +00001161
Tim Petersfa26f7c2001-09-24 08:05:11 +00001162 def spilldata(msg, attrs, predicate):
1163 ok, attrs = _split_list(attrs, predicate)
Tim Peters28355492001-09-23 21:29:55 +00001164 if ok:
Tim Petersf4aad8e2001-09-24 22:40:47 +00001165 hr.maybe()
Tim Peters28355492001-09-23 21:29:55 +00001166 push(msg)
1167 for name, kind, homecls, value in ok:
Guido van Rossumd59da4b2007-05-22 18:11:13 +00001168 if hasattr(value, '__call__') or inspect.isdatadescriptor(value):
Ka-Ping Yeebba6acc2005-02-19 22:58:26 +00001169 doc = getdoc(value)
Guido van Rossum5e355b22002-05-21 20:56:15 +00001170 else:
1171 doc = None
Tim Peters28355492001-09-23 21:29:55 +00001172 push(self.docother(getattr(object, name),
Georg Brandl8b813db2005-10-01 16:32:31 +00001173 name, mod, maxlen=70, doc=doc) + '\n')
Tim Peters28355492001-09-23 21:29:55 +00001174 return attrs
1175
Guido van Rossum1bc535d2007-05-15 18:46:22 +00001176 attrs = [(name, kind, cls, value)
1177 for name, kind, cls, value in classify_class_attrs(object)
1178 if visiblename(name)]
1179
Tim Petersfa26f7c2001-09-24 08:05:11 +00001180 while attrs:
Tim Peters351e3622001-09-27 03:29:51 +00001181 if mro:
Raymond Hettinger756b3f32004-01-29 06:37:52 +00001182 thisclass = mro.popleft()
Tim Peters351e3622001-09-27 03:29:51 +00001183 else:
1184 thisclass = attrs[0][2]
Tim Petersfa26f7c2001-09-24 08:05:11 +00001185 attrs, inherited = _split_list(attrs, lambda t: t[2] is thisclass)
1186
Georg Brandl1a3284e2007-12-02 09:40:06 +00001187 if thisclass is builtins.object:
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +00001188 attrs = inherited
1189 continue
1190 elif thisclass is object:
Tim Peters28355492001-09-23 21:29:55 +00001191 tag = "defined here"
1192 else:
Tim Petersfa26f7c2001-09-24 08:05:11 +00001193 tag = "inherited from %s" % classname(thisclass,
1194 object.__module__)
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +00001195 filter(lambda t: not t[0].startswith('_'), attrs)
Tim Peters28355492001-09-23 21:29:55 +00001196
1197 # Sort attrs by name.
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +00001198 attrs.sort()
Tim Peters28355492001-09-23 21:29:55 +00001199
1200 # Pump out the attrs, segregated by kind.
Tim Petersf4aad8e2001-09-24 22:40:47 +00001201 attrs = spill("Methods %s:\n" % tag, attrs,
Tim Peters28355492001-09-23 21:29:55 +00001202 lambda t: t[1] == 'method')
Tim Petersf4aad8e2001-09-24 22:40:47 +00001203 attrs = spill("Class methods %s:\n" % tag, attrs,
Tim Peters28355492001-09-23 21:29:55 +00001204 lambda t: t[1] == 'class method')
Tim Petersf4aad8e2001-09-24 22:40:47 +00001205 attrs = spill("Static methods %s:\n" % tag, attrs,
Tim Peters28355492001-09-23 21:29:55 +00001206 lambda t: t[1] == 'static method')
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +00001207 attrs = spilldescriptors("Data descriptors %s:\n" % tag, attrs,
1208 lambda t: t[1] == 'data descriptor')
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +00001209 attrs = spilldata("Data and other attributes %s:\n" % tag, attrs,
1210 lambda t: t[1] == 'data')
Tim Peters28355492001-09-23 21:29:55 +00001211 assert attrs == []
Tim Peters351e3622001-09-27 03:29:51 +00001212 attrs = inherited
Tim Peters28355492001-09-23 21:29:55 +00001213
1214 contents = '\n'.join(contents)
1215 if not contents:
1216 return title + '\n'
Neal Norwitz9d72bb42007-04-17 08:48:32 +00001217 return title + '\n' + self.indent(contents.rstrip(), ' | ') + '\n'
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001218
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001219 def formatvalue(self, object):
1220 """Format an argument default value as text."""
1221 return '=' + self.repr(object)
1222
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001223 def docroutine(self, object, name=None, mod=None, cl=None):
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001224 """Produce text documentation for a function or method object."""
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001225 realname = object.__name__
1226 name = name or realname
1227 note = ''
Ka-Ping Yee3bda8792001-03-23 13:17:50 +00001228 skipdocs = 0
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001229 if inspect.ismethod(object):
Christian Heimesff737952007-11-27 10:40:20 +00001230 imclass = object.__self__.__class__
Ka-Ping Yee3bda8792001-03-23 13:17:50 +00001231 if cl:
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001232 if imclass is not cl:
1233 note = ' from ' + classname(imclass, mod)
Ka-Ping Yee3bda8792001-03-23 13:17:50 +00001234 else:
Christian Heimesff737952007-11-27 10:40:20 +00001235 if object.__self__ is not None:
Ka-Ping Yeeb7a48302001-04-12 20:39:14 +00001236 note = ' method of %s instance' % classname(
Christian Heimesff737952007-11-27 10:40:20 +00001237 object.__self__.__class__, mod)
Ka-Ping Yeeb7a48302001-04-12 20:39:14 +00001238 else:
1239 note = ' unbound %s method' % classname(imclass,mod)
Christian Heimesff737952007-11-27 10:40:20 +00001240 object = object.__func__
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001241
1242 if name == realname:
1243 title = self.bold(realname)
1244 else:
Raymond Hettinger54f02222002-06-01 14:18:47 +00001245 if (cl and realname in cl.__dict__ and
Ka-Ping Yee3bda8792001-03-23 13:17:50 +00001246 cl.__dict__[realname] is object):
1247 skipdocs = 1
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001248 title = self.bold(name) + ' = ' + realname
Tim Peters4bcfa312001-09-20 06:08:24 +00001249 if inspect.isfunction(object):
Guido van Rossum2e65f892007-02-28 22:03:49 +00001250 args, varargs, varkw, defaults, kwonlyargs, kwdefaults, ann = \
1251 inspect.getfullargspec(object)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001252 argspec = inspect.formatargspec(
Guido van Rossum2e65f892007-02-28 22:03:49 +00001253 args, varargs, varkw, defaults, kwonlyargs, kwdefaults, ann,
1254 formatvalue=self.formatvalue,
1255 formatannotation=inspect.formatannotationrelativeto(object))
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001256 if realname == '<lambda>':
Georg Brandl501dd0d2006-02-17 09:45:40 +00001257 title = self.bold(name) + ' lambda '
Guido van Rossum2e65f892007-02-28 22:03:49 +00001258 # XXX lambda's won't usually have func_annotations['return']
1259 # since the syntax doesn't support but it is possible.
1260 # So removing parentheses isn't truly safe.
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001261 argspec = argspec[1:-1] # remove parentheses
Tim Peters4bcfa312001-09-20 06:08:24 +00001262 else:
1263 argspec = '(...)'
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001264 decl = title + argspec + note
1265
Ka-Ping Yee3bda8792001-03-23 13:17:50 +00001266 if skipdocs:
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001267 return decl + '\n'
Ka-Ping Yee3bda8792001-03-23 13:17:50 +00001268 else:
1269 doc = getdoc(object) or ''
Neal Norwitz9d72bb42007-04-17 08:48:32 +00001270 return decl + '\n' + (doc and self.indent(doc).rstrip() + '\n')
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001271
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +00001272 def _docdescriptor(self, name, value, mod):
Johannes Gijsbers8de645a2004-11-07 19:16:05 +00001273 results = []
1274 push = results.append
1275
1276 if name:
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +00001277 push(self.bold(name))
1278 push('\n')
Johannes Gijsbers8de645a2004-11-07 19:16:05 +00001279 doc = getdoc(value) or ''
1280 if doc:
1281 push(self.indent(doc))
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +00001282 push('\n')
1283 return ''.join(results)
Johannes Gijsbers8de645a2004-11-07 19:16:05 +00001284
1285 def docproperty(self, object, name=None, mod=None, cl=None):
1286 """Produce text documentation for a property."""
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +00001287 return self._docdescriptor(name, object, mod)
Johannes Gijsbers8de645a2004-11-07 19:16:05 +00001288
Thomas Wouters0e3f5912006-08-11 14:57:12 +00001289 def docdata(self, object, name=None, mod=None, cl=None):
1290 """Produce text documentation for a data descriptor."""
1291 return self._docdescriptor(name, object, mod)
1292
Georg Brandl8b813db2005-10-01 16:32:31 +00001293 def docother(self, object, name=None, mod=None, parent=None, maxlen=None, doc=None):
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001294 """Produce text documentation for a data object."""
1295 repr = self.repr(object)
1296 if maxlen:
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001297 line = (name and name + ' = ' or '') + repr
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001298 chop = maxlen - len(line)
1299 if chop < 0: repr = repr[:chop] + '...'
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001300 line = (name and self.bold(name) + ' = ' or '') + repr
Tim Peters28355492001-09-23 21:29:55 +00001301 if doc is not None:
1302 line += '\n' + self.indent(str(doc))
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001303 return line
1304
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001305# --------------------------------------------------------- user interfaces
1306
1307def pager(text):
1308 """The first time this is called, determine what kind of pager to use."""
1309 global pager
1310 pager = getpager()
1311 pager(text)
1312
1313def getpager():
1314 """Decide what method to use for paging through text."""
Guido van Rossuma01a8b62007-05-27 09:20:14 +00001315 if not hasattr(sys.stdout, "isatty"):
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001316 return plainpager
1317 if not sys.stdin.isatty() or not sys.stdout.isatty():
1318 return plainpager
Raymond Hettinger54f02222002-06-01 14:18:47 +00001319 if 'PAGER' in os.environ:
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +00001320 if sys.platform == 'win32': # pipes completely broken in Windows
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001321 return lambda text: tempfilepager(plain(text), os.environ['PAGER'])
Raymond Hettingerdbecd932005-02-06 06:57:08 +00001322 elif os.environ.get('TERM') in ('dumb', 'emacs'):
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001323 return lambda text: pipepager(plain(text), os.environ['PAGER'])
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +00001324 else:
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001325 return lambda text: pipepager(text, os.environ['PAGER'])
Ka-Ping Yeea487e4e2005-11-05 04:49:18 +00001326 if os.environ.get('TERM') in ('dumb', 'emacs'):
1327 return plainpager
Andrew MacIntyre54e0eab2002-03-03 03:12:30 +00001328 if sys.platform == 'win32' or sys.platform.startswith('os2'):
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001329 return lambda text: tempfilepager(plain(text), 'more <')
Skip Montanarod404bee2002-09-26 21:44:57 +00001330 if hasattr(os, 'system') and os.system('(less) 2>/dev/null') == 0:
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001331 return lambda text: pipepager(text, 'less')
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001332
1333 import tempfile
Guido van Rossum3b0a3292002-08-09 16:38:32 +00001334 (fd, filename) = tempfile.mkstemp()
1335 os.close(fd)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001336 try:
1337 if hasattr(os, 'system') and os.system('more %s' % filename) == 0:
1338 return lambda text: pipepager(text, 'more')
1339 else:
1340 return ttypager
1341 finally:
1342 os.unlink(filename)
1343
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001344def plain(text):
1345 """Remove boldface formatting from text."""
1346 return re.sub('.\b', '', text)
1347
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001348def pipepager(text, cmd):
1349 """Page through text by feeding it to another program."""
1350 pipe = os.popen(cmd, 'w')
1351 try:
1352 pipe.write(text)
1353 pipe.close()
1354 except IOError:
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001355 pass # Ignore broken pipes caused by quitting the pager program.
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001356
1357def tempfilepager(text, cmd):
1358 """Page through text by invoking a program on a temporary file."""
1359 import tempfile
Tim Peters550e4e52003-02-07 01:53:46 +00001360 filename = tempfile.mktemp()
1361 file = open(filename, 'w')
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001362 file.write(text)
1363 file.close()
1364 try:
Ka-Ping Yeec92cdf72001-03-02 05:54:35 +00001365 os.system(cmd + ' ' + filename)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001366 finally:
1367 os.unlink(filename)
1368
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001369def ttypager(text):
1370 """Page through text on a text terminal."""
Neal Norwitz9d72bb42007-04-17 08:48:32 +00001371 lines = plain(text).split('\n')
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001372 try:
1373 import tty
1374 fd = sys.stdin.fileno()
1375 old = tty.tcgetattr(fd)
1376 tty.setcbreak(fd)
1377 getchar = lambda: sys.stdin.read(1)
Ka-Ping Yee457aab22001-02-27 23:36:29 +00001378 except (ImportError, AttributeError):
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001379 tty = None
1380 getchar = lambda: sys.stdin.readline()[:-1][:1]
1381
1382 try:
1383 r = inc = os.environ.get('LINES', 25) - 1
Neal Norwitz9d72bb42007-04-17 08:48:32 +00001384 sys.stdout.write('\n'.join(lines[:inc]) + '\n')
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001385 while lines[r:]:
1386 sys.stdout.write('-- more --')
1387 sys.stdout.flush()
1388 c = getchar()
1389
Raymond Hettingerdbecd932005-02-06 06:57:08 +00001390 if c in ('q', 'Q'):
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001391 sys.stdout.write('\r \r')
1392 break
Raymond Hettingerdbecd932005-02-06 06:57:08 +00001393 elif c in ('\r', '\n'):
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001394 sys.stdout.write('\r \r' + lines[r] + '\n')
1395 r = r + 1
1396 continue
Raymond Hettingerdbecd932005-02-06 06:57:08 +00001397 if c in ('b', 'B', '\x1b'):
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001398 r = r - inc - inc
1399 if r < 0: r = 0
Neal Norwitz9d72bb42007-04-17 08:48:32 +00001400 sys.stdout.write('\n' + '\n'.join(lines[r:r+inc]) + '\n')
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001401 r = r + inc
1402
1403 finally:
1404 if tty:
1405 tty.tcsetattr(fd, tty.TCSAFLUSH, old)
1406
1407def plainpager(text):
1408 """Simply print unformatted text. This is the ultimate fallback."""
1409 sys.stdout.write(plain(text))
1410
1411def describe(thing):
Ka-Ping Yee3bda8792001-03-23 13:17:50 +00001412 """Produce a short description of the given thing."""
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001413 if inspect.ismodule(thing):
1414 if thing.__name__ in sys.builtin_module_names:
1415 return 'built-in module ' + thing.__name__
1416 if hasattr(thing, '__path__'):
1417 return 'package ' + thing.__name__
1418 else:
1419 return 'module ' + thing.__name__
1420 if inspect.isbuiltin(thing):
1421 return 'built-in function ' + thing.__name__
Thomas Wouters0e3f5912006-08-11 14:57:12 +00001422 if inspect.isgetsetdescriptor(thing):
1423 return 'getset descriptor %s.%s.%s' % (
1424 thing.__objclass__.__module__, thing.__objclass__.__name__,
1425 thing.__name__)
1426 if inspect.ismemberdescriptor(thing):
1427 return 'member descriptor %s.%s.%s' % (
1428 thing.__objclass__.__module__, thing.__objclass__.__name__,
1429 thing.__name__)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001430 if inspect.isclass(thing):
1431 return 'class ' + thing.__name__
1432 if inspect.isfunction(thing):
1433 return 'function ' + thing.__name__
1434 if inspect.ismethod(thing):
1435 return 'method ' + thing.__name__
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001436 return type(thing).__name__
1437
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001438def locate(path, forceload=0):
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001439 """Locate an object by name or dotted path, importing as necessary."""
Neal Norwitz9d72bb42007-04-17 08:48:32 +00001440 parts = [part for part in path.split('.') if part]
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001441 module, n = None, 0
1442 while n < len(parts):
Neal Norwitz9d72bb42007-04-17 08:48:32 +00001443 nextmodule = safeimport('.'.join(parts[:n+1]), forceload)
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001444 if nextmodule: module, n = nextmodule, n + 1
1445 else: break
1446 if module:
1447 object = module
1448 for part in parts[n:]:
1449 try: object = getattr(object, part)
1450 except AttributeError: return None
1451 return object
1452 else:
Georg Brandl1a3284e2007-12-02 09:40:06 +00001453 if hasattr(builtins, path):
1454 return getattr(builtins, path)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001455
1456# --------------------------------------- interactive interpreter interface
1457
1458text = TextDoc()
1459html = HTMLDoc()
1460
Ka-Ping Yee45daeb02002-08-11 15:11:33 +00001461def resolve(thing, forceload=0):
1462 """Given an object or a path to an object, get the object and its name."""
1463 if isinstance(thing, str):
1464 object = locate(thing, forceload)
1465 if not object:
Collin Winterce36ad82007-08-30 01:19:48 +00001466 raise ImportError('no Python documentation found for %r' % thing)
Ka-Ping Yee45daeb02002-08-11 15:11:33 +00001467 return object, thing
1468 else:
1469 return thing, getattr(thing, '__name__', None)
1470
Guido van Rossumd8faa362007-04-27 19:54:29 +00001471def render_doc(thing, title='Python Library Documentation: %s', forceload=0):
1472 """Render text documentation, given an object or a path to an object."""
1473 object, name = resolve(thing, forceload)
1474 desc = describe(object)
1475 module = inspect.getmodule(object)
1476 if name and '.' in name:
1477 desc += ' in ' + name[:name.rfind('.')]
1478 elif module and module is not object:
1479 desc += ' in module ' + module.__name__
1480 elif not (inspect.ismodule(object) or
1481 inspect.isclass(object) or
1482 inspect.isroutine(object) or
1483 inspect.isgetsetdescriptor(object) or
1484 inspect.ismemberdescriptor(object) or
1485 isinstance(object, property)):
1486 # If the passed object is a piece of data or an instance,
1487 # document its available methods instead of its value.
1488 object = type(object)
1489 desc += ' object'
1490 return title % desc + '\n\n' + text.document(object, name)
1491
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001492def doc(thing, title='Python Library Documentation: %s', forceload=0):
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001493 """Display text documentation, given an object or a path to an object."""
Ka-Ping Yee45daeb02002-08-11 15:11:33 +00001494 try:
1495 object, name = resolve(thing, forceload)
1496 desc = describe(object)
1497 module = inspect.getmodule(object)
1498 if name and '.' in name:
1499 desc += ' in ' + name[:name.rfind('.')]
1500 elif module and module is not object:
1501 desc += ' in module ' + module.__name__
Thomas Woutersfc7bb8c2007-01-15 15:49:28 +00001502 elif not (inspect.ismodule(object) or
1503 inspect.isclass(object) or
1504 inspect.isroutine(object) or
1505 inspect.isgetsetdescriptor(object) or
1506 inspect.ismemberdescriptor(object) or
1507 isinstance(object, property)):
Ka-Ping Yeebba6acc2005-02-19 22:58:26 +00001508 # If the passed object is a piece of data or an instance,
1509 # document its available methods instead of its value.
1510 object = type(object)
1511 desc += ' object'
Guido van Rossumd8faa362007-04-27 19:54:29 +00001512 pager(render_doc(thing, title, forceload))
Guido van Rossumb940e112007-01-10 16:19:56 +00001513 except (ImportError, ErrorDuringImport) as value:
Guido van Rossumbe19ed72007-02-09 05:37:30 +00001514 print(value)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001515
Ka-Ping Yee45daeb02002-08-11 15:11:33 +00001516def writedoc(thing, forceload=0):
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001517 """Write HTML documentation to a file in the current directory."""
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001518 try:
Ka-Ping Yee45daeb02002-08-11 15:11:33 +00001519 object, name = resolve(thing, forceload)
1520 page = html.page(describe(object), html.document(object, name))
1521 file = open(name + '.html', 'w')
1522 file.write(page)
1523 file.close()
Guido van Rossumbe19ed72007-02-09 05:37:30 +00001524 print('wrote', name + '.html')
Guido van Rossumb940e112007-01-10 16:19:56 +00001525 except (ImportError, ErrorDuringImport) as value:
Guido van Rossumbe19ed72007-02-09 05:37:30 +00001526 print(value)
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001527
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001528def writedocs(dir, pkgpath='', done=None):
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001529 """Write out HTML documentation for all modules in a directory tree."""
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001530 if done is None: done = {}
Thomas Wouters49fd7fa2006-04-21 10:40:58 +00001531 for importer, modname, ispkg in pkgutil.walk_packages([dir], pkgpath):
1532 writedoc(modname)
1533 return
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001534
1535class Helper:
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001536 keywords = {
1537 'and': 'BOOLEAN',
Guido van Rossumd8faa362007-04-27 19:54:29 +00001538 'as': 'with',
Ka-Ping Yee79c009d2001-04-13 10:53:25 +00001539 'assert': ('ref/assert', ''),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001540 'break': ('ref/break', 'while for'),
1541 'class': ('ref/class', 'CLASSES SPECIALMETHODS'),
1542 'continue': ('ref/continue', 'while for'),
1543 'def': ('ref/function', ''),
1544 'del': ('ref/del', 'BASICMETHODS'),
1545 'elif': 'if',
1546 'else': ('ref/if', 'while for'),
1547 'except': 'try',
1548 'exec': ('ref/exec', ''),
1549 'finally': 'try',
1550 'for': ('ref/for', 'break continue while'),
1551 'from': 'import',
1552 'global': ('ref/global', 'NAMESPACES'),
1553 'if': ('ref/if', 'TRUTHVALUE'),
1554 'import': ('ref/import', 'MODULES'),
1555 'in': ('ref/comparisons', 'SEQUENCEMETHODS2'),
1556 'is': 'COMPARISON',
Neal Norwitz742dde42003-03-30 20:31:34 +00001557 'lambda': ('ref/lambdas', 'FUNCTIONS'),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001558 'not': 'BOOLEAN',
1559 'or': 'BOOLEAN',
Neal Norwitz742dde42003-03-30 20:31:34 +00001560 'pass': ('ref/pass', ''),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001561 'print': ('ref/print', ''),
1562 'raise': ('ref/raise', 'EXCEPTIONS'),
Ka-Ping Yee79c009d2001-04-13 10:53:25 +00001563 'return': ('ref/return', 'FUNCTIONS'),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001564 'try': ('ref/try', 'EXCEPTIONS'),
1565 'while': ('ref/while', 'break continue if TRUTHVALUE'),
Guido van Rossumd8faa362007-04-27 19:54:29 +00001566 'with': ('ref/with', 'CONTEXTMANAGERS EXCEPTIONS yield'),
Tim Petersfb05c4e2002-10-30 05:21:00 +00001567 'yield': ('ref/yield', ''),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001568 }
1569
1570 topics = {
1571 'TYPES': ('ref/types', 'STRINGS UNICODE NUMBERS SEQUENCES MAPPINGS FUNCTIONS CLASSES MODULES FILES inspect'),
Ka-Ping Yee79c009d2001-04-13 10:53:25 +00001572 'STRINGS': ('ref/strings', 'str UNICODE SEQUENCES STRINGMETHODS FORMATTING TYPES'),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001573 'STRINGMETHODS': ('lib/string-methods', 'STRINGS FORMATTING'),
1574 'FORMATTING': ('lib/typesseq-strings', 'OPERATORS'),
Neal Norwitz54f871e2003-05-26 13:49:54 +00001575 'UNICODE': ('ref/strings', 'encodings unicode SEQUENCES STRINGMETHODS FORMATTING TYPES'),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001576 'NUMBERS': ('ref/numbers', 'INTEGER FLOAT COMPLEX TYPES'),
1577 'INTEGER': ('ref/integers', 'int range'),
1578 'FLOAT': ('ref/floating', 'float math'),
1579 'COMPLEX': ('ref/imaginary', 'complex cmath'),
Guido van Rossum805365e2007-05-07 22:24:25 +00001580 'SEQUENCES': ('lib/typesseq', 'STRINGMETHODS FORMATTING range LISTS'),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001581 'MAPPINGS': 'DICTIONARIES',
1582 'FUNCTIONS': ('lib/typesfunctions', 'def TYPES'),
1583 'METHODS': ('lib/typesmethods', 'class def CLASSES TYPES'),
1584 'CODEOBJECTS': ('lib/bltin-code-objects', 'compile FUNCTIONS TYPES'),
Ka-Ping Yee79c009d2001-04-13 10:53:25 +00001585 'TYPEOBJECTS': ('lib/bltin-type-objects', 'types TYPES'),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001586 'FRAMEOBJECTS': 'TYPES',
1587 'TRACEBACKS': 'TYPES',
1588 'NONE': ('lib/bltin-null-object', ''),
1589 'ELLIPSIS': ('lib/bltin-ellipsis-object', 'SLICINGS'),
1590 'FILES': ('lib/bltin-file-objects', ''),
1591 'SPECIALATTRIBUTES': ('lib/specialattrs', ''),
1592 'CLASSES': ('ref/types', 'class SPECIALMETHODS PRIVATENAMES'),
1593 'MODULES': ('lib/typesmodules', 'import'),
1594 'PACKAGES': 'import',
Neal Norwitz2eca4402006-08-29 04:40:24 +00001595 '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 +00001596 'OPERATORS': 'EXPRESSIONS',
1597 'PRECEDENCE': 'EXPRESSIONS',
1598 'OBJECTS': ('ref/objects', 'TYPES'),
1599 'SPECIALMETHODS': ('ref/specialnames', 'BASICMETHODS ATTRIBUTEMETHODS CALLABLEMETHODS SEQUENCEMETHODS1 MAPPINGMETHODS SEQUENCEMETHODS2 NUMBERMETHODS CLASSES'),
Ka-Ping Yee79c009d2001-04-13 10:53:25 +00001600 'BASICMETHODS': ('ref/customization', 'cmp hash repr str SPECIALMETHODS'),
1601 'ATTRIBUTEMETHODS': ('ref/attribute-access', 'ATTRIBUTES SPECIALMETHODS'),
1602 'CALLABLEMETHODS': ('ref/callable-types', 'CALLS SPECIALMETHODS'),
1603 'SEQUENCEMETHODS1': ('ref/sequence-types', 'SEQUENCES SEQUENCEMETHODS2 SPECIALMETHODS'),
1604 'SEQUENCEMETHODS2': ('ref/sequence-methods', 'SEQUENCES SEQUENCEMETHODS1 SPECIALMETHODS'),
1605 'MAPPINGMETHODS': ('ref/sequence-types', 'MAPPINGS SPECIALMETHODS'),
1606 'NUMBERMETHODS': ('ref/numeric-types', 'NUMBERS AUGMENTEDASSIGNMENT SPECIALMETHODS'),
Neal Norwitz54f871e2003-05-26 13:49:54 +00001607 'EXECUTION': ('ref/execmodel', 'NAMESPACES DYNAMICFEATURES EXCEPTIONS'),
1608 'NAMESPACES': ('ref/naming', 'global ASSIGNMENT DELETION DYNAMICFEATURES'),
1609 'DYNAMICFEATURES': ('ref/dynamic-features', ''),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001610 'SCOPING': 'NAMESPACES',
1611 'FRAMES': 'NAMESPACES',
1612 'EXCEPTIONS': ('ref/exceptions', 'try except finally raise'),
Neal Norwitz54f871e2003-05-26 13:49:54 +00001613 'COERCIONS': ('ref/coercion-rules','CONVERSIONS'),
1614 'CONVERSIONS': ('ref/conversions', 'COERCIONS'),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001615 'IDENTIFIERS': ('ref/identifiers', 'keywords SPECIALIDENTIFIERS'),
1616 'SPECIALIDENTIFIERS': ('ref/id-classes', ''),
Ka-Ping Yee79c009d2001-04-13 10:53:25 +00001617 'PRIVATENAMES': ('ref/atom-identifiers', ''),
Neal Norwitz2eca4402006-08-29 04:40:24 +00001618 'LITERALS': ('ref/atom-literals', 'STRINGS NUMBERS TUPLELITERALS LISTLITERALS DICTIONARYLITERALS'),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001619 'TUPLES': 'SEQUENCES',
Ka-Ping Yee79c009d2001-04-13 10:53:25 +00001620 'TUPLELITERALS': ('ref/exprlists', 'TUPLES LITERALS'),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001621 'LISTS': ('lib/typesseq-mutable', 'LISTLITERALS'),
Ka-Ping Yee79c009d2001-04-13 10:53:25 +00001622 'LISTLITERALS': ('ref/lists', 'LISTS LITERALS'),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001623 'DICTIONARIES': ('lib/typesmapping', 'DICTIONARYLITERALS'),
Ka-Ping Yee79c009d2001-04-13 10:53:25 +00001624 'DICTIONARYLITERALS': ('ref/dict', 'DICTIONARIES LITERALS'),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001625 'ATTRIBUTES': ('ref/attribute-references', 'getattr hasattr setattr ATTRIBUTEMETHODS'),
1626 'SUBSCRIPTS': ('ref/subscriptions', 'SEQUENCEMETHODS1'),
1627 'SLICINGS': ('ref/slicings', 'SEQUENCEMETHODS2'),
1628 'CALLS': ('ref/calls', 'EXPRESSIONS'),
1629 'POWER': ('ref/power', 'EXPRESSIONS'),
1630 'UNARY': ('ref/unary', 'EXPRESSIONS'),
1631 'BINARY': ('ref/binary', 'EXPRESSIONS'),
1632 'SHIFTING': ('ref/shifting', 'EXPRESSIONS'),
1633 'BITWISE': ('ref/bitwise', 'EXPRESSIONS'),
1634 'COMPARISON': ('ref/comparisons', 'EXPRESSIONS BASICMETHODS'),
Neal Norwitzf98159c2003-02-07 20:49:40 +00001635 'BOOLEAN': ('ref/Booleans', 'EXPRESSIONS TRUTHVALUE'),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001636 'ASSERTION': 'assert',
1637 'ASSIGNMENT': ('ref/assignment', 'AUGMENTEDASSIGNMENT'),
Ka-Ping Yee79c009d2001-04-13 10:53:25 +00001638 'AUGMENTEDASSIGNMENT': ('ref/augassign', 'NUMBERMETHODS'),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001639 'DELETION': 'del',
1640 'PRINTING': 'print',
1641 'RETURNING': 'return',
1642 'IMPORTING': 'import',
1643 'CONDITIONAL': 'if',
1644 'LOOPING': ('ref/compound', 'for while break continue'),
1645 'TRUTHVALUE': ('lib/truth', 'if while and or not BASICMETHODS'),
Ka-Ping Yee79c009d2001-04-13 10:53:25 +00001646 'DEBUGGING': ('lib/module-pdb', 'pdb'),
Guido van Rossumd8faa362007-04-27 19:54:29 +00001647 'CONTEXTMANAGERS': ('ref/context-managers', 'with'),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001648 }
1649
1650 def __init__(self, input, output):
1651 self.input = input
1652 self.output = output
1653 self.docdir = None
1654 execdir = os.path.dirname(sys.executable)
1655 homedir = os.environ.get('PYTHONHOME')
Guido van Rossumd8faa362007-04-27 19:54:29 +00001656 join = os.path.join
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001657 for dir in [os.environ.get('PYTHONDOCS'),
1658 homedir and os.path.join(homedir, 'doc'),
Guido van Rossumd8faa362007-04-27 19:54:29 +00001659 join(execdir, 'doc'), # for Windows
1660 join(sys.prefix, 'doc/python-docs-' + sys.version.split()[0]),
1661 join(sys.prefix, 'doc/python-' + sys.version.split()[0]),
1662 join(sys.prefix, 'doc/python-docs-' + sys.version[:3]),
1663 join(sys.prefix, 'doc/python-' + sys.version[:3]),
1664 join(sys.prefix, 'Resources/English.lproj/Documentation')]:
1665 if dir and os.path.isdir(join(dir, 'lib')):
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001666 self.docdir = dir
Guido van Rossumd8faa362007-04-27 19:54:29 +00001667 break
1668 if dir and os.path.isdir(join(dir, 'html', 'lib')):
1669 self.docdir = join(dir, 'html')
1670 break
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001671
Ka-Ping Yee79c009d2001-04-13 10:53:25 +00001672 def __repr__(self):
Ka-Ping Yee9bc576b2001-04-13 13:57:31 +00001673 if inspect.stack()[1][3] == '?':
Ka-Ping Yee79c009d2001-04-13 10:53:25 +00001674 self()
1675 return ''
Ka-Ping Yee9bc576b2001-04-13 13:57:31 +00001676 return '<pydoc.Helper instance>'
Ka-Ping Yee79c009d2001-04-13 10:53:25 +00001677
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001678 def __call__(self, request=None):
1679 if request is not None:
1680 self.help(request)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001681 else:
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001682 self.intro()
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001683 self.interact()
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001684 self.output.write('''
Fred Drakee61967f2001-05-10 18:41:02 +00001685You are now leaving help and returning to the Python interpreter.
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001686If you want to ask for help on a particular object directly from the
1687interpreter, you can type "help(object)". Executing "help('string')"
1688has the same effect as typing a particular string at the help> prompt.
1689''')
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001690
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001691 def interact(self):
1692 self.output.write('\n')
Guido van Rossum8ca162f2002-04-07 06:36:23 +00001693 while True:
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001694 try:
Johannes Gijsberse7691d32004-08-17 13:21:53 +00001695 request = self.getline('help> ')
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001696 if not request: break
Johannes Gijsberse7691d32004-08-17 13:21:53 +00001697 except (KeyboardInterrupt, EOFError):
1698 break
Neal Norwitz9d72bb42007-04-17 08:48:32 +00001699 request = replace(request, '"', '', "'", '').strip()
1700 if request.lower() in ('q', 'quit'): break
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001701 self.help(request)
1702
Johannes Gijsberse7691d32004-08-17 13:21:53 +00001703 def getline(self, prompt):
Guido van Rossum7c086c02008-01-02 03:52:38 +00001704 """Read one line, using input() when appropriate."""
Johannes Gijsberse7691d32004-08-17 13:21:53 +00001705 if self.input is sys.stdin:
Guido van Rossum7c086c02008-01-02 03:52:38 +00001706 return input(prompt)
Johannes Gijsberse7691d32004-08-17 13:21:53 +00001707 else:
1708 self.output.write(prompt)
1709 self.output.flush()
1710 return self.input.readline()
1711
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001712 def help(self, request):
1713 if type(request) is type(''):
1714 if request == 'help': self.intro()
1715 elif request == 'keywords': self.listkeywords()
1716 elif request == 'topics': self.listtopics()
1717 elif request == 'modules': self.listmodules()
1718 elif request[:8] == 'modules ':
Neal Norwitz9d72bb42007-04-17 08:48:32 +00001719 self.listmodules(request.split()[1])
Raymond Hettinger54f02222002-06-01 14:18:47 +00001720 elif request in self.keywords: self.showtopic(request)
1721 elif request in self.topics: self.showtopic(request)
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001722 elif request: doc(request, 'Help on %s:')
1723 elif isinstance(request, Helper): self()
1724 else: doc(request, 'Help on %s:')
1725 self.output.write('\n')
1726
1727 def intro(self):
1728 self.output.write('''
1729Welcome to Python %s! This is the online help utility.
1730
1731If this is your first time using Python, you should definitely check out
1732the tutorial on the Internet at http://www.python.org/doc/tut/.
1733
1734Enter the name of any module, keyword, or topic to get help on writing
1735Python programs and using Python modules. To quit this help utility and
1736return to the interpreter, just type "quit".
1737
1738To get a list of available modules, keywords, or topics, type "modules",
1739"keywords", or "topics". Each module also comes with a one-line summary
1740of what it does; to list the modules whose summaries contain a given word
1741such as "spam", type "modules spam".
1742''' % sys.version[:3])
1743
1744 def list(self, items, columns=4, width=80):
Guido van Rossum486364b2007-06-30 05:01:58 +00001745 items = list(sorted(items))
1746 colw = width // columns
1747 rows = (len(items) + columns - 1) // columns
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001748 for row in range(rows):
1749 for col in range(columns):
1750 i = col * rows + row
1751 if i < len(items):
1752 self.output.write(items[i])
1753 if col < columns - 1:
Guido van Rossum486364b2007-06-30 05:01:58 +00001754 self.output.write(' ' + ' ' * (colw - 1 - len(items[i])))
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001755 self.output.write('\n')
1756
1757 def listkeywords(self):
1758 self.output.write('''
1759Here is a list of the Python keywords. Enter any keyword to get more help.
1760
1761''')
1762 self.list(self.keywords.keys())
1763
1764 def listtopics(self):
1765 self.output.write('''
1766Here is a list of available topics. Enter any topic name to get more help.
1767
1768''')
1769 self.list(self.topics.keys())
1770
1771 def showtopic(self, topic):
1772 if not self.docdir:
1773 self.output.write('''
1774Sorry, topic and keyword documentation is not available because the Python
1775HTML documentation files could not be found. If you have installed them,
1776please set the environment variable PYTHONDOCS to indicate their location.
Thomas Wouters902d6eb2007-01-09 23:18:33 +00001777
1778On the Microsoft Windows operating system, the files can be built by
1779running "hh -decompile . PythonNN.chm" in the C:\PythonNN\Doc> directory.
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001780''')
1781 return
1782 target = self.topics.get(topic, self.keywords.get(topic))
1783 if not target:
1784 self.output.write('no documentation found for %s\n' % repr(topic))
1785 return
1786 if type(target) is type(''):
1787 return self.showtopic(target)
1788
1789 filename, xrefs = target
1790 filename = self.docdir + '/' + filename + '.html'
1791 try:
1792 file = open(filename)
1793 except:
1794 self.output.write('could not read docs from %s\n' % filename)
1795 return
1796
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001797 divpat = re.compile('<div[^>]*navigat.*?</div.*?>', re.I | re.S)
1798 addrpat = re.compile('<address.*?>.*?</address.*?>', re.I | re.S)
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001799 document = re.sub(addrpat, '', re.sub(divpat, '', file.read()))
1800 file.close()
1801
Guido van Rossum34d19282007-08-09 01:03:29 +00001802 import htmllib, formatter, io
1803 buffer = io.StringIO()
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001804 parser = htmllib.HTMLParser(
1805 formatter.AbstractFormatter(formatter.DumbWriter(buffer)))
1806 parser.start_table = parser.do_p
1807 parser.end_table = lambda parser=parser: parser.do_p({})
1808 parser.start_tr = parser.do_br
1809 parser.start_td = parser.start_th = lambda a, b=buffer: b.write('\t')
1810 parser.feed(document)
1811 buffer = replace(buffer.getvalue(), '\xa0', ' ', '\n', '\n ')
Neal Norwitz9d72bb42007-04-17 08:48:32 +00001812 pager(' ' + buffer.strip() + '\n')
Ka-Ping Yeeda793892001-04-13 11:02:51 +00001813 if xrefs:
Guido van Rossum34d19282007-08-09 01:03:29 +00001814 buffer = io.StringIO()
Ka-Ping Yeeda793892001-04-13 11:02:51 +00001815 formatter.DumbWriter(buffer).send_flowing_data(
Neal Norwitz9d72bb42007-04-17 08:48:32 +00001816 'Related help topics: ' + ', '.join(xrefs.split()) + '\n')
Ka-Ping Yeeda793892001-04-13 11:02:51 +00001817 self.output.write('\n%s\n' % buffer.getvalue())
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001818
1819 def listmodules(self, key=''):
1820 if key:
1821 self.output.write('''
1822Here is a list of matching modules. Enter any module name to get more help.
1823
1824''')
1825 apropos(key)
1826 else:
1827 self.output.write('''
1828Please wait a moment while I gather a list of all available modules...
1829
1830''')
1831 modules = {}
1832 def callback(path, modname, desc, modules=modules):
1833 if modname and modname[-9:] == '.__init__':
1834 modname = modname[:-9] + ' (package)'
Neal Norwitz9d72bb42007-04-17 08:48:32 +00001835 if modname.find('.') < 0:
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001836 modules[modname] = 1
Christian Heimesd32ed6f2008-01-14 18:49:24 +00001837 def onerror(modname):
1838 callback(None, modname, None)
1839 ModuleScanner().run(callback, onerror=onerror)
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001840 self.list(modules.keys())
1841 self.output.write('''
1842Enter any module name to get more help. Or, type "modules spam" to search
1843for modules whose descriptions contain the word "spam".
1844''')
1845
1846help = Helper(sys.stdin, sys.stdout)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001847
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001848class Scanner:
1849 """A generic tree iterator."""
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001850 def __init__(self, roots, children, descendp):
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001851 self.roots = roots[:]
1852 self.state = []
1853 self.children = children
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001854 self.descendp = descendp
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001855
1856 def next(self):
1857 if not self.state:
1858 if not self.roots:
1859 return None
1860 root = self.roots.pop(0)
1861 self.state = [(root, self.children(root))]
1862 node, children = self.state[-1]
1863 if not children:
1864 self.state.pop()
1865 return self.next()
1866 child = children.pop(0)
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001867 if self.descendp(child):
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001868 self.state.append((child, self.children(child)))
1869 return child
1870
Thomas Wouters49fd7fa2006-04-21 10:40:58 +00001871
1872class ModuleScanner:
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001873 """An interruptible scanner that searches module synopses."""
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001874
Christian Heimesd32ed6f2008-01-14 18:49:24 +00001875 def run(self, callback, key=None, completer=None, onerror=None):
Neal Norwitz9d72bb42007-04-17 08:48:32 +00001876 if key: key = key.lower()
Guido van Rossum8ca162f2002-04-07 06:36:23 +00001877 self.quit = False
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001878 seen = {}
1879
1880 for modname in sys.builtin_module_names:
Ka-Ping Yee239432a2001-03-02 02:45:08 +00001881 if modname != '__main__':
1882 seen[modname] = 1
Ka-Ping Yee66246962001-04-12 11:59:50 +00001883 if key is None:
1884 callback(None, modname, '')
1885 else:
Neal Norwitz9d72bb42007-04-17 08:48:32 +00001886 name = __import__(modname).__doc__ or ''
1887 desc = name.split('\n')[0]
1888 name = modname + ' - ' + desc
1889 if name.lower().find(key) >= 0:
Ka-Ping Yee66246962001-04-12 11:59:50 +00001890 callback(None, modname, desc)
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001891
Christian Heimesd32ed6f2008-01-14 18:49:24 +00001892 for importer, modname, ispkg in pkgutil.walk_packages(onerror=onerror):
Thomas Wouters49fd7fa2006-04-21 10:40:58 +00001893 if self.quit:
1894 break
1895 if key is None:
1896 callback(None, modname, '')
1897 else:
1898 loader = importer.find_module(modname)
1899 if hasattr(loader,'get_source'):
Guido van Rossum34d19282007-08-09 01:03:29 +00001900 import io
Thomas Wouters49fd7fa2006-04-21 10:40:58 +00001901 desc = source_synopsis(
Guido van Rossum34d19282007-08-09 01:03:29 +00001902 io.StringIO(loader.get_source(modname))
Thomas Wouters49fd7fa2006-04-21 10:40:58 +00001903 ) or ''
1904 if hasattr(loader,'get_filename'):
1905 path = loader.get_filename(modname)
Ka-Ping Yee66246962001-04-12 11:59:50 +00001906 else:
Thomas Wouters49fd7fa2006-04-21 10:40:58 +00001907 path = None
1908 else:
1909 module = loader.load_module(modname)
1910 desc = (module.__doc__ or '').splitlines()[0]
1911 path = getattr(module,'__file__',None)
Neal Norwitz9d72bb42007-04-17 08:48:32 +00001912 name = modname + ' - ' + desc
1913 if name.lower().find(key) >= 0:
Thomas Wouters49fd7fa2006-04-21 10:40:58 +00001914 callback(path, modname, desc)
1915
1916 if completer:
1917 completer()
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001918
1919def apropos(key):
1920 """Print all the one-line module summaries that contain a substring."""
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001921 def callback(path, modname, desc):
1922 if modname[-9:] == '.__init__':
1923 modname = modname[:-9] + ' (package)'
Guido van Rossumbe19ed72007-02-09 05:37:30 +00001924 print(modname, desc and '- ' + desc)
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +00001925 try: import warnings
1926 except ImportError: pass
1927 else: warnings.filterwarnings('ignore') # ignore problems during import
Ka-Ping Yee66246962001-04-12 11:59:50 +00001928 ModuleScanner().run(callback, key)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001929
1930# --------------------------------------------------- web browser interface
1931
Ka-Ping Yee66246962001-04-12 11:59:50 +00001932def serve(port, callback=None, completer=None):
Ka-Ping Yeefd540692001-04-12 12:54:36 +00001933 import BaseHTTPServer, mimetools, select
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001934
1935 # Patch up mimetools.Message so it doesn't break if rfc822 is reloaded.
1936 class Message(mimetools.Message):
1937 def __init__(self, fp, seekable=1):
1938 Message = self.__class__
1939 Message.__bases__[0].__bases__[0].__init__(self, fp, seekable)
1940 self.encodingheader = self.getheader('content-transfer-encoding')
1941 self.typeheader = self.getheader('content-type')
1942 self.parsetype()
1943 self.parseplist()
1944
1945 class DocHandler(BaseHTTPServer.BaseHTTPRequestHandler):
1946 def send_document(self, title, contents):
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001947 try:
1948 self.send_response(200)
1949 self.send_header('Content-Type', 'text/html')
1950 self.end_headers()
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001951 self.wfile.write(html.page(title, contents))
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001952 except IOError: pass
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001953
1954 def do_GET(self):
1955 path = self.path
1956 if path[-5:] == '.html': path = path[:-5]
1957 if path[:1] == '/': path = path[1:]
1958 if path and path != '.':
1959 try:
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001960 obj = locate(path, forceload=1)
Guido van Rossumb940e112007-01-10 16:19:56 +00001961 except ErrorDuringImport as value:
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001962 self.send_document(path, html.escape(str(value)))
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001963 return
Ka-Ping Yee3bda8792001-03-23 13:17:50 +00001964 if obj:
1965 self.send_document(describe(obj), html.document(obj, path))
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001966 else:
1967 self.send_document(path,
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001968'no Python documentation found for %s' % repr(path))
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001969 else:
1970 heading = html.heading(
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001971'<big><big><strong>Python: Index of Modules</strong></big></big>',
1972'#ffffff', '#7799ee')
Ka-Ping Yee239432a2001-03-02 02:45:08 +00001973 def bltinlink(name):
1974 return '<a href="%s.html">%s</a>' % (name, name)
Ka-Ping Yee3bda8792001-03-23 13:17:50 +00001975 names = filter(lambda x: x != '__main__',
1976 sys.builtin_module_names)
Ka-Ping Yee239432a2001-03-02 02:45:08 +00001977 contents = html.multicolumn(names, bltinlink)
1978 indices = ['<p>' + html.bigsection(
1979 'Built-in Modules', '#ffffff', '#ee77aa', contents)]
1980
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001981 seen = {}
Thomas Wouters49fd7fa2006-04-21 10:40:58 +00001982 for dir in sys.path:
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001983 indices.append(html.index(dir, seen))
Neal Norwitz9d72bb42007-04-17 08:48:32 +00001984 contents = heading + ' '.join(indices) + '''<p align=right>
Tim Peters2306d242001-09-25 03:18:32 +00001985<font color="#909090" face="helvetica, arial"><strong>
1986pydoc</strong> by Ka-Ping Yee &lt;ping@lfw.org&gt;</font>'''
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001987 self.send_document('Index of Modules', contents)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001988
1989 def log_message(self, *args): pass
1990
Ka-Ping Yeefd540692001-04-12 12:54:36 +00001991 class DocServer(BaseHTTPServer.HTTPServer):
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001992 def __init__(self, port, callback):
Ka-Ping Yeedb8ed152001-03-02 05:58:17 +00001993 host = (sys.platform == 'mac') and '127.0.0.1' or 'localhost'
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001994 self.address = ('', port)
Ka-Ping Yeedb8ed152001-03-02 05:58:17 +00001995 self.url = 'http://%s:%d/' % (host, port)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001996 self.callback = callback
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001997 self.base.__init__(self, self.address, self.handler)
1998
1999 def serve_until_quit(self):
2000 import select
Guido van Rossum8ca162f2002-04-07 06:36:23 +00002001 self.quit = False
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002002 while not self.quit:
2003 rd, wr, ex = select.select([self.socket.fileno()], [], [], 1)
2004 if rd: self.handle_request()
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002005
2006 def server_activate(self):
2007 self.base.server_activate(self)
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002008 if self.callback: self.callback(self)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002009
2010 DocServer.base = BaseHTTPServer.HTTPServer
2011 DocServer.handler = DocHandler
2012 DocHandler.MessageClass = Message
2013 try:
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00002014 try:
2015 DocServer(port, callback).serve_until_quit()
2016 except (KeyboardInterrupt, select.error):
2017 pass
2018 finally:
Ka-Ping Yee66246962001-04-12 11:59:50 +00002019 if completer: completer()
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002020
2021# ----------------------------------------------------- graphical interface
2022
2023def gui():
2024 """Graphical interface (starts web server and pops up a control window)."""
2025 class GUI:
2026 def __init__(self, window, port=7464):
2027 self.window = window
2028 self.server = None
2029 self.scanner = None
2030
2031 import Tkinter
2032 self.server_frm = Tkinter.Frame(window)
2033 self.title_lbl = Tkinter.Label(self.server_frm,
2034 text='Starting server...\n ')
2035 self.open_btn = Tkinter.Button(self.server_frm,
2036 text='open browser', command=self.open, state='disabled')
2037 self.quit_btn = Tkinter.Button(self.server_frm,
2038 text='quit serving', command=self.quit, state='disabled')
2039
2040 self.search_frm = Tkinter.Frame(window)
2041 self.search_lbl = Tkinter.Label(self.search_frm, text='Search for')
2042 self.search_ent = Tkinter.Entry(self.search_frm)
2043 self.search_ent.bind('<Return>', self.search)
2044 self.stop_btn = Tkinter.Button(self.search_frm,
2045 text='stop', pady=0, command=self.stop, state='disabled')
2046 if sys.platform == 'win32':
Ka-Ping Yeec92cdf72001-03-02 05:54:35 +00002047 # Trying to hide and show this button crashes under Windows.
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002048 self.stop_btn.pack(side='right')
2049
2050 self.window.title('pydoc')
2051 self.window.protocol('WM_DELETE_WINDOW', self.quit)
2052 self.title_lbl.pack(side='top', fill='x')
2053 self.open_btn.pack(side='left', fill='x', expand=1)
2054 self.quit_btn.pack(side='right', fill='x', expand=1)
2055 self.server_frm.pack(side='top', fill='x')
2056
2057 self.search_lbl.pack(side='left')
2058 self.search_ent.pack(side='right', fill='x', expand=1)
2059 self.search_frm.pack(side='top', fill='x')
2060 self.search_ent.focus_set()
2061
Ka-Ping Yeec92cdf72001-03-02 05:54:35 +00002062 font = ('helvetica', sys.platform == 'win32' and 8 or 10)
Ka-Ping Yee239432a2001-03-02 02:45:08 +00002063 self.result_lst = Tkinter.Listbox(window, font=font, height=6)
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002064 self.result_lst.bind('<Button-1>', self.select)
2065 self.result_lst.bind('<Double-Button-1>', self.goto)
2066 self.result_scr = Tkinter.Scrollbar(window,
2067 orient='vertical', command=self.result_lst.yview)
2068 self.result_lst.config(yscrollcommand=self.result_scr.set)
2069
2070 self.result_frm = Tkinter.Frame(window)
2071 self.goto_btn = Tkinter.Button(self.result_frm,
2072 text='go to selected', command=self.goto)
2073 self.hide_btn = Tkinter.Button(self.result_frm,
2074 text='hide results', command=self.hide)
2075 self.goto_btn.pack(side='left', fill='x', expand=1)
2076 self.hide_btn.pack(side='right', fill='x', expand=1)
2077
2078 self.window.update()
2079 self.minwidth = self.window.winfo_width()
2080 self.minheight = self.window.winfo_height()
2081 self.bigminheight = (self.server_frm.winfo_reqheight() +
2082 self.search_frm.winfo_reqheight() +
2083 self.result_lst.winfo_reqheight() +
2084 self.result_frm.winfo_reqheight())
2085 self.bigwidth, self.bigheight = self.minwidth, self.bigminheight
2086 self.expanded = 0
2087 self.window.wm_geometry('%dx%d' % (self.minwidth, self.minheight))
2088 self.window.wm_minsize(self.minwidth, self.minheight)
Martin v. Löwis5b26abb2002-12-28 09:23:09 +00002089 self.window.tk.willdispatch()
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002090
2091 import threading
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00002092 threading.Thread(
2093 target=serve, args=(port, self.ready, self.quit)).start()
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002094
2095 def ready(self, server):
2096 self.server = server
2097 self.title_lbl.config(
2098 text='Python documentation server at\n' + server.url)
2099 self.open_btn.config(state='normal')
2100 self.quit_btn.config(state='normal')
2101
Ka-Ping Yee239432a2001-03-02 02:45:08 +00002102 def open(self, event=None, url=None):
2103 url = url or self.server.url
2104 try:
2105 import webbrowser
2106 webbrowser.open(url)
2107 except ImportError: # pre-webbrowser.py compatibility
Ka-Ping Yeec92cdf72001-03-02 05:54:35 +00002108 if sys.platform == 'win32':
Ka-Ping Yee239432a2001-03-02 02:45:08 +00002109 os.system('start "%s"' % url)
2110 elif sys.platform == 'mac':
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +00002111 try: import ic
Ka-Ping Yee239432a2001-03-02 02:45:08 +00002112 except ImportError: pass
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +00002113 else: ic.launchurl(url)
Ka-Ping Yee239432a2001-03-02 02:45:08 +00002114 else:
2115 rc = os.system('netscape -remote "openURL(%s)" &' % url)
2116 if rc: os.system('netscape "%s" &' % url)
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002117
2118 def quit(self, event=None):
2119 if self.server:
2120 self.server.quit = 1
2121 self.window.quit()
2122
2123 def search(self, event=None):
2124 key = self.search_ent.get()
2125 self.stop_btn.pack(side='right')
2126 self.stop_btn.config(state='normal')
2127 self.search_lbl.config(text='Searching for "%s"...' % key)
2128 self.search_ent.forget()
2129 self.search_lbl.pack(side='left')
2130 self.result_lst.delete(0, 'end')
2131 self.goto_btn.config(state='disabled')
2132 self.expand()
2133
2134 import threading
2135 if self.scanner:
2136 self.scanner.quit = 1
2137 self.scanner = ModuleScanner()
2138 threading.Thread(target=self.scanner.run,
Ka-Ping Yee6dcfa382001-04-12 20:27:31 +00002139 args=(self.update, key, self.done)).start()
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002140
2141 def update(self, path, modname, desc):
2142 if modname[-9:] == '.__init__':
2143 modname = modname[:-9] + ' (package)'
2144 self.result_lst.insert('end',
2145 modname + ' - ' + (desc or '(no description)'))
2146
2147 def stop(self, event=None):
2148 if self.scanner:
2149 self.scanner.quit = 1
2150 self.scanner = None
2151
2152 def done(self):
2153 self.scanner = None
2154 self.search_lbl.config(text='Search for')
2155 self.search_lbl.pack(side='left')
2156 self.search_ent.pack(side='right', fill='x', expand=1)
2157 if sys.platform != 'win32': self.stop_btn.forget()
2158 self.stop_btn.config(state='disabled')
2159
2160 def select(self, event=None):
2161 self.goto_btn.config(state='normal')
2162
2163 def goto(self, event=None):
2164 selection = self.result_lst.curselection()
2165 if selection:
Neal Norwitz9d72bb42007-04-17 08:48:32 +00002166 modname = self.result_lst.get(selection[0]).split()[0]
Ka-Ping Yee239432a2001-03-02 02:45:08 +00002167 self.open(url=self.server.url + modname + '.html')
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002168
2169 def collapse(self):
2170 if not self.expanded: return
2171 self.result_frm.forget()
2172 self.result_scr.forget()
2173 self.result_lst.forget()
2174 self.bigwidth = self.window.winfo_width()
2175 self.bigheight = self.window.winfo_height()
2176 self.window.wm_geometry('%dx%d' % (self.minwidth, self.minheight))
2177 self.window.wm_minsize(self.minwidth, self.minheight)
2178 self.expanded = 0
2179
2180 def expand(self):
2181 if self.expanded: return
2182 self.result_frm.pack(side='bottom', fill='x')
2183 self.result_scr.pack(side='right', fill='y')
2184 self.result_lst.pack(side='top', fill='both', expand=1)
2185 self.window.wm_geometry('%dx%d' % (self.bigwidth, self.bigheight))
2186 self.window.wm_minsize(self.minwidth, self.bigminheight)
2187 self.expanded = 1
2188
2189 def hide(self, event=None):
2190 self.stop()
2191 self.collapse()
2192
2193 import Tkinter
2194 try:
Martin v. Löwise09bd932004-08-22 16:13:26 +00002195 root = Tkinter.Tk()
2196 # Tk will crash if pythonw.exe has an XP .manifest
2197 # file and the root has is not destroyed explicitly.
2198 # If the problem is ever fixed in Tk, the explicit
2199 # destroy can go.
2200 try:
2201 gui = GUI(root)
2202 root.mainloop()
2203 finally:
2204 root.destroy()
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002205 except KeyboardInterrupt:
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002206 pass
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002207
2208# -------------------------------------------------- command-line interface
2209
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00002210def ispath(x):
Neal Norwitz9d72bb42007-04-17 08:48:32 +00002211 return isinstance(x, str) and x.find(os.sep) >= 0
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00002212
Ka-Ping Yee1d384632001-03-01 00:24:32 +00002213def cli():
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002214 """Command-line interface (looks at sys.argv to decide what to do)."""
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002215 import getopt
Guido van Rossum756aa932007-04-07 03:04:01 +00002216 class BadUsage(Exception): pass
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002217
Ka-Ping Yee3bda8792001-03-23 13:17:50 +00002218 # Scripts don't get the current directory in their path by default.
Ka-Ping Yeef78a81b2001-03-27 08:13:42 +00002219 scriptdir = os.path.dirname(sys.argv[0])
2220 if scriptdir in sys.path:
2221 sys.path.remove(scriptdir)
Ka-Ping Yee3bda8792001-03-23 13:17:50 +00002222 sys.path.insert(0, '.')
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002223
Ka-Ping Yee3bda8792001-03-23 13:17:50 +00002224 try:
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002225 opts, args = getopt.getopt(sys.argv[1:], 'gk:p:w')
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002226 writing = 0
2227
2228 for opt, val in opts:
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002229 if opt == '-g':
2230 gui()
2231 return
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002232 if opt == '-k':
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002233 apropos(val)
2234 return
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002235 if opt == '-p':
2236 try:
2237 port = int(val)
2238 except ValueError:
2239 raise BadUsage
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002240 def ready(server):
Guido van Rossumbe19ed72007-02-09 05:37:30 +00002241 print('pydoc server ready at %s' % server.url)
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00002242 def stopped():
Guido van Rossumbe19ed72007-02-09 05:37:30 +00002243 print('pydoc server stopped')
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00002244 serve(port, ready, stopped)
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002245 return
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002246 if opt == '-w':
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002247 writing = 1
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002248
2249 if not args: raise BadUsage
2250 for arg in args:
Ka-Ping Yee45daeb02002-08-11 15:11:33 +00002251 if ispath(arg) and not os.path.exists(arg):
Guido van Rossumbe19ed72007-02-09 05:37:30 +00002252 print('file %r does not exist' % arg)
Ka-Ping Yee45daeb02002-08-11 15:11:33 +00002253 break
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002254 try:
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00002255 if ispath(arg) and os.path.isfile(arg):
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002256 arg = importfile(arg)
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00002257 if writing:
2258 if ispath(arg) and os.path.isdir(arg):
2259 writedocs(arg)
2260 else:
2261 writedoc(arg)
2262 else:
Martin v. Löwisb8c084e2003-06-14 09:03:46 +00002263 help.help(arg)
Guido van Rossumb940e112007-01-10 16:19:56 +00002264 except ErrorDuringImport as value:
Guido van Rossumbe19ed72007-02-09 05:37:30 +00002265 print(value)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002266
2267 except (getopt.error, BadUsage):
Martin v. Löwisf9b08b82003-10-31 13:05:21 +00002268 cmd = os.path.basename(sys.argv[0])
Guido van Rossumbe19ed72007-02-09 05:37:30 +00002269 print("""pydoc - the Python documentation tool
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002270
2271%s <name> ...
2272 Show text documentation on something. <name> may be the name of a
Martin v. Löwisb8c084e2003-06-14 09:03:46 +00002273 Python keyword, topic, function, module, or package, or a dotted
2274 reference to a class or function within a module or module in a
2275 package. If <name> contains a '%s', it is used as the path to a
2276 Python source file to document. If name is 'keywords', 'topics',
2277 or 'modules', a listing of these things is displayed.
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002278
2279%s -k <keyword>
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002280 Search for a keyword in the synopsis lines of all available modules.
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002281
2282%s -p <port>
2283 Start an HTTP server on the given port on the local machine.
2284
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002285%s -g
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +00002286 Pop up a graphical interface for finding and serving documentation.
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002287
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002288%s -w <name> ...
2289 Write out the HTML documentation for a module to a file in the current
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +00002290 directory. If <name> contains a '%s', it is treated as a filename; if
2291 it names a directory, documentation is written for all the contents.
Guido van Rossumbe19ed72007-02-09 05:37:30 +00002292""" % (cmd, os.sep, cmd, cmd, cmd, cmd, os.sep))
Ka-Ping Yee1d384632001-03-01 00:24:32 +00002293
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002294if __name__ == '__main__': cli()