blob: 57bd5e2298c5f53020e73b7d4c88e965d8859c15 [file] [log] [blame]
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001#!/usr/bin/env python
Amaury Forgeot d'Arc768db922008-04-24 21:00:04 +00002# -*- coding: latin-1 -*-
Ka-Ping Yee1d384632001-03-01 00:24:32 +00003"""Generate Python documentation in HTML or text for interactive use.
Ka-Ping Yeedd175342001-02-27 14:43:46 +00004
Ka-Ping Yeedd175342001-02-27 14:43:46 +00005In the Python interpreter, do "from pydoc import help" to provide online
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00006help. Calling help(thing) on a Python object documents the object.
7
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00008Or, at the shell command line outside of Python:
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00009
Ka-Ping Yee37f7b382001-03-23 00:12:53 +000010Run "pydoc <name>" to show documentation on something. <name> may be
11the name of a function, module, package, or a dotted reference to a
12class or function within a module or module in a package. If the
13argument contains a path segment delimiter (e.g. slash on Unix,
14backslash on Windows) it is treated as the path to a Python source file.
Ka-Ping Yee66efbc72001-03-01 13:55:20 +000015
Ka-Ping Yee37f7b382001-03-23 00:12:53 +000016Run "pydoc -k <keyword>" to search for a keyword in the synopsis lines
17of all available modules.
Ka-Ping Yee66efbc72001-03-01 13:55:20 +000018
Ka-Ping Yee37f7b382001-03-23 00:12:53 +000019Run "pydoc -p <port>" to start an HTTP server on a given port on the
20local machine to generate documentation web pages.
Ka-Ping Yee66efbc72001-03-01 13:55:20 +000021
Ka-Ping Yee37f7b382001-03-23 00:12:53 +000022For platforms without a command line, "pydoc -g" starts the HTTP server
23and also pops up a little window for controlling it.
24
25Run "pydoc -w <name>" to write out the HTML documentation for a module
26to a file named "<name>.html".
Skip Montanaro4997a692003-09-10 16:47:51 +000027
28Module docs for core modules are assumed to be in
29
Georg Brandl86def6c2008-01-21 20:36:10 +000030 http://docs.python.org/library/
Skip Montanaro4997a692003-09-10 16:47:51 +000031
32This can be overridden by setting the PYTHONDOCS environment variable
33to a different URL or to a local directory containing the Library
34Reference Manual pages.
Ka-Ping Yee66efbc72001-03-01 13:55:20 +000035"""
Ka-Ping Yeedd175342001-02-27 14:43:46 +000036
37__author__ = "Ka-Ping Yee <ping@lfw.org>"
Ka-Ping Yee6f3f9a42001-02-27 22:42:36 +000038__date__ = "26 February 2001"
Jeremy Hylton3e0055f2005-10-20 19:59:25 +000039
Ka-Ping Yee09d7d9a2001-02-27 22:43:48 +000040__version__ = "$Revision$"
Martin v. Löwis6fe8f192004-11-14 10:21:04 +000041__credits__ = """Guido van Rossum, for an excellent programming language.
Ka-Ping Yee5e2b1732001-02-27 23:35:09 +000042Tommy Burnette, the original creator of manpy.
Ka-Ping Yee6f3f9a42001-02-27 22:42:36 +000043Paul Prescod, for all his work on onlinehelp.
44Richard Chamberlain, for the first implementation of textdoc.
Raymond Hettingered2dbe32005-01-01 07:51:01 +000045"""
Ka-Ping Yeedd175342001-02-27 14:43:46 +000046
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +000047# Known bugs that can't be fixed here:
48# - imp.load_module() cannot be prevented from clobbering existing
49# loaded modules, so calling synopsis() on a binary module file
50# changes the contents of any existing module with the same name.
51# - If the __file__ attribute on a module is a relative path and
52# the current directory is changed with os.chdir(), an incorrect
53# path will be displayed.
Ka-Ping Yee66efbc72001-03-01 13:55:20 +000054
Georg Brandl1a3284e2007-12-02 09:40:06 +000055import sys, imp, os, re, inspect, builtins, pkgutil
Alexandre Vassalotti1f2ba4b2008-05-16 07:12:44 +000056from reprlib 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",
Georg Brandl86def6c2008-01-21 20:36:10 +0000344 "http://docs.python.org/library")
Skip Montanaro4997a692003-09-10 16:47:51 +0000345 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'))))):
353 if docloc.startswith("http://"):
Georg Brandl86def6c2008-01-21 20:36:10 +0000354 docloc = "%s/%s" % (docloc.rstrip("/"), object.__name__)
Skip Montanaro4997a692003-09-10 16:47:51 +0000355 else:
Georg Brandl86def6c2008-01-21 20:36:10 +0000356 docloc = os.path.join(docloc, object.__name__ + ".html")
Skip Montanaro4997a692003-09-10 16:47:51 +0000357 else:
358 docloc = None
359 return docloc
360
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000361# -------------------------------------------- HTML documentation generator
362
363class HTMLRepr(Repr):
364 """Class for safely making an HTML representation of a Python object."""
365 def __init__(self):
366 Repr.__init__(self)
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000367 self.maxlist = self.maxtuple = 20
368 self.maxdict = 10
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000369 self.maxstring = self.maxother = 100
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000370
371 def escape(self, text):
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +0000372 return replace(text, '&', '&amp;', '<', '&lt;', '>', '&gt;')
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000373
374 def repr(self, object):
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000375 return Repr.repr(self, object)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000376
377 def repr1(self, x, level):
Skip Montanaro0fe8fce2003-06-27 15:45:41 +0000378 if hasattr(type(x), '__name__'):
Neal Norwitz9d72bb42007-04-17 08:48:32 +0000379 methodname = 'repr_' + '_'.join(type(x).__name__.split())
Skip Montanaro0fe8fce2003-06-27 15:45:41 +0000380 if hasattr(self, methodname):
381 return getattr(self, methodname)(x, level)
382 return self.escape(cram(stripid(repr(x)), self.maxother))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000383
384 def repr_string(self, x, level):
Ka-Ping Yeea2fe1032001-03-02 01:19:14 +0000385 test = cram(x, self.maxstring)
386 testrepr = repr(test)
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +0000387 if '\\' in test and '\\' not in replace(testrepr, r'\\', ''):
Ka-Ping Yeea2fe1032001-03-02 01:19:14 +0000388 # Backslashes are only literal in the string and are never
389 # needed to make any special characters, so show a raw string.
390 return 'r' + testrepr[0] + self.escape(test) + testrepr[0]
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +0000391 return re.sub(r'((\\[\\abfnrtv\'"]|\\[0-9]..|\\x..|\\u....)+)',
Ka-Ping Yeea2fe1032001-03-02 01:19:14 +0000392 r'<font color="#c040c0">\1</font>',
393 self.escape(testrepr))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000394
Skip Montanarodf708782002-03-07 22:58:02 +0000395 repr_str = repr_string
396
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000397 def repr_instance(self, x, level):
398 try:
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000399 return self.escape(cram(stripid(repr(x)), self.maxstring))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000400 except:
401 return self.escape('<%s instance>' % x.__class__.__name__)
402
403 repr_unicode = repr_string
404
405class HTMLDoc(Doc):
406 """Formatter class for HTML documentation."""
407
408 # ------------------------------------------- HTML formatting utilities
409
410 _repr_instance = HTMLRepr()
411 repr = _repr_instance.repr
412 escape = _repr_instance.escape
413
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000414 def page(self, title, contents):
415 """Format an HTML page."""
416 return '''
Tim Peters59ed4482001-10-31 04:20:26 +0000417<!doctype html PUBLIC "-//W3C//DTD HTML 4.0 Transitional//EN">
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000418<html><head><title>Python: %s</title>
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000419</head><body bgcolor="#f0f0f8">
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000420%s
421</body></html>''' % (title, contents)
422
423 def heading(self, title, fgcol, bgcol, extras=''):
424 """Format a page heading."""
425 return '''
Tim Peters59ed4482001-10-31 04:20:26 +0000426<table width="100%%" cellspacing=0 cellpadding=2 border=0 summary="heading">
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000427<tr bgcolor="%s">
Tim Peters2306d242001-09-25 03:18:32 +0000428<td valign=bottom>&nbsp;<br>
429<font color="%s" face="helvetica, arial">&nbsp;<br>%s</font></td
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000430><td align=right valign=bottom
Ka-Ping Yee987ec902001-03-23 13:35:45 +0000431><font color="%s" face="helvetica, arial">%s</font></td></tr></table>
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000432 ''' % (bgcol, fgcol, title, fgcol, extras or '&nbsp;')
433
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000434 def section(self, title, fgcol, bgcol, contents, width=6,
435 prelude='', marginalia=None, gap='&nbsp;'):
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000436 """Format a section with a heading."""
437 if marginalia is None:
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +0000438 marginalia = '<tt>' + '&nbsp;' * width + '</tt>'
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000439 result = '''<p>
Tim Peters59ed4482001-10-31 04:20:26 +0000440<table width="100%%" cellspacing=0 cellpadding=2 border=0 summary="section">
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000441<tr bgcolor="%s">
Tim Peters2306d242001-09-25 03:18:32 +0000442<td colspan=3 valign=bottom>&nbsp;<br>
443<font color="%s" face="helvetica, arial">%s</font></td></tr>
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000444 ''' % (bgcol, fgcol, title)
445 if prelude:
446 result = result + '''
Ka-Ping Yee987ec902001-03-23 13:35:45 +0000447<tr bgcolor="%s"><td rowspan=2>%s</td>
448<td colspan=2>%s</td></tr>
449<tr><td>%s</td>''' % (bgcol, marginalia, prelude, gap)
450 else:
451 result = result + '''
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000452<tr><td bgcolor="%s">%s</td><td>%s</td>''' % (bgcol, marginalia, gap)
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000453
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +0000454 return result + '\n<td width="100%%">%s</td></tr></table>' % contents
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000455
456 def bigsection(self, title, *args):
457 """Format a section with a big heading."""
458 title = '<big><strong>%s</strong></big>' % title
Guido van Rossum68468eb2003-02-27 20:14:51 +0000459 return self.section(title, *args)
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000460
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000461 def preformat(self, text):
462 """Format literal preformatted text."""
Neal Norwitz9d72bb42007-04-17 08:48:32 +0000463 text = self.escape(text.expandtabs())
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +0000464 return replace(text, '\n\n', '\n \n', '\n\n', '\n \n',
465 ' ', '&nbsp;', '\n', '<br>\n')
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000466
467 def multicolumn(self, list, format, cols=4):
468 """Format a list of items into a multi-column list."""
469 result = ''
Guido van Rossumc1f779c2007-07-03 08:25:58 +0000470 rows = (len(list)+cols-1)//cols
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000471 for col in range(cols):
Guido van Rossumc1f779c2007-07-03 08:25:58 +0000472 result = result + '<td width="%d%%" valign=top>' % (100//cols)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000473 for i in range(rows*col, rows*col+rows):
474 if i < len(list):
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000475 result = result + format(list[i]) + '<br>\n'
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000476 result = result + '</td>'
Tim Peters59ed4482001-10-31 04:20:26 +0000477 return '<table width="100%%" summary="list"><tr>%s</tr></table>' % result
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000478
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000479 def grey(self, text): return '<font color="#909090">%s</font>' % text
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000480
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000481 def namelink(self, name, *dicts):
482 """Make a link for an identifier, given name-to-URL mappings."""
483 for dict in dicts:
Raymond Hettinger54f02222002-06-01 14:18:47 +0000484 if name in dict:
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000485 return '<a href="%s">%s</a>' % (dict[name], name)
486 return name
487
Ka-Ping Yeeb7a48302001-04-12 20:39:14 +0000488 def classlink(self, object, modname):
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000489 """Make a link for a class."""
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000490 name, module = object.__name__, sys.modules.get(object.__module__)
491 if hasattr(module, name) and getattr(module, name) is object:
Ka-Ping Yeeb7a48302001-04-12 20:39:14 +0000492 return '<a href="%s.html#%s">%s</a>' % (
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000493 module.__name__, name, classname(object, modname))
494 return classname(object, modname)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000495
496 def modulelink(self, object):
497 """Make a link for a module."""
498 return '<a href="%s.html">%s</a>' % (object.__name__, object.__name__)
499
Guido van Rossum1bc535d2007-05-15 18:46:22 +0000500 def modpkglink(self, modpkginfo):
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000501 """Make a link for a module or package to display in an index."""
Guido van Rossum1bc535d2007-05-15 18:46:22 +0000502 name, path, ispackage, shadowed = modpkginfo
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000503 if shadowed:
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000504 return self.grey(name)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000505 if path:
506 url = '%s.%s.html' % (path, name)
507 else:
508 url = '%s.html' % name
509 if ispackage:
510 text = '<strong>%s</strong>&nbsp;(package)' % name
511 else:
512 text = name
513 return '<a href="%s">%s</a>' % (url, text)
514
515 def markup(self, text, escape=None, funcs={}, classes={}, methods={}):
516 """Mark up some plain text, given a context of symbols to look for.
517 Each context dictionary maps object names to anchor names."""
518 escape = escape or self.escape
519 results = []
520 here = 0
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000521 pattern = re.compile(r'\b((http|ftp)://\S+[\w/]|'
522 r'RFC[- ]?(\d+)|'
Ka-Ping Yeef78a81b2001-03-27 08:13:42 +0000523 r'PEP[- ]?(\d+)|'
Neil Schemenauerd69711c2002-03-24 23:02:07 +0000524 r'(self\.)?(\w+))')
Guido van Rossum8ca162f2002-04-07 06:36:23 +0000525 while True:
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000526 match = pattern.search(text, here)
527 if not match: break
528 start, end = match.span()
529 results.append(escape(text[here:start]))
530
Ka-Ping Yeef78a81b2001-03-27 08:13:42 +0000531 all, scheme, rfc, pep, selfdot, name = match.groups()
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000532 if scheme:
Neil Schemenauercddc1a02002-03-24 23:11:21 +0000533 url = escape(all).replace('"', '&quot;')
534 results.append('<a href="%s">%s</a>' % (url, url))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000535 elif rfc:
Ka-Ping Yeef78a81b2001-03-27 08:13:42 +0000536 url = 'http://www.rfc-editor.org/rfc/rfc%d.txt' % int(rfc)
537 results.append('<a href="%s">%s</a>' % (url, escape(all)))
538 elif pep:
Christian Heimes2202f872008-02-06 14:31:34 +0000539 url = 'http://www.python.org/dev/peps/pep-%04d/' % int(pep)
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000540 results.append('<a href="%s">%s</a>' % (url, escape(all)))
541 elif text[end:end+1] == '(':
542 results.append(self.namelink(name, methods, funcs, classes))
543 elif selfdot:
544 results.append('self.<strong>%s</strong>' % name)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000545 else:
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000546 results.append(self.namelink(name, classes))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000547 here = end
548 results.append(escape(text[here:]))
Neal Norwitz9d72bb42007-04-17 08:48:32 +0000549 return ''.join(results)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000550
551 # ---------------------------------------------- type-specific routines
552
Ka-Ping Yeeb7a48302001-04-12 20:39:14 +0000553 def formattree(self, tree, modname, parent=None):
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000554 """Produce HTML for a class tree as given by inspect.getclasstree()."""
555 result = ''
556 for entry in tree:
557 if type(entry) is type(()):
558 c, bases = entry
Tim Peters2306d242001-09-25 03:18:32 +0000559 result = result + '<dt><font face="helvetica, arial">'
Ka-Ping Yeeb7a48302001-04-12 20:39:14 +0000560 result = result + self.classlink(c, modname)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000561 if bases and bases != (parent,):
562 parents = []
563 for base in bases:
Ka-Ping Yeeb7a48302001-04-12 20:39:14 +0000564 parents.append(self.classlink(base, modname))
Neal Norwitz9d72bb42007-04-17 08:48:32 +0000565 result = result + '(' + ', '.join(parents) + ')'
Tim Peters2306d242001-09-25 03:18:32 +0000566 result = result + '\n</font></dt>'
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000567 elif type(entry) is type([]):
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000568 result = result + '<dd>\n%s</dd>\n' % self.formattree(
Ka-Ping Yeeb7a48302001-04-12 20:39:14 +0000569 entry, modname, c)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000570 return '<dl>\n%s</dl>\n' % result
571
Tim Peters8dd7ade2001-10-18 19:56:17 +0000572 def docmodule(self, object, name=None, mod=None, *ignored):
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000573 """Produce HTML documentation for a module object."""
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000574 name = object.__name__ # ignore the passed-in name
Skip Montanaroa5616d22004-06-11 04:46:12 +0000575 try:
576 all = object.__all__
577 except AttributeError:
578 all = None
Neal Norwitz9d72bb42007-04-17 08:48:32 +0000579 parts = name.split('.')
Ka-Ping Yee66efbc72001-03-01 13:55:20 +0000580 links = []
581 for i in range(len(parts)-1):
582 links.append(
583 '<a href="%s.html"><font color="#ffffff">%s</font></a>' %
Neal Norwitz9d72bb42007-04-17 08:48:32 +0000584 ('.'.join(parts[:i+1]), parts[i]))
585 linkedname = '.'.join(links + parts[-1:])
Ka-Ping Yee66efbc72001-03-01 13:55:20 +0000586 head = '<big><big><strong>%s</strong></big></big>' % linkedname
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000587 try:
Ka-Ping Yee239432a2001-03-02 02:45:08 +0000588 path = inspect.getabsfile(object)
Ka-Ping Yee6191a232001-04-13 15:00:27 +0000589 url = path
590 if sys.platform == 'win32':
591 import nturl2path
592 url = nturl2path.pathname2url(path)
593 filelink = '<a href="file:%s">%s</a>' % (url, path)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000594 except TypeError:
595 filelink = '(built-in)'
Ka-Ping Yee6f3f9a42001-02-27 22:42:36 +0000596 info = []
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000597 if hasattr(object, '__version__'):
Ka-Ping Yee6f3f9a42001-02-27 22:42:36 +0000598 version = str(object.__version__)
Ka-Ping Yee40c49912001-02-27 22:46:01 +0000599 if version[:11] == '$' + 'Revision: ' and version[-1:] == '$':
Neal Norwitz9d72bb42007-04-17 08:48:32 +0000600 version = version[11:-1].strip()
Ka-Ping Yee1d384632001-03-01 00:24:32 +0000601 info.append('version %s' % self.escape(version))
Ka-Ping Yee6f3f9a42001-02-27 22:42:36 +0000602 if hasattr(object, '__date__'):
603 info.append(self.escape(str(object.__date__)))
604 if info:
Neal Norwitz9d72bb42007-04-17 08:48:32 +0000605 head = head + ' (%s)' % ', '.join(info)
Skip Montanaro4997a692003-09-10 16:47:51 +0000606 docloc = self.getdocloc(object)
607 if docloc is not None:
608 docloc = '<br><a href="%(docloc)s">Module Docs</a>' % locals()
609 else:
610 docloc = ''
Ka-Ping Yee66efbc72001-03-01 13:55:20 +0000611 result = self.heading(
Skip Montanaro4997a692003-09-10 16:47:51 +0000612 head, '#ffffff', '#7799ee',
613 '<a href=".">index</a><br>' + filelink + docloc)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000614
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000615 modules = inspect.getmembers(object, inspect.ismodule)
616
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000617 classes, cdict = [], {}
618 for key, value in inspect.getmembers(object, inspect.isclass):
Johannes Gijsbers4c11f602004-08-30 14:13:04 +0000619 # if __all__ exists, believe it. Otherwise use old heuristic.
620 if (all is not None or
621 (inspect.getmodule(value) or object) is object):
Skip Montanaroa5616d22004-06-11 04:46:12 +0000622 if visiblename(key, all):
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000623 classes.append((key, value))
624 cdict[key] = cdict[value] = '#' + key
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000625 for key, value in classes:
626 for base in value.__bases__:
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000627 key, modname = base.__name__, base.__module__
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000628 module = sys.modules.get(modname)
629 if modname != name and module and hasattr(module, key):
630 if getattr(module, key) is base:
Raymond Hettinger54f02222002-06-01 14:18:47 +0000631 if not key in cdict:
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000632 cdict[key] = cdict[base] = modname + '.html#' + key
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000633 funcs, fdict = [], {}
634 for key, value in inspect.getmembers(object, inspect.isroutine):
Johannes Gijsbers4c11f602004-08-30 14:13:04 +0000635 # if __all__ exists, believe it. Otherwise use old heuristic.
636 if (all is not None or
637 inspect.isbuiltin(value) or inspect.getmodule(value) is object):
Skip Montanaroa5616d22004-06-11 04:46:12 +0000638 if visiblename(key, all):
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000639 funcs.append((key, value))
640 fdict[key] = '#-' + key
641 if inspect.isfunction(value): fdict[value] = fdict[key]
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000642 data = []
643 for key, value in inspect.getmembers(object, isdata):
Skip Montanaroa5616d22004-06-11 04:46:12 +0000644 if visiblename(key, all):
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000645 data.append((key, value))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000646
647 doc = self.markup(getdoc(object), self.preformat, fdict, cdict)
648 doc = doc and '<tt>%s</tt>' % doc
Tim Peters2306d242001-09-25 03:18:32 +0000649 result = result + '<p>%s</p>\n' % doc
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000650
651 if hasattr(object, '__path__'):
652 modpkgs = []
Thomas Wouters49fd7fa2006-04-21 10:40:58 +0000653 for importer, modname, ispkg in pkgutil.iter_modules(object.__path__):
654 modpkgs.append((modname, name, ispkg, 0))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000655 modpkgs.sort()
656 contents = self.multicolumn(modpkgs, self.modpkglink)
657 result = result + self.bigsection(
658 'Package Contents', '#ffffff', '#aa55cc', contents)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000659 elif modules:
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000660 contents = self.multicolumn(
Guido van Rossum1bc535d2007-05-15 18:46:22 +0000661 modules, lambda t: self.modulelink(t[1]))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000662 result = result + self.bigsection(
Christian Heimes7131fd92008-02-19 14:21:46 +0000663 'Modules', '#ffffff', '#aa55cc', contents)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000664
665 if classes:
Guido van Rossum1bc535d2007-05-15 18:46:22 +0000666 classlist = [value for (key, value) in classes]
Ka-Ping Yeeb7a48302001-04-12 20:39:14 +0000667 contents = [
668 self.formattree(inspect.getclasstree(classlist, 1), name)]
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000669 for key, value in classes:
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +0000670 contents.append(self.document(value, key, name, fdict, cdict))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000671 result = result + self.bigsection(
Neal Norwitz9d72bb42007-04-17 08:48:32 +0000672 'Classes', '#ffffff', '#ee77aa', ' '.join(contents))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000673 if funcs:
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000674 contents = []
675 for key, value in funcs:
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +0000676 contents.append(self.document(value, key, name, fdict, cdict))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000677 result = result + self.bigsection(
Neal Norwitz9d72bb42007-04-17 08:48:32 +0000678 'Functions', '#ffffff', '#eeaa77', ' '.join(contents))
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000679 if data:
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000680 contents = []
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000681 for key, value in data:
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000682 contents.append(self.document(value, key))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000683 result = result + self.bigsection(
Neal Norwitz9d72bb42007-04-17 08:48:32 +0000684 'Data', '#ffffff', '#55aa55', '<br>\n'.join(contents))
Ka-Ping Yee6f3f9a42001-02-27 22:42:36 +0000685 if hasattr(object, '__author__'):
686 contents = self.markup(str(object.__author__), self.preformat)
687 result = result + self.bigsection(
688 'Author', '#ffffff', '#7799ee', contents)
Ka-Ping Yee6f3f9a42001-02-27 22:42:36 +0000689 if hasattr(object, '__credits__'):
690 contents = self.markup(str(object.__credits__), self.preformat)
691 result = result + self.bigsection(
692 'Credits', '#ffffff', '#7799ee', contents)
693
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000694 return result
695
Tim Peters8dd7ade2001-10-18 19:56:17 +0000696 def docclass(self, object, name=None, mod=None, funcs={}, classes={},
697 *ignored):
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000698 """Produce HTML documentation for a class object."""
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000699 realname = object.__name__
700 name = name or realname
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000701 bases = object.__bases__
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000702
Tim Petersb47879b2001-09-24 04:47:19 +0000703 contents = []
704 push = contents.append
705
Tim Petersfa26f7c2001-09-24 08:05:11 +0000706 # Cute little class to pump out a horizontal rule between sections.
707 class HorizontalRule:
708 def __init__(self):
709 self.needone = 0
710 def maybe(self):
711 if self.needone:
712 push('<hr>\n')
713 self.needone = 1
714 hr = HorizontalRule()
715
Tim Petersc86f6ca2001-09-26 21:31:51 +0000716 # List the mro, if non-trivial.
Raymond Hettinger756b3f32004-01-29 06:37:52 +0000717 mro = deque(inspect.getmro(object))
Tim Petersc86f6ca2001-09-26 21:31:51 +0000718 if len(mro) > 2:
719 hr.maybe()
720 push('<dl><dt>Method resolution order:</dt>\n')
721 for base in mro:
722 push('<dd>%s</dd>\n' % self.classlink(base,
723 object.__module__))
724 push('</dl>\n')
725
Tim Petersb47879b2001-09-24 04:47:19 +0000726 def spill(msg, attrs, predicate):
Tim Petersfa26f7c2001-09-24 08:05:11 +0000727 ok, attrs = _split_list(attrs, predicate)
Tim Petersb47879b2001-09-24 04:47:19 +0000728 if ok:
Tim Petersfa26f7c2001-09-24 08:05:11 +0000729 hr.maybe()
Tim Petersb47879b2001-09-24 04:47:19 +0000730 push(msg)
731 for name, kind, homecls, value in ok:
732 push(self.document(getattr(object, name), name, mod,
733 funcs, classes, mdict, object))
734 push('\n')
735 return attrs
736
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +0000737 def spilldescriptors(msg, attrs, predicate):
Tim Petersfa26f7c2001-09-24 08:05:11 +0000738 ok, attrs = _split_list(attrs, predicate)
Tim Petersb47879b2001-09-24 04:47:19 +0000739 if ok:
Tim Petersfa26f7c2001-09-24 08:05:11 +0000740 hr.maybe()
Tim Petersb47879b2001-09-24 04:47:19 +0000741 push(msg)
742 for name, kind, homecls, value in ok:
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +0000743 push(self._docdescriptor(name, value, mod))
Tim Petersb47879b2001-09-24 04:47:19 +0000744 return attrs
745
Tim Petersfa26f7c2001-09-24 08:05:11 +0000746 def spilldata(msg, attrs, predicate):
747 ok, attrs = _split_list(attrs, predicate)
Tim Petersb47879b2001-09-24 04:47:19 +0000748 if ok:
Tim Petersfa26f7c2001-09-24 08:05:11 +0000749 hr.maybe()
Tim Petersb47879b2001-09-24 04:47:19 +0000750 push(msg)
751 for name, kind, homecls, value in ok:
752 base = self.docother(getattr(object, name), name, mod)
Guido van Rossumd59da4b2007-05-22 18:11:13 +0000753 if hasattr(value, '__call__') or inspect.isdatadescriptor(value):
Guido van Rossum5e355b22002-05-21 20:56:15 +0000754 doc = getattr(value, "__doc__", None)
755 else:
756 doc = None
Tim Petersb47879b2001-09-24 04:47:19 +0000757 if doc is None:
758 push('<dl><dt>%s</dl>\n' % base)
759 else:
760 doc = self.markup(getdoc(value), self.preformat,
761 funcs, classes, mdict)
Tim Peters2306d242001-09-25 03:18:32 +0000762 doc = '<dd><tt>%s</tt>' % doc
Tim Petersb47879b2001-09-24 04:47:19 +0000763 push('<dl><dt>%s%s</dl>\n' % (base, doc))
764 push('\n')
765 return attrs
766
Guido van Rossum1bc535d2007-05-15 18:46:22 +0000767 attrs = [(name, kind, cls, value)
768 for name, kind, cls, value in classify_class_attrs(object)
769 if visiblename(name)]
770
Tim Petersb47879b2001-09-24 04:47:19 +0000771 mdict = {}
772 for key, kind, homecls, value in attrs:
773 mdict[key] = anchor = '#' + name + '-' + key
774 value = getattr(object, key)
775 try:
776 # The value may not be hashable (e.g., a data attr with
777 # a dict or list value).
778 mdict[value] = anchor
779 except TypeError:
780 pass
781
Tim Petersfa26f7c2001-09-24 08:05:11 +0000782 while attrs:
Tim Peters351e3622001-09-27 03:29:51 +0000783 if mro:
Raymond Hettinger756b3f32004-01-29 06:37:52 +0000784 thisclass = mro.popleft()
Tim Peters351e3622001-09-27 03:29:51 +0000785 else:
786 thisclass = attrs[0][2]
Tim Petersfa26f7c2001-09-24 08:05:11 +0000787 attrs, inherited = _split_list(attrs, lambda t: t[2] is thisclass)
788
Georg Brandl1a3284e2007-12-02 09:40:06 +0000789 if thisclass is builtins.object:
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000790 attrs = inherited
791 continue
792 elif thisclass is object:
793 tag = 'defined here'
Tim Petersb47879b2001-09-24 04:47:19 +0000794 else:
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000795 tag = 'inherited from %s' % self.classlink(thisclass,
796 object.__module__)
Tim Petersb47879b2001-09-24 04:47:19 +0000797 tag += ':<br>\n'
798
799 # Sort attrs by name.
Raymond Hettingerd4cb56d2008-01-30 02:55:10 +0000800 attrs.sort(key=lambda t: t[0])
Tim Petersb47879b2001-09-24 04:47:19 +0000801
802 # Pump out the attrs, segregated by kind.
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000803 attrs = spill('Methods %s' % tag, attrs,
Tim Petersb47879b2001-09-24 04:47:19 +0000804 lambda t: t[1] == 'method')
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000805 attrs = spill('Class methods %s' % tag, attrs,
Tim Petersb47879b2001-09-24 04:47:19 +0000806 lambda t: t[1] == 'class method')
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000807 attrs = spill('Static methods %s' % tag, attrs,
Tim Petersb47879b2001-09-24 04:47:19 +0000808 lambda t: t[1] == 'static method')
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +0000809 attrs = spilldescriptors('Data descriptors %s' % tag, attrs,
810 lambda t: t[1] == 'data descriptor')
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000811 attrs = spilldata('Data and other attributes %s' % tag, attrs,
Tim Petersfa26f7c2001-09-24 08:05:11 +0000812 lambda t: t[1] == 'data')
Tim Petersb47879b2001-09-24 04:47:19 +0000813 assert attrs == []
Tim Peters351e3622001-09-27 03:29:51 +0000814 attrs = inherited
Tim Petersb47879b2001-09-24 04:47:19 +0000815
816 contents = ''.join(contents)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000817
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000818 if name == realname:
819 title = '<a name="%s">class <strong>%s</strong></a>' % (
820 name, realname)
821 else:
822 title = '<strong>%s</strong> = <a name="%s">class %s</a>' % (
823 name, name, realname)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000824 if bases:
825 parents = []
826 for base in bases:
Ka-Ping Yeeb7a48302001-04-12 20:39:14 +0000827 parents.append(self.classlink(base, object.__module__))
Neal Norwitz9d72bb42007-04-17 08:48:32 +0000828 title = title + '(%s)' % ', '.join(parents)
Tim Peters2306d242001-09-25 03:18:32 +0000829 doc = self.markup(getdoc(object), self.preformat, funcs, classes, mdict)
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000830 doc = doc and '<tt>%s<br>&nbsp;</tt>' % doc
Tim Petersc86f6ca2001-09-26 21:31:51 +0000831
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000832 return self.section(title, '#000000', '#ffc8d8', contents, 3, doc)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000833
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000834 def formatvalue(self, object):
835 """Format an argument default value as text."""
Tim Peters2306d242001-09-25 03:18:32 +0000836 return self.grey('=' + self.repr(object))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000837
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +0000838 def docroutine(self, object, name=None, mod=None,
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000839 funcs={}, classes={}, methods={}, cl=None):
Ka-Ping Yee66efbc72001-03-01 13:55:20 +0000840 """Produce HTML documentation for a function or method object."""
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000841 realname = object.__name__
842 name = name or realname
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000843 anchor = (cl and cl.__name__ or '') + '-' + name
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000844 note = ''
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000845 skipdocs = 0
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000846 if inspect.ismethod(object):
Christian Heimesff737952007-11-27 10:40:20 +0000847 imclass = object.__self__.__class__
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000848 if cl:
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +0000849 if imclass is not cl:
Ka-Ping Yeeb7a48302001-04-12 20:39:14 +0000850 note = ' from ' + self.classlink(imclass, mod)
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000851 else:
Christian Heimesff737952007-11-27 10:40:20 +0000852 if object.__self__ is not None:
Ka-Ping Yeeb7a48302001-04-12 20:39:14 +0000853 note = ' method of %s instance' % self.classlink(
Christian Heimesff737952007-11-27 10:40:20 +0000854 object.__self__.__class__, mod)
Ka-Ping Yeeb7a48302001-04-12 20:39:14 +0000855 else:
856 note = ' unbound %s method' % self.classlink(imclass,mod)
Christian Heimesff737952007-11-27 10:40:20 +0000857 object = object.__func__
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000858
859 if name == realname:
860 title = '<a name="%s"><strong>%s</strong></a>' % (anchor, realname)
861 else:
Raymond Hettinger54f02222002-06-01 14:18:47 +0000862 if (cl and realname in cl.__dict__ and
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000863 cl.__dict__[realname] is object):
Ka-Ping Yeee280c062001-03-23 14:05:53 +0000864 reallink = '<a href="#%s">%s</a>' % (
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000865 cl.__name__ + '-' + realname, realname)
866 skipdocs = 1
867 else:
868 reallink = realname
869 title = '<a name="%s"><strong>%s</strong></a> = %s' % (
870 anchor, name, reallink)
Tim Peters4bcfa312001-09-20 06:08:24 +0000871 if inspect.isfunction(object):
Guido van Rossum2e65f892007-02-28 22:03:49 +0000872 args, varargs, kwonlyargs, kwdefaults, varkw, defaults, ann = \
873 inspect.getfullargspec(object)
Ka-Ping Yee66efbc72001-03-01 13:55:20 +0000874 argspec = inspect.formatargspec(
Guido van Rossum2e65f892007-02-28 22:03:49 +0000875 args, varargs, kwonlyargs, kwdefaults, varkw, defaults, ann,
876 formatvalue=self.formatvalue,
877 formatannotation=inspect.formatannotationrelativeto(object))
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000878 if realname == '<lambda>':
Tim Peters59ed4482001-10-31 04:20:26 +0000879 title = '<strong>%s</strong> <em>lambda</em> ' % name
Guido van Rossum2e65f892007-02-28 22:03:49 +0000880 # XXX lambda's won't usually have func_annotations['return']
881 # since the syntax doesn't support but it is possible.
882 # So removing parentheses isn't truly safe.
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000883 argspec = argspec[1:-1] # remove parentheses
Tim Peters4bcfa312001-09-20 06:08:24 +0000884 else:
885 argspec = '(...)'
Ka-Ping Yee66efbc72001-03-01 13:55:20 +0000886
Tim Peters2306d242001-09-25 03:18:32 +0000887 decl = title + argspec + (note and self.grey(
888 '<font face="helvetica, arial">%s</font>' % note))
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000889
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000890 if skipdocs:
Tim Peters2306d242001-09-25 03:18:32 +0000891 return '<dl><dt>%s</dt></dl>\n' % decl
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000892 else:
893 doc = self.markup(
894 getdoc(object), self.preformat, funcs, classes, methods)
Tim Peters2306d242001-09-25 03:18:32 +0000895 doc = doc and '<dd><tt>%s</tt></dd>' % doc
896 return '<dl><dt>%s</dt>%s</dl>\n' % (decl, doc)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000897
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +0000898 def _docdescriptor(self, name, value, mod):
Johannes Gijsbers8de645a2004-11-07 19:16:05 +0000899 results = []
900 push = results.append
901
902 if name:
903 push('<dl><dt><strong>%s</strong></dt>\n' % name)
904 if value.__doc__ is not None:
Ka-Ping Yeebba6acc2005-02-19 22:58:26 +0000905 doc = self.markup(getdoc(value), self.preformat)
Johannes Gijsbers8de645a2004-11-07 19:16:05 +0000906 push('<dd><tt>%s</tt></dd>\n' % doc)
Johannes Gijsbers8de645a2004-11-07 19:16:05 +0000907 push('</dl>\n')
908
909 return ''.join(results)
910
911 def docproperty(self, object, name=None, mod=None, cl=None):
912 """Produce html documentation for a property."""
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +0000913 return self._docdescriptor(name, object, mod)
Johannes Gijsbers8de645a2004-11-07 19:16:05 +0000914
Tim Peters8dd7ade2001-10-18 19:56:17 +0000915 def docother(self, object, name=None, mod=None, *ignored):
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000916 """Produce HTML documentation for a data object."""
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +0000917 lhs = name and '<strong>%s</strong> = ' % name or ''
918 return lhs + self.repr(object)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000919
Thomas Wouters0e3f5912006-08-11 14:57:12 +0000920 def docdata(self, object, name=None, mod=None, cl=None):
921 """Produce html documentation for a data descriptor."""
922 return self._docdescriptor(name, object, mod)
923
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000924 def index(self, dir, shadowed=None):
925 """Generate an HTML index for a directory of modules."""
926 modpkgs = []
927 if shadowed is None: shadowed = {}
Thomas Wouters49fd7fa2006-04-21 10:40:58 +0000928 for importer, name, ispkg in pkgutil.iter_modules([dir]):
929 modpkgs.append((name, '', ispkg, name in shadowed))
930 shadowed[name] = 1
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000931
932 modpkgs.sort()
933 contents = self.multicolumn(modpkgs, self.modpkglink)
934 return self.bigsection(dir, '#ffffff', '#ee77aa', contents)
935
936# -------------------------------------------- text documentation generator
937
938class TextRepr(Repr):
939 """Class for safely making a text representation of a Python object."""
940 def __init__(self):
941 Repr.__init__(self)
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000942 self.maxlist = self.maxtuple = 20
943 self.maxdict = 10
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000944 self.maxstring = self.maxother = 100
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000945
946 def repr1(self, x, level):
Skip Montanaro0fe8fce2003-06-27 15:45:41 +0000947 if hasattr(type(x), '__name__'):
Neal Norwitz9d72bb42007-04-17 08:48:32 +0000948 methodname = 'repr_' + '_'.join(type(x).__name__.split())
Skip Montanaro0fe8fce2003-06-27 15:45:41 +0000949 if hasattr(self, methodname):
950 return getattr(self, methodname)(x, level)
951 return cram(stripid(repr(x)), self.maxother)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000952
Ka-Ping Yeea2fe1032001-03-02 01:19:14 +0000953 def repr_string(self, x, level):
954 test = cram(x, self.maxstring)
955 testrepr = repr(test)
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +0000956 if '\\' in test and '\\' not in replace(testrepr, r'\\', ''):
Ka-Ping Yeea2fe1032001-03-02 01:19:14 +0000957 # Backslashes are only literal in the string and are never
958 # needed to make any special characters, so show a raw string.
959 return 'r' + testrepr[0] + test + testrepr[0]
960 return testrepr
961
Skip Montanarodf708782002-03-07 22:58:02 +0000962 repr_str = repr_string
963
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000964 def repr_instance(self, x, level):
965 try:
Ka-Ping Yee1d384632001-03-01 00:24:32 +0000966 return cram(stripid(repr(x)), self.maxstring)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000967 except:
968 return '<%s instance>' % x.__class__.__name__
969
970class TextDoc(Doc):
971 """Formatter class for text documentation."""
972
973 # ------------------------------------------- text formatting utilities
974
975 _repr_instance = TextRepr()
976 repr = _repr_instance.repr
977
978 def bold(self, text):
979 """Format a string in bold by overstriking."""
Neal Norwitz9d72bb42007-04-17 08:48:32 +0000980 return ''.join(map(lambda ch: ch + '\b' + ch, text))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000981
982 def indent(self, text, prefix=' '):
983 """Indent text by prepending a given prefix to each line."""
984 if not text: return ''
Collin Winter72e110c2007-07-17 00:27:30 +0000985 lines = [prefix + line for line in text.split('\n')]
Neal Norwitz9d72bb42007-04-17 08:48:32 +0000986 if lines: lines[-1] = lines[-1].rstrip()
987 return '\n'.join(lines)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000988
989 def section(self, title, contents):
990 """Format a section with a given heading."""
Neal Norwitz9d72bb42007-04-17 08:48:32 +0000991 clean_contents = self.indent(contents).rstrip()
992 return self.bold(title) + '\n' + clean_contents + '\n\n'
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000993
994 # ---------------------------------------------- type-specific routines
995
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000996 def formattree(self, tree, modname, parent=None, prefix=''):
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000997 """Render in text a class tree as returned by inspect.getclasstree()."""
998 result = ''
999 for entry in tree:
1000 if type(entry) is type(()):
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001001 c, bases = entry
1002 result = result + prefix + classname(c, modname)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001003 if bases and bases != (parent,):
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001004 parents = map(lambda c, m=modname: classname(c, m), bases)
Neal Norwitz9d72bb42007-04-17 08:48:32 +00001005 result = result + '(%s)' % ', '.join(parents)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001006 result = result + '\n'
1007 elif type(entry) is type([]):
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001008 result = result + self.formattree(
1009 entry, modname, c, prefix + ' ')
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001010 return result
1011
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001012 def docmodule(self, object, name=None, mod=None):
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001013 """Produce text documentation for a given module object."""
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001014 name = object.__name__ # ignore the passed-in name
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +00001015 synop, desc = splitdoc(getdoc(object))
1016 result = self.section('NAME', name + (synop and ' - ' + synop))
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001017
1018 try:
Skip Montanaroa5616d22004-06-11 04:46:12 +00001019 all = object.__all__
1020 except AttributeError:
1021 all = None
1022
1023 try:
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001024 file = inspect.getabsfile(object)
1025 except TypeError:
1026 file = '(built-in)'
Ka-Ping Yee239432a2001-03-02 02:45:08 +00001027 result = result + self.section('FILE', file)
Skip Montanaro4997a692003-09-10 16:47:51 +00001028
1029 docloc = self.getdocloc(object)
1030 if docloc is not None:
1031 result = result + self.section('MODULE DOCS', docloc)
1032
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +00001033 if desc:
1034 result = result + self.section('DESCRIPTION', desc)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001035
1036 classes = []
1037 for key, value in inspect.getmembers(object, inspect.isclass):
Johannes Gijsbers4c11f602004-08-30 14:13:04 +00001038 # if __all__ exists, believe it. Otherwise use old heuristic.
1039 if (all is not None
1040 or (inspect.getmodule(value) or object) is object):
Skip Montanaroa5616d22004-06-11 04:46:12 +00001041 if visiblename(key, all):
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +00001042 classes.append((key, value))
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001043 funcs = []
1044 for key, value in inspect.getmembers(object, inspect.isroutine):
Johannes Gijsbers4c11f602004-08-30 14:13:04 +00001045 # if __all__ exists, believe it. Otherwise use old heuristic.
1046 if (all is not None or
1047 inspect.isbuiltin(value) or inspect.getmodule(value) is object):
Skip Montanaroa5616d22004-06-11 04:46:12 +00001048 if visiblename(key, all):
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +00001049 funcs.append((key, value))
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001050 data = []
1051 for key, value in inspect.getmembers(object, isdata):
Skip Montanaroa5616d22004-06-11 04:46:12 +00001052 if visiblename(key, all):
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001053 data.append((key, value))
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001054
Christian Heimes1af737c2008-01-23 08:24:23 +00001055 modpkgs = []
1056 modpkgs_names = set()
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001057 if hasattr(object, '__path__'):
Thomas Wouters49fd7fa2006-04-21 10:40:58 +00001058 for importer, modname, ispkg in pkgutil.iter_modules(object.__path__):
Christian Heimes1af737c2008-01-23 08:24:23 +00001059 modpkgs_names.add(modname)
Thomas Wouters49fd7fa2006-04-21 10:40:58 +00001060 if ispkg:
1061 modpkgs.append(modname + ' (package)')
1062 else:
1063 modpkgs.append(modname)
1064
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001065 modpkgs.sort()
1066 result = result + self.section(
Neal Norwitz9d72bb42007-04-17 08:48:32 +00001067 'PACKAGE CONTENTS', '\n'.join(modpkgs))
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001068
Christian Heimes1af737c2008-01-23 08:24:23 +00001069 # Detect submodules as sometimes created by C extensions
1070 submodules = []
1071 for key, value in inspect.getmembers(object, inspect.ismodule):
1072 if value.__name__.startswith(name + '.') and key not in modpkgs_names:
1073 submodules.append(key)
1074 if submodules:
1075 submodules.sort()
1076 result = result + self.section(
Amaury Forgeot d'Arc768db922008-04-24 21:00:04 +00001077 'SUBMODULES', '\n'.join(submodules))
Christian Heimes1af737c2008-01-23 08:24:23 +00001078
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001079 if classes:
Guido van Rossum1bc535d2007-05-15 18:46:22 +00001080 classlist = [value for key, value in classes]
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001081 contents = [self.formattree(
1082 inspect.getclasstree(classlist, 1), name)]
1083 for key, value in classes:
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001084 contents.append(self.document(value, key, name))
Neal Norwitz9d72bb42007-04-17 08:48:32 +00001085 result = result + self.section('CLASSES', '\n'.join(contents))
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001086
1087 if funcs:
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001088 contents = []
1089 for key, value in funcs:
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001090 contents.append(self.document(value, key, name))
Neal Norwitz9d72bb42007-04-17 08:48:32 +00001091 result = result + self.section('FUNCTIONS', '\n'.join(contents))
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001092
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001093 if data:
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001094 contents = []
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001095 for key, value in data:
Georg Brandl8b813db2005-10-01 16:32:31 +00001096 contents.append(self.docother(value, key, name, maxlen=70))
Neal Norwitz9d72bb42007-04-17 08:48:32 +00001097 result = result + self.section('DATA', '\n'.join(contents))
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001098
1099 if hasattr(object, '__version__'):
1100 version = str(object.__version__)
Ka-Ping Yee1d384632001-03-01 00:24:32 +00001101 if version[:11] == '$' + 'Revision: ' and version[-1:] == '$':
Neal Norwitz9d72bb42007-04-17 08:48:32 +00001102 version = version[11:-1].strip()
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001103 result = result + self.section('VERSION', version)
Ka-Ping Yee6f3f9a42001-02-27 22:42:36 +00001104 if hasattr(object, '__date__'):
1105 result = result + self.section('DATE', str(object.__date__))
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001106 if hasattr(object, '__author__'):
Ka-Ping Yee6f3f9a42001-02-27 22:42:36 +00001107 result = result + self.section('AUTHOR', str(object.__author__))
1108 if hasattr(object, '__credits__'):
1109 result = result + self.section('CREDITS', str(object.__credits__))
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001110 return result
1111
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001112 def docclass(self, object, name=None, mod=None):
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001113 """Produce text documentation for a given class object."""
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001114 realname = object.__name__
1115 name = name or realname
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001116 bases = object.__bases__
1117
Tim Petersc86f6ca2001-09-26 21:31:51 +00001118 def makename(c, m=object.__module__):
1119 return classname(c, m)
1120
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001121 if name == realname:
1122 title = 'class ' + self.bold(realname)
1123 else:
1124 title = self.bold(name) + ' = class ' + realname
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001125 if bases:
Ka-Ping Yee3bda8792001-03-23 13:17:50 +00001126 parents = map(makename, bases)
Neal Norwitz9d72bb42007-04-17 08:48:32 +00001127 title = title + '(%s)' % ', '.join(parents)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001128
1129 doc = getdoc(object)
Tim Peters28355492001-09-23 21:29:55 +00001130 contents = doc and [doc + '\n'] or []
1131 push = contents.append
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001132
Tim Petersc86f6ca2001-09-26 21:31:51 +00001133 # List the mro, if non-trivial.
Raymond Hettinger756b3f32004-01-29 06:37:52 +00001134 mro = deque(inspect.getmro(object))
Tim Petersc86f6ca2001-09-26 21:31:51 +00001135 if len(mro) > 2:
1136 push("Method resolution order:")
1137 for base in mro:
1138 push(' ' + makename(base))
1139 push('')
1140
Tim Petersf4aad8e2001-09-24 22:40:47 +00001141 # Cute little class to pump out a horizontal rule between sections.
1142 class HorizontalRule:
1143 def __init__(self):
1144 self.needone = 0
1145 def maybe(self):
1146 if self.needone:
1147 push('-' * 70)
1148 self.needone = 1
1149 hr = HorizontalRule()
1150
Tim Peters28355492001-09-23 21:29:55 +00001151 def spill(msg, attrs, predicate):
Tim Petersfa26f7c2001-09-24 08:05:11 +00001152 ok, attrs = _split_list(attrs, predicate)
Tim Peters28355492001-09-23 21:29:55 +00001153 if ok:
Tim Petersf4aad8e2001-09-24 22:40:47 +00001154 hr.maybe()
Tim Peters28355492001-09-23 21:29:55 +00001155 push(msg)
1156 for name, kind, homecls, value in ok:
1157 push(self.document(getattr(object, name),
1158 name, mod, object))
1159 return attrs
1160
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +00001161 def spilldescriptors(msg, attrs, predicate):
Tim Petersfa26f7c2001-09-24 08:05:11 +00001162 ok, attrs = _split_list(attrs, predicate)
Tim Peters28355492001-09-23 21:29:55 +00001163 if ok:
Tim Petersf4aad8e2001-09-24 22:40:47 +00001164 hr.maybe()
Tim Peters28355492001-09-23 21:29:55 +00001165 push(msg)
1166 for name, kind, homecls, value in ok:
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +00001167 push(self._docdescriptor(name, value, mod))
Tim Peters28355492001-09-23 21:29:55 +00001168 return attrs
Tim Petersb47879b2001-09-24 04:47:19 +00001169
Tim Petersfa26f7c2001-09-24 08:05:11 +00001170 def spilldata(msg, attrs, predicate):
1171 ok, attrs = _split_list(attrs, predicate)
Tim Peters28355492001-09-23 21:29:55 +00001172 if ok:
Tim Petersf4aad8e2001-09-24 22:40:47 +00001173 hr.maybe()
Tim Peters28355492001-09-23 21:29:55 +00001174 push(msg)
1175 for name, kind, homecls, value in ok:
Guido van Rossumd59da4b2007-05-22 18:11:13 +00001176 if hasattr(value, '__call__') or inspect.isdatadescriptor(value):
Ka-Ping Yeebba6acc2005-02-19 22:58:26 +00001177 doc = getdoc(value)
Guido van Rossum5e355b22002-05-21 20:56:15 +00001178 else:
1179 doc = None
Tim Peters28355492001-09-23 21:29:55 +00001180 push(self.docother(getattr(object, name),
Georg Brandl8b813db2005-10-01 16:32:31 +00001181 name, mod, maxlen=70, doc=doc) + '\n')
Tim Peters28355492001-09-23 21:29:55 +00001182 return attrs
1183
Guido van Rossum1bc535d2007-05-15 18:46:22 +00001184 attrs = [(name, kind, cls, value)
1185 for name, kind, cls, value in classify_class_attrs(object)
1186 if visiblename(name)]
1187
Tim Petersfa26f7c2001-09-24 08:05:11 +00001188 while attrs:
Tim Peters351e3622001-09-27 03:29:51 +00001189 if mro:
Raymond Hettinger756b3f32004-01-29 06:37:52 +00001190 thisclass = mro.popleft()
Tim Peters351e3622001-09-27 03:29:51 +00001191 else:
1192 thisclass = attrs[0][2]
Tim Petersfa26f7c2001-09-24 08:05:11 +00001193 attrs, inherited = _split_list(attrs, lambda t: t[2] is thisclass)
1194
Georg Brandl1a3284e2007-12-02 09:40:06 +00001195 if thisclass is builtins.object:
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +00001196 attrs = inherited
1197 continue
1198 elif thisclass is object:
Tim Peters28355492001-09-23 21:29:55 +00001199 tag = "defined here"
1200 else:
Tim Petersfa26f7c2001-09-24 08:05:11 +00001201 tag = "inherited from %s" % classname(thisclass,
1202 object.__module__)
Tim Peters28355492001-09-23 21:29:55 +00001203
1204 # Sort attrs by name.
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +00001205 attrs.sort()
Tim Peters28355492001-09-23 21:29:55 +00001206
1207 # Pump out the attrs, segregated by kind.
Tim Petersf4aad8e2001-09-24 22:40:47 +00001208 attrs = spill("Methods %s:\n" % tag, attrs,
Tim Peters28355492001-09-23 21:29:55 +00001209 lambda t: t[1] == 'method')
Tim Petersf4aad8e2001-09-24 22:40:47 +00001210 attrs = spill("Class methods %s:\n" % tag, attrs,
Tim Peters28355492001-09-23 21:29:55 +00001211 lambda t: t[1] == 'class method')
Tim Petersf4aad8e2001-09-24 22:40:47 +00001212 attrs = spill("Static methods %s:\n" % tag, attrs,
Tim Peters28355492001-09-23 21:29:55 +00001213 lambda t: t[1] == 'static method')
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +00001214 attrs = spilldescriptors("Data descriptors %s:\n" % tag, attrs,
1215 lambda t: t[1] == 'data descriptor')
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +00001216 attrs = spilldata("Data and other attributes %s:\n" % tag, attrs,
1217 lambda t: t[1] == 'data')
Tim Peters28355492001-09-23 21:29:55 +00001218 assert attrs == []
Tim Peters351e3622001-09-27 03:29:51 +00001219 attrs = inherited
Tim Peters28355492001-09-23 21:29:55 +00001220
1221 contents = '\n'.join(contents)
1222 if not contents:
1223 return title + '\n'
Neal Norwitz9d72bb42007-04-17 08:48:32 +00001224 return title + '\n' + self.indent(contents.rstrip(), ' | ') + '\n'
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001225
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001226 def formatvalue(self, object):
1227 """Format an argument default value as text."""
1228 return '=' + self.repr(object)
1229
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001230 def docroutine(self, object, name=None, mod=None, cl=None):
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001231 """Produce text documentation for a function or method object."""
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001232 realname = object.__name__
1233 name = name or realname
1234 note = ''
Ka-Ping Yee3bda8792001-03-23 13:17:50 +00001235 skipdocs = 0
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001236 if inspect.ismethod(object):
Christian Heimesff737952007-11-27 10:40:20 +00001237 imclass = object.__self__.__class__
Ka-Ping Yee3bda8792001-03-23 13:17:50 +00001238 if cl:
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001239 if imclass is not cl:
1240 note = ' from ' + classname(imclass, mod)
Ka-Ping Yee3bda8792001-03-23 13:17:50 +00001241 else:
Christian Heimesff737952007-11-27 10:40:20 +00001242 if object.__self__ is not None:
Ka-Ping Yeeb7a48302001-04-12 20:39:14 +00001243 note = ' method of %s instance' % classname(
Christian Heimesff737952007-11-27 10:40:20 +00001244 object.__self__.__class__, mod)
Ka-Ping Yeeb7a48302001-04-12 20:39:14 +00001245 else:
1246 note = ' unbound %s method' % classname(imclass,mod)
Christian Heimesff737952007-11-27 10:40:20 +00001247 object = object.__func__
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001248
1249 if name == realname:
1250 title = self.bold(realname)
1251 else:
Raymond Hettinger54f02222002-06-01 14:18:47 +00001252 if (cl and realname in cl.__dict__ and
Ka-Ping Yee3bda8792001-03-23 13:17:50 +00001253 cl.__dict__[realname] is object):
1254 skipdocs = 1
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001255 title = self.bold(name) + ' = ' + realname
Tim Peters4bcfa312001-09-20 06:08:24 +00001256 if inspect.isfunction(object):
Guido van Rossum2e65f892007-02-28 22:03:49 +00001257 args, varargs, varkw, defaults, kwonlyargs, kwdefaults, ann = \
1258 inspect.getfullargspec(object)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001259 argspec = inspect.formatargspec(
Guido van Rossum2e65f892007-02-28 22:03:49 +00001260 args, varargs, varkw, defaults, kwonlyargs, kwdefaults, ann,
1261 formatvalue=self.formatvalue,
1262 formatannotation=inspect.formatannotationrelativeto(object))
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001263 if realname == '<lambda>':
Georg Brandl501dd0d2006-02-17 09:45:40 +00001264 title = self.bold(name) + ' lambda '
Guido van Rossum2e65f892007-02-28 22:03:49 +00001265 # XXX lambda's won't usually have func_annotations['return']
1266 # since the syntax doesn't support but it is possible.
1267 # So removing parentheses isn't truly safe.
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001268 argspec = argspec[1:-1] # remove parentheses
Tim Peters4bcfa312001-09-20 06:08:24 +00001269 else:
1270 argspec = '(...)'
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001271 decl = title + argspec + note
1272
Ka-Ping Yee3bda8792001-03-23 13:17:50 +00001273 if skipdocs:
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001274 return decl + '\n'
Ka-Ping Yee3bda8792001-03-23 13:17:50 +00001275 else:
1276 doc = getdoc(object) or ''
Neal Norwitz9d72bb42007-04-17 08:48:32 +00001277 return decl + '\n' + (doc and self.indent(doc).rstrip() + '\n')
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001278
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +00001279 def _docdescriptor(self, name, value, mod):
Johannes Gijsbers8de645a2004-11-07 19:16:05 +00001280 results = []
1281 push = results.append
1282
1283 if name:
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +00001284 push(self.bold(name))
1285 push('\n')
Johannes Gijsbers8de645a2004-11-07 19:16:05 +00001286 doc = getdoc(value) or ''
1287 if doc:
1288 push(self.indent(doc))
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +00001289 push('\n')
1290 return ''.join(results)
Johannes Gijsbers8de645a2004-11-07 19:16:05 +00001291
1292 def docproperty(self, object, name=None, mod=None, cl=None):
1293 """Produce text documentation for a property."""
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +00001294 return self._docdescriptor(name, object, mod)
Johannes Gijsbers8de645a2004-11-07 19:16:05 +00001295
Thomas Wouters0e3f5912006-08-11 14:57:12 +00001296 def docdata(self, object, name=None, mod=None, cl=None):
1297 """Produce text documentation for a data descriptor."""
1298 return self._docdescriptor(name, object, mod)
1299
Georg Brandl8b813db2005-10-01 16:32:31 +00001300 def docother(self, object, name=None, mod=None, parent=None, maxlen=None, doc=None):
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001301 """Produce text documentation for a data object."""
1302 repr = self.repr(object)
1303 if maxlen:
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001304 line = (name and name + ' = ' or '') + repr
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001305 chop = maxlen - len(line)
1306 if chop < 0: repr = repr[:chop] + '...'
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001307 line = (name and self.bold(name) + ' = ' or '') + repr
Tim Peters28355492001-09-23 21:29:55 +00001308 if doc is not None:
1309 line += '\n' + self.indent(str(doc))
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001310 return line
1311
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001312# --------------------------------------------------------- user interfaces
1313
1314def pager(text):
1315 """The first time this is called, determine what kind of pager to use."""
1316 global pager
1317 pager = getpager()
1318 pager(text)
1319
1320def getpager():
1321 """Decide what method to use for paging through text."""
Guido van Rossuma01a8b62007-05-27 09:20:14 +00001322 if not hasattr(sys.stdout, "isatty"):
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001323 return plainpager
1324 if not sys.stdin.isatty() or not sys.stdout.isatty():
1325 return plainpager
Raymond Hettinger54f02222002-06-01 14:18:47 +00001326 if 'PAGER' in os.environ:
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +00001327 if sys.platform == 'win32': # pipes completely broken in Windows
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001328 return lambda text: tempfilepager(plain(text), os.environ['PAGER'])
Raymond Hettingerdbecd932005-02-06 06:57:08 +00001329 elif os.environ.get('TERM') in ('dumb', 'emacs'):
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001330 return lambda text: pipepager(plain(text), os.environ['PAGER'])
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +00001331 else:
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001332 return lambda text: pipepager(text, os.environ['PAGER'])
Ka-Ping Yeea487e4e2005-11-05 04:49:18 +00001333 if os.environ.get('TERM') in ('dumb', 'emacs'):
1334 return plainpager
Andrew MacIntyre54e0eab2002-03-03 03:12:30 +00001335 if sys.platform == 'win32' or sys.platform.startswith('os2'):
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001336 return lambda text: tempfilepager(plain(text), 'more <')
Skip Montanarod404bee2002-09-26 21:44:57 +00001337 if hasattr(os, 'system') and os.system('(less) 2>/dev/null') == 0:
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001338 return lambda text: pipepager(text, 'less')
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001339
1340 import tempfile
Guido van Rossum3b0a3292002-08-09 16:38:32 +00001341 (fd, filename) = tempfile.mkstemp()
1342 os.close(fd)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001343 try:
1344 if hasattr(os, 'system') and os.system('more %s' % filename) == 0:
1345 return lambda text: pipepager(text, 'more')
1346 else:
1347 return ttypager
1348 finally:
1349 os.unlink(filename)
1350
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001351def plain(text):
1352 """Remove boldface formatting from text."""
1353 return re.sub('.\b', '', text)
1354
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001355def pipepager(text, cmd):
1356 """Page through text by feeding it to another program."""
1357 pipe = os.popen(cmd, 'w')
1358 try:
1359 pipe.write(text)
1360 pipe.close()
1361 except IOError:
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001362 pass # Ignore broken pipes caused by quitting the pager program.
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001363
1364def tempfilepager(text, cmd):
1365 """Page through text by invoking a program on a temporary file."""
1366 import tempfile
Tim Peters550e4e52003-02-07 01:53:46 +00001367 filename = tempfile.mktemp()
1368 file = open(filename, 'w')
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001369 file.write(text)
1370 file.close()
1371 try:
Ka-Ping Yeec92cdf72001-03-02 05:54:35 +00001372 os.system(cmd + ' ' + filename)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001373 finally:
1374 os.unlink(filename)
1375
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001376def ttypager(text):
1377 """Page through text on a text terminal."""
Neal Norwitz9d72bb42007-04-17 08:48:32 +00001378 lines = plain(text).split('\n')
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001379 try:
1380 import tty
1381 fd = sys.stdin.fileno()
1382 old = tty.tcgetattr(fd)
1383 tty.setcbreak(fd)
1384 getchar = lambda: sys.stdin.read(1)
Ka-Ping Yee457aab22001-02-27 23:36:29 +00001385 except (ImportError, AttributeError):
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001386 tty = None
1387 getchar = lambda: sys.stdin.readline()[:-1][:1]
1388
1389 try:
1390 r = inc = os.environ.get('LINES', 25) - 1
Neal Norwitz9d72bb42007-04-17 08:48:32 +00001391 sys.stdout.write('\n'.join(lines[:inc]) + '\n')
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001392 while lines[r:]:
1393 sys.stdout.write('-- more --')
1394 sys.stdout.flush()
1395 c = getchar()
1396
Raymond Hettingerdbecd932005-02-06 06:57:08 +00001397 if c in ('q', 'Q'):
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001398 sys.stdout.write('\r \r')
1399 break
Raymond Hettingerdbecd932005-02-06 06:57:08 +00001400 elif c in ('\r', '\n'):
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001401 sys.stdout.write('\r \r' + lines[r] + '\n')
1402 r = r + 1
1403 continue
Raymond Hettingerdbecd932005-02-06 06:57:08 +00001404 if c in ('b', 'B', '\x1b'):
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001405 r = r - inc - inc
1406 if r < 0: r = 0
Neal Norwitz9d72bb42007-04-17 08:48:32 +00001407 sys.stdout.write('\n' + '\n'.join(lines[r:r+inc]) + '\n')
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001408 r = r + inc
1409
1410 finally:
1411 if tty:
1412 tty.tcsetattr(fd, tty.TCSAFLUSH, old)
1413
1414def plainpager(text):
1415 """Simply print unformatted text. This is the ultimate fallback."""
1416 sys.stdout.write(plain(text))
1417
1418def describe(thing):
Ka-Ping Yee3bda8792001-03-23 13:17:50 +00001419 """Produce a short description of the given thing."""
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001420 if inspect.ismodule(thing):
1421 if thing.__name__ in sys.builtin_module_names:
1422 return 'built-in module ' + thing.__name__
1423 if hasattr(thing, '__path__'):
1424 return 'package ' + thing.__name__
1425 else:
1426 return 'module ' + thing.__name__
1427 if inspect.isbuiltin(thing):
1428 return 'built-in function ' + thing.__name__
Thomas Wouters0e3f5912006-08-11 14:57:12 +00001429 if inspect.isgetsetdescriptor(thing):
1430 return 'getset descriptor %s.%s.%s' % (
1431 thing.__objclass__.__module__, thing.__objclass__.__name__,
1432 thing.__name__)
1433 if inspect.ismemberdescriptor(thing):
1434 return 'member descriptor %s.%s.%s' % (
1435 thing.__objclass__.__module__, thing.__objclass__.__name__,
1436 thing.__name__)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001437 if inspect.isclass(thing):
1438 return 'class ' + thing.__name__
1439 if inspect.isfunction(thing):
1440 return 'function ' + thing.__name__
1441 if inspect.ismethod(thing):
1442 return 'method ' + thing.__name__
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001443 return type(thing).__name__
1444
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001445def locate(path, forceload=0):
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001446 """Locate an object by name or dotted path, importing as necessary."""
Neal Norwitz9d72bb42007-04-17 08:48:32 +00001447 parts = [part for part in path.split('.') if part]
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001448 module, n = None, 0
1449 while n < len(parts):
Neal Norwitz9d72bb42007-04-17 08:48:32 +00001450 nextmodule = safeimport('.'.join(parts[:n+1]), forceload)
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001451 if nextmodule: module, n = nextmodule, n + 1
1452 else: break
1453 if module:
1454 object = module
1455 for part in parts[n:]:
1456 try: object = getattr(object, part)
1457 except AttributeError: return None
1458 return object
1459 else:
Georg Brandl1a3284e2007-12-02 09:40:06 +00001460 if hasattr(builtins, path):
1461 return getattr(builtins, path)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001462
1463# --------------------------------------- interactive interpreter interface
1464
1465text = TextDoc()
1466html = HTMLDoc()
1467
Ka-Ping Yee45daeb02002-08-11 15:11:33 +00001468def resolve(thing, forceload=0):
1469 """Given an object or a path to an object, get the object and its name."""
1470 if isinstance(thing, str):
1471 object = locate(thing, forceload)
1472 if not object:
Collin Winterce36ad82007-08-30 01:19:48 +00001473 raise ImportError('no Python documentation found for %r' % thing)
Ka-Ping Yee45daeb02002-08-11 15:11:33 +00001474 return object, thing
1475 else:
1476 return thing, getattr(thing, '__name__', None)
1477
Guido van Rossumd8faa362007-04-27 19:54:29 +00001478def render_doc(thing, title='Python Library Documentation: %s', forceload=0):
1479 """Render text documentation, given an object or a path to an object."""
1480 object, name = resolve(thing, forceload)
1481 desc = describe(object)
1482 module = inspect.getmodule(object)
1483 if name and '.' in name:
1484 desc += ' in ' + name[:name.rfind('.')]
1485 elif module and module is not object:
1486 desc += ' in module ' + module.__name__
Amaury Forgeot d'Arc768db922008-04-24 21:00:04 +00001487
1488 if not (inspect.ismodule(object) or
Guido van Rossumd8faa362007-04-27 19:54:29 +00001489 inspect.isclass(object) or
1490 inspect.isroutine(object) or
1491 inspect.isgetsetdescriptor(object) or
1492 inspect.ismemberdescriptor(object) or
1493 isinstance(object, property)):
1494 # If the passed object is a piece of data or an instance,
1495 # document its available methods instead of its value.
1496 object = type(object)
1497 desc += ' object'
1498 return title % desc + '\n\n' + text.document(object, name)
1499
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001500def doc(thing, title='Python Library Documentation: %s', forceload=0):
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001501 """Display text documentation, given an object or a path to an object."""
Ka-Ping Yee45daeb02002-08-11 15:11:33 +00001502 try:
Guido van Rossumd8faa362007-04-27 19:54:29 +00001503 pager(render_doc(thing, title, forceload))
Guido van Rossumb940e112007-01-10 16:19:56 +00001504 except (ImportError, ErrorDuringImport) as value:
Guido van Rossumbe19ed72007-02-09 05:37:30 +00001505 print(value)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001506
Ka-Ping Yee45daeb02002-08-11 15:11:33 +00001507def writedoc(thing, forceload=0):
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001508 """Write HTML documentation to a file in the current directory."""
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001509 try:
Ka-Ping Yee45daeb02002-08-11 15:11:33 +00001510 object, name = resolve(thing, forceload)
1511 page = html.page(describe(object), html.document(object, name))
1512 file = open(name + '.html', 'w')
1513 file.write(page)
1514 file.close()
Guido van Rossumbe19ed72007-02-09 05:37:30 +00001515 print('wrote', name + '.html')
Guido van Rossumb940e112007-01-10 16:19:56 +00001516 except (ImportError, ErrorDuringImport) as value:
Guido van Rossumbe19ed72007-02-09 05:37:30 +00001517 print(value)
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001518
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001519def writedocs(dir, pkgpath='', done=None):
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001520 """Write out HTML documentation for all modules in a directory tree."""
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001521 if done is None: done = {}
Thomas Wouters49fd7fa2006-04-21 10:40:58 +00001522 for importer, modname, ispkg in pkgutil.walk_packages([dir], pkgpath):
1523 writedoc(modname)
1524 return
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001525
1526class Helper:
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001527 keywords = {
1528 'and': 'BOOLEAN',
Guido van Rossumd8faa362007-04-27 19:54:29 +00001529 'as': 'with',
Ka-Ping Yee79c009d2001-04-13 10:53:25 +00001530 'assert': ('ref/assert', ''),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001531 'break': ('ref/break', 'while for'),
1532 'class': ('ref/class', 'CLASSES SPECIALMETHODS'),
1533 'continue': ('ref/continue', 'while for'),
1534 'def': ('ref/function', ''),
1535 'del': ('ref/del', 'BASICMETHODS'),
1536 'elif': 'if',
1537 'else': ('ref/if', 'while for'),
1538 'except': 'try',
1539 'exec': ('ref/exec', ''),
1540 'finally': 'try',
1541 'for': ('ref/for', 'break continue while'),
1542 'from': 'import',
1543 'global': ('ref/global', 'NAMESPACES'),
1544 'if': ('ref/if', 'TRUTHVALUE'),
1545 'import': ('ref/import', 'MODULES'),
1546 'in': ('ref/comparisons', 'SEQUENCEMETHODS2'),
1547 'is': 'COMPARISON',
Neal Norwitz742dde42003-03-30 20:31:34 +00001548 'lambda': ('ref/lambdas', 'FUNCTIONS'),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001549 'not': 'BOOLEAN',
1550 'or': 'BOOLEAN',
Neal Norwitz742dde42003-03-30 20:31:34 +00001551 'pass': ('ref/pass', ''),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001552 'print': ('ref/print', ''),
1553 'raise': ('ref/raise', 'EXCEPTIONS'),
Ka-Ping Yee79c009d2001-04-13 10:53:25 +00001554 'return': ('ref/return', 'FUNCTIONS'),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001555 'try': ('ref/try', 'EXCEPTIONS'),
1556 'while': ('ref/while', 'break continue if TRUTHVALUE'),
Guido van Rossumd8faa362007-04-27 19:54:29 +00001557 'with': ('ref/with', 'CONTEXTMANAGERS EXCEPTIONS yield'),
Tim Petersfb05c4e2002-10-30 05:21:00 +00001558 'yield': ('ref/yield', ''),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001559 }
1560
1561 topics = {
1562 'TYPES': ('ref/types', 'STRINGS UNICODE NUMBERS SEQUENCES MAPPINGS FUNCTIONS CLASSES MODULES FILES inspect'),
Ka-Ping Yee79c009d2001-04-13 10:53:25 +00001563 'STRINGS': ('ref/strings', 'str UNICODE SEQUENCES STRINGMETHODS FORMATTING TYPES'),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001564 'STRINGMETHODS': ('lib/string-methods', 'STRINGS FORMATTING'),
1565 'FORMATTING': ('lib/typesseq-strings', 'OPERATORS'),
Neal Norwitz54f871e2003-05-26 13:49:54 +00001566 'UNICODE': ('ref/strings', 'encodings unicode SEQUENCES STRINGMETHODS FORMATTING TYPES'),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001567 'NUMBERS': ('ref/numbers', 'INTEGER FLOAT COMPLEX TYPES'),
1568 'INTEGER': ('ref/integers', 'int range'),
1569 'FLOAT': ('ref/floating', 'float math'),
1570 'COMPLEX': ('ref/imaginary', 'complex cmath'),
Guido van Rossum805365e2007-05-07 22:24:25 +00001571 'SEQUENCES': ('lib/typesseq', 'STRINGMETHODS FORMATTING range LISTS'),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001572 'MAPPINGS': 'DICTIONARIES',
1573 'FUNCTIONS': ('lib/typesfunctions', 'def TYPES'),
1574 'METHODS': ('lib/typesmethods', 'class def CLASSES TYPES'),
1575 'CODEOBJECTS': ('lib/bltin-code-objects', 'compile FUNCTIONS TYPES'),
Ka-Ping Yee79c009d2001-04-13 10:53:25 +00001576 'TYPEOBJECTS': ('lib/bltin-type-objects', 'types TYPES'),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001577 'FRAMEOBJECTS': 'TYPES',
1578 'TRACEBACKS': 'TYPES',
1579 'NONE': ('lib/bltin-null-object', ''),
1580 'ELLIPSIS': ('lib/bltin-ellipsis-object', 'SLICINGS'),
1581 'FILES': ('lib/bltin-file-objects', ''),
1582 'SPECIALATTRIBUTES': ('lib/specialattrs', ''),
1583 'CLASSES': ('ref/types', 'class SPECIALMETHODS PRIVATENAMES'),
1584 'MODULES': ('lib/typesmodules', 'import'),
1585 'PACKAGES': 'import',
Neal Norwitz2eca4402006-08-29 04:40:24 +00001586 '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 +00001587 'OPERATORS': 'EXPRESSIONS',
1588 'PRECEDENCE': 'EXPRESSIONS',
1589 'OBJECTS': ('ref/objects', 'TYPES'),
1590 'SPECIALMETHODS': ('ref/specialnames', 'BASICMETHODS ATTRIBUTEMETHODS CALLABLEMETHODS SEQUENCEMETHODS1 MAPPINGMETHODS SEQUENCEMETHODS2 NUMBERMETHODS CLASSES'),
Ka-Ping Yee79c009d2001-04-13 10:53:25 +00001591 'BASICMETHODS': ('ref/customization', 'cmp hash repr str SPECIALMETHODS'),
1592 'ATTRIBUTEMETHODS': ('ref/attribute-access', 'ATTRIBUTES SPECIALMETHODS'),
1593 'CALLABLEMETHODS': ('ref/callable-types', 'CALLS SPECIALMETHODS'),
1594 'SEQUENCEMETHODS1': ('ref/sequence-types', 'SEQUENCES SEQUENCEMETHODS2 SPECIALMETHODS'),
1595 'SEQUENCEMETHODS2': ('ref/sequence-methods', 'SEQUENCES SEQUENCEMETHODS1 SPECIALMETHODS'),
1596 'MAPPINGMETHODS': ('ref/sequence-types', 'MAPPINGS SPECIALMETHODS'),
1597 'NUMBERMETHODS': ('ref/numeric-types', 'NUMBERS AUGMENTEDASSIGNMENT SPECIALMETHODS'),
Neal Norwitz54f871e2003-05-26 13:49:54 +00001598 'EXECUTION': ('ref/execmodel', 'NAMESPACES DYNAMICFEATURES EXCEPTIONS'),
1599 'NAMESPACES': ('ref/naming', 'global ASSIGNMENT DELETION DYNAMICFEATURES'),
1600 'DYNAMICFEATURES': ('ref/dynamic-features', ''),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001601 'SCOPING': 'NAMESPACES',
1602 'FRAMES': 'NAMESPACES',
1603 'EXCEPTIONS': ('ref/exceptions', 'try except finally raise'),
Neal Norwitz54f871e2003-05-26 13:49:54 +00001604 'COERCIONS': ('ref/coercion-rules','CONVERSIONS'),
1605 'CONVERSIONS': ('ref/conversions', 'COERCIONS'),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001606 'IDENTIFIERS': ('ref/identifiers', 'keywords SPECIALIDENTIFIERS'),
1607 'SPECIALIDENTIFIERS': ('ref/id-classes', ''),
Ka-Ping Yee79c009d2001-04-13 10:53:25 +00001608 'PRIVATENAMES': ('ref/atom-identifiers', ''),
Neal Norwitz2eca4402006-08-29 04:40:24 +00001609 'LITERALS': ('ref/atom-literals', 'STRINGS NUMBERS TUPLELITERALS LISTLITERALS DICTIONARYLITERALS'),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001610 'TUPLES': 'SEQUENCES',
Ka-Ping Yee79c009d2001-04-13 10:53:25 +00001611 'TUPLELITERALS': ('ref/exprlists', 'TUPLES LITERALS'),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001612 'LISTS': ('lib/typesseq-mutable', 'LISTLITERALS'),
Ka-Ping Yee79c009d2001-04-13 10:53:25 +00001613 'LISTLITERALS': ('ref/lists', 'LISTS LITERALS'),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001614 'DICTIONARIES': ('lib/typesmapping', 'DICTIONARYLITERALS'),
Ka-Ping Yee79c009d2001-04-13 10:53:25 +00001615 'DICTIONARYLITERALS': ('ref/dict', 'DICTIONARIES LITERALS'),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001616 'ATTRIBUTES': ('ref/attribute-references', 'getattr hasattr setattr ATTRIBUTEMETHODS'),
1617 'SUBSCRIPTS': ('ref/subscriptions', 'SEQUENCEMETHODS1'),
1618 'SLICINGS': ('ref/slicings', 'SEQUENCEMETHODS2'),
1619 'CALLS': ('ref/calls', 'EXPRESSIONS'),
1620 'POWER': ('ref/power', 'EXPRESSIONS'),
1621 'UNARY': ('ref/unary', 'EXPRESSIONS'),
1622 'BINARY': ('ref/binary', 'EXPRESSIONS'),
1623 'SHIFTING': ('ref/shifting', 'EXPRESSIONS'),
1624 'BITWISE': ('ref/bitwise', 'EXPRESSIONS'),
1625 'COMPARISON': ('ref/comparisons', 'EXPRESSIONS BASICMETHODS'),
Neal Norwitzf98159c2003-02-07 20:49:40 +00001626 'BOOLEAN': ('ref/Booleans', 'EXPRESSIONS TRUTHVALUE'),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001627 'ASSERTION': 'assert',
1628 'ASSIGNMENT': ('ref/assignment', 'AUGMENTEDASSIGNMENT'),
Ka-Ping Yee79c009d2001-04-13 10:53:25 +00001629 'AUGMENTEDASSIGNMENT': ('ref/augassign', 'NUMBERMETHODS'),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001630 'DELETION': 'del',
1631 'PRINTING': 'print',
1632 'RETURNING': 'return',
1633 'IMPORTING': 'import',
1634 'CONDITIONAL': 'if',
1635 'LOOPING': ('ref/compound', 'for while break continue'),
1636 'TRUTHVALUE': ('lib/truth', 'if while and or not BASICMETHODS'),
Ka-Ping Yee79c009d2001-04-13 10:53:25 +00001637 'DEBUGGING': ('lib/module-pdb', 'pdb'),
Guido van Rossumd8faa362007-04-27 19:54:29 +00001638 'CONTEXTMANAGERS': ('ref/context-managers', 'with'),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001639 }
1640
1641 def __init__(self, input, output):
1642 self.input = input
1643 self.output = output
1644 self.docdir = None
1645 execdir = os.path.dirname(sys.executable)
1646 homedir = os.environ.get('PYTHONHOME')
Guido van Rossumd8faa362007-04-27 19:54:29 +00001647 join = os.path.join
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001648 for dir in [os.environ.get('PYTHONDOCS'),
1649 homedir and os.path.join(homedir, 'doc'),
Guido van Rossumd8faa362007-04-27 19:54:29 +00001650 join(execdir, 'doc'), # for Windows
1651 join(sys.prefix, 'doc/python-docs-' + sys.version.split()[0]),
1652 join(sys.prefix, 'doc/python-' + sys.version.split()[0]),
1653 join(sys.prefix, 'doc/python-docs-' + sys.version[:3]),
1654 join(sys.prefix, 'doc/python-' + sys.version[:3]),
1655 join(sys.prefix, 'Resources/English.lproj/Documentation')]:
1656 if dir and os.path.isdir(join(dir, 'lib')):
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001657 self.docdir = dir
Guido van Rossumd8faa362007-04-27 19:54:29 +00001658 break
1659 if dir and os.path.isdir(join(dir, 'html', 'lib')):
1660 self.docdir = join(dir, 'html')
1661 break
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001662
Ka-Ping Yee79c009d2001-04-13 10:53:25 +00001663 def __repr__(self):
Ka-Ping Yee9bc576b2001-04-13 13:57:31 +00001664 if inspect.stack()[1][3] == '?':
Ka-Ping Yee79c009d2001-04-13 10:53:25 +00001665 self()
1666 return ''
Ka-Ping Yee9bc576b2001-04-13 13:57:31 +00001667 return '<pydoc.Helper instance>'
Ka-Ping Yee79c009d2001-04-13 10:53:25 +00001668
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001669 def __call__(self, request=None):
1670 if request is not None:
1671 self.help(request)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001672 else:
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001673 self.intro()
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001674 self.interact()
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001675 self.output.write('''
Fred Drakee61967f2001-05-10 18:41:02 +00001676You are now leaving help and returning to the Python interpreter.
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001677If you want to ask for help on a particular object directly from the
1678interpreter, you can type "help(object)". Executing "help('string')"
1679has the same effect as typing a particular string at the help> prompt.
1680''')
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001681
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001682 def interact(self):
1683 self.output.write('\n')
Guido van Rossum8ca162f2002-04-07 06:36:23 +00001684 while True:
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001685 try:
Johannes Gijsberse7691d32004-08-17 13:21:53 +00001686 request = self.getline('help> ')
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001687 if not request: break
Johannes Gijsberse7691d32004-08-17 13:21:53 +00001688 except (KeyboardInterrupt, EOFError):
1689 break
Neal Norwitz9d72bb42007-04-17 08:48:32 +00001690 request = replace(request, '"', '', "'", '').strip()
1691 if request.lower() in ('q', 'quit'): break
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001692 self.help(request)
1693
Johannes Gijsberse7691d32004-08-17 13:21:53 +00001694 def getline(self, prompt):
Guido van Rossum7c086c02008-01-02 03:52:38 +00001695 """Read one line, using input() when appropriate."""
Johannes Gijsberse7691d32004-08-17 13:21:53 +00001696 if self.input is sys.stdin:
Guido van Rossum7c086c02008-01-02 03:52:38 +00001697 return input(prompt)
Johannes Gijsberse7691d32004-08-17 13:21:53 +00001698 else:
1699 self.output.write(prompt)
1700 self.output.flush()
1701 return self.input.readline()
1702
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001703 def help(self, request):
1704 if type(request) is type(''):
1705 if request == 'help': self.intro()
1706 elif request == 'keywords': self.listkeywords()
1707 elif request == 'topics': self.listtopics()
1708 elif request == 'modules': self.listmodules()
1709 elif request[:8] == 'modules ':
Neal Norwitz9d72bb42007-04-17 08:48:32 +00001710 self.listmodules(request.split()[1])
Raymond Hettinger54f02222002-06-01 14:18:47 +00001711 elif request in self.keywords: self.showtopic(request)
1712 elif request in self.topics: self.showtopic(request)
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001713 elif request: doc(request, 'Help on %s:')
1714 elif isinstance(request, Helper): self()
1715 else: doc(request, 'Help on %s:')
1716 self.output.write('\n')
1717
1718 def intro(self):
1719 self.output.write('''
1720Welcome to Python %s! This is the online help utility.
1721
1722If this is your first time using Python, you should definitely check out
Georg Brandl86def6c2008-01-21 20:36:10 +00001723the tutorial on the Internet at http://docs.python.org/tutorial/.
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001724
1725Enter the name of any module, keyword, or topic to get help on writing
1726Python programs and using Python modules. To quit this help utility and
1727return to the interpreter, just type "quit".
1728
1729To get a list of available modules, keywords, or topics, type "modules",
1730"keywords", or "topics". Each module also comes with a one-line summary
1731of what it does; to list the modules whose summaries contain a given word
1732such as "spam", type "modules spam".
1733''' % sys.version[:3])
1734
1735 def list(self, items, columns=4, width=80):
Guido van Rossum486364b2007-06-30 05:01:58 +00001736 items = list(sorted(items))
1737 colw = width // columns
1738 rows = (len(items) + columns - 1) // columns
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001739 for row in range(rows):
1740 for col in range(columns):
1741 i = col * rows + row
1742 if i < len(items):
1743 self.output.write(items[i])
1744 if col < columns - 1:
Guido van Rossum486364b2007-06-30 05:01:58 +00001745 self.output.write(' ' + ' ' * (colw - 1 - len(items[i])))
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001746 self.output.write('\n')
1747
1748 def listkeywords(self):
1749 self.output.write('''
1750Here is a list of the Python keywords. Enter any keyword to get more help.
1751
1752''')
1753 self.list(self.keywords.keys())
1754
1755 def listtopics(self):
1756 self.output.write('''
1757Here is a list of available topics. Enter any topic name to get more help.
1758
1759''')
1760 self.list(self.topics.keys())
1761
1762 def showtopic(self, topic):
1763 if not self.docdir:
1764 self.output.write('''
1765Sorry, topic and keyword documentation is not available because the Python
1766HTML documentation files could not be found. If you have installed them,
1767please set the environment variable PYTHONDOCS to indicate their location.
Thomas Wouters902d6eb2007-01-09 23:18:33 +00001768
1769On the Microsoft Windows operating system, the files can be built by
1770running "hh -decompile . PythonNN.chm" in the C:\PythonNN\Doc> directory.
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001771''')
1772 return
1773 target = self.topics.get(topic, self.keywords.get(topic))
1774 if not target:
1775 self.output.write('no documentation found for %s\n' % repr(topic))
1776 return
1777 if type(target) is type(''):
1778 return self.showtopic(target)
1779
1780 filename, xrefs = target
1781 filename = self.docdir + '/' + filename + '.html'
1782 try:
1783 file = open(filename)
1784 except:
1785 self.output.write('could not read docs from %s\n' % filename)
1786 return
1787
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001788 divpat = re.compile('<div[^>]*navigat.*?</div.*?>', re.I | re.S)
1789 addrpat = re.compile('<address.*?>.*?</address.*?>', re.I | re.S)
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001790 document = re.sub(addrpat, '', re.sub(divpat, '', file.read()))
1791 file.close()
1792
Guido van Rossum34d19282007-08-09 01:03:29 +00001793 import htmllib, formatter, io
1794 buffer = io.StringIO()
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001795 parser = htmllib.HTMLParser(
1796 formatter.AbstractFormatter(formatter.DumbWriter(buffer)))
1797 parser.start_table = parser.do_p
1798 parser.end_table = lambda parser=parser: parser.do_p({})
1799 parser.start_tr = parser.do_br
1800 parser.start_td = parser.start_th = lambda a, b=buffer: b.write('\t')
1801 parser.feed(document)
1802 buffer = replace(buffer.getvalue(), '\xa0', ' ', '\n', '\n ')
Neal Norwitz9d72bb42007-04-17 08:48:32 +00001803 pager(' ' + buffer.strip() + '\n')
Ka-Ping Yeeda793892001-04-13 11:02:51 +00001804 if xrefs:
Guido van Rossum34d19282007-08-09 01:03:29 +00001805 buffer = io.StringIO()
Ka-Ping Yeeda793892001-04-13 11:02:51 +00001806 formatter.DumbWriter(buffer).send_flowing_data(
Neal Norwitz9d72bb42007-04-17 08:48:32 +00001807 'Related help topics: ' + ', '.join(xrefs.split()) + '\n')
Ka-Ping Yeeda793892001-04-13 11:02:51 +00001808 self.output.write('\n%s\n' % buffer.getvalue())
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001809
1810 def listmodules(self, key=''):
1811 if key:
1812 self.output.write('''
1813Here is a list of matching modules. Enter any module name to get more help.
1814
1815''')
1816 apropos(key)
1817 else:
1818 self.output.write('''
1819Please wait a moment while I gather a list of all available modules...
1820
1821''')
1822 modules = {}
1823 def callback(path, modname, desc, modules=modules):
1824 if modname and modname[-9:] == '.__init__':
1825 modname = modname[:-9] + ' (package)'
Neal Norwitz9d72bb42007-04-17 08:48:32 +00001826 if modname.find('.') < 0:
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001827 modules[modname] = 1
Christian Heimesd32ed6f2008-01-14 18:49:24 +00001828 def onerror(modname):
1829 callback(None, modname, None)
1830 ModuleScanner().run(callback, onerror=onerror)
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001831 self.list(modules.keys())
1832 self.output.write('''
1833Enter any module name to get more help. Or, type "modules spam" to search
1834for modules whose descriptions contain the word "spam".
1835''')
1836
1837help = Helper(sys.stdin, sys.stdout)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001838
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001839class Scanner:
1840 """A generic tree iterator."""
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001841 def __init__(self, roots, children, descendp):
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001842 self.roots = roots[:]
1843 self.state = []
1844 self.children = children
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001845 self.descendp = descendp
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001846
1847 def next(self):
1848 if not self.state:
1849 if not self.roots:
1850 return None
1851 root = self.roots.pop(0)
1852 self.state = [(root, self.children(root))]
1853 node, children = self.state[-1]
1854 if not children:
1855 self.state.pop()
1856 return self.next()
1857 child = children.pop(0)
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001858 if self.descendp(child):
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001859 self.state.append((child, self.children(child)))
1860 return child
1861
Thomas Wouters49fd7fa2006-04-21 10:40:58 +00001862
1863class ModuleScanner:
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001864 """An interruptible scanner that searches module synopses."""
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001865
Christian Heimesd32ed6f2008-01-14 18:49:24 +00001866 def run(self, callback, key=None, completer=None, onerror=None):
Neal Norwitz9d72bb42007-04-17 08:48:32 +00001867 if key: key = key.lower()
Guido van Rossum8ca162f2002-04-07 06:36:23 +00001868 self.quit = False
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001869 seen = {}
1870
1871 for modname in sys.builtin_module_names:
Ka-Ping Yee239432a2001-03-02 02:45:08 +00001872 if modname != '__main__':
1873 seen[modname] = 1
Ka-Ping Yee66246962001-04-12 11:59:50 +00001874 if key is None:
1875 callback(None, modname, '')
1876 else:
Neal Norwitz9d72bb42007-04-17 08:48:32 +00001877 name = __import__(modname).__doc__ or ''
1878 desc = name.split('\n')[0]
1879 name = modname + ' - ' + desc
1880 if name.lower().find(key) >= 0:
Ka-Ping Yee66246962001-04-12 11:59:50 +00001881 callback(None, modname, desc)
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001882
Christian Heimesd32ed6f2008-01-14 18:49:24 +00001883 for importer, modname, ispkg in pkgutil.walk_packages(onerror=onerror):
Thomas Wouters49fd7fa2006-04-21 10:40:58 +00001884 if self.quit:
1885 break
1886 if key is None:
1887 callback(None, modname, '')
1888 else:
1889 loader = importer.find_module(modname)
1890 if hasattr(loader,'get_source'):
Guido van Rossum34d19282007-08-09 01:03:29 +00001891 import io
Thomas Wouters49fd7fa2006-04-21 10:40:58 +00001892 desc = source_synopsis(
Guido van Rossum34d19282007-08-09 01:03:29 +00001893 io.StringIO(loader.get_source(modname))
Thomas Wouters49fd7fa2006-04-21 10:40:58 +00001894 ) or ''
1895 if hasattr(loader,'get_filename'):
1896 path = loader.get_filename(modname)
Ka-Ping Yee66246962001-04-12 11:59:50 +00001897 else:
Thomas Wouters49fd7fa2006-04-21 10:40:58 +00001898 path = None
1899 else:
1900 module = loader.load_module(modname)
1901 desc = (module.__doc__ or '').splitlines()[0]
1902 path = getattr(module,'__file__',None)
Neal Norwitz9d72bb42007-04-17 08:48:32 +00001903 name = modname + ' - ' + desc
1904 if name.lower().find(key) >= 0:
Thomas Wouters49fd7fa2006-04-21 10:40:58 +00001905 callback(path, modname, desc)
1906
1907 if completer:
1908 completer()
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001909
1910def apropos(key):
1911 """Print all the one-line module summaries that contain a substring."""
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001912 def callback(path, modname, desc):
1913 if modname[-9:] == '.__init__':
1914 modname = modname[:-9] + ' (package)'
Guido van Rossumbe19ed72007-02-09 05:37:30 +00001915 print(modname, desc and '- ' + desc)
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +00001916 try: import warnings
1917 except ImportError: pass
1918 else: warnings.filterwarnings('ignore') # ignore problems during import
Ka-Ping Yee66246962001-04-12 11:59:50 +00001919 ModuleScanner().run(callback, key)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001920
1921# --------------------------------------------------- web browser interface
1922
Ka-Ping Yee66246962001-04-12 11:59:50 +00001923def serve(port, callback=None, completer=None):
Ka-Ping Yeefd540692001-04-12 12:54:36 +00001924 import BaseHTTPServer, mimetools, select
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001925
1926 # Patch up mimetools.Message so it doesn't break if rfc822 is reloaded.
1927 class Message(mimetools.Message):
1928 def __init__(self, fp, seekable=1):
1929 Message = self.__class__
1930 Message.__bases__[0].__bases__[0].__init__(self, fp, seekable)
1931 self.encodingheader = self.getheader('content-transfer-encoding')
1932 self.typeheader = self.getheader('content-type')
1933 self.parsetype()
1934 self.parseplist()
1935
1936 class DocHandler(BaseHTTPServer.BaseHTTPRequestHandler):
1937 def send_document(self, title, contents):
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001938 try:
1939 self.send_response(200)
Georg Brandl825fc8b2008-01-19 20:44:32 +00001940 self.send_header('Content-Type', 'text/html; charset=UTF-8')
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001941 self.end_headers()
Georg Brandl825fc8b2008-01-19 20:44:32 +00001942 self.wfile.write(html.page(title, contents).encode('utf-8'))
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001943 except IOError: pass
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001944
1945 def do_GET(self):
1946 path = self.path
1947 if path[-5:] == '.html': path = path[:-5]
1948 if path[:1] == '/': path = path[1:]
1949 if path and path != '.':
1950 try:
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001951 obj = locate(path, forceload=1)
Guido van Rossumb940e112007-01-10 16:19:56 +00001952 except ErrorDuringImport as value:
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001953 self.send_document(path, html.escape(str(value)))
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001954 return
Ka-Ping Yee3bda8792001-03-23 13:17:50 +00001955 if obj:
1956 self.send_document(describe(obj), html.document(obj, path))
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001957 else:
1958 self.send_document(path,
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001959'no Python documentation found for %s' % repr(path))
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001960 else:
1961 heading = html.heading(
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001962'<big><big><strong>Python: Index of Modules</strong></big></big>',
1963'#ffffff', '#7799ee')
Ka-Ping Yee239432a2001-03-02 02:45:08 +00001964 def bltinlink(name):
1965 return '<a href="%s.html">%s</a>' % (name, name)
Georg Brandl661b0a12008-01-27 18:16:00 +00001966 names = [x for x in sys.builtin_module_names if x != '__main__']
1967 contents = html.multicolumn(names, bltinlink)
Ka-Ping Yee239432a2001-03-02 02:45:08 +00001968 indices = ['<p>' + html.bigsection(
1969 'Built-in Modules', '#ffffff', '#ee77aa', contents)]
1970
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001971 seen = {}
Thomas Wouters49fd7fa2006-04-21 10:40:58 +00001972 for dir in sys.path:
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001973 indices.append(html.index(dir, seen))
Neal Norwitz9d72bb42007-04-17 08:48:32 +00001974 contents = heading + ' '.join(indices) + '''<p align=right>
Tim Peters2306d242001-09-25 03:18:32 +00001975<font color="#909090" face="helvetica, arial"><strong>
1976pydoc</strong> by Ka-Ping Yee &lt;ping@lfw.org&gt;</font>'''
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001977 self.send_document('Index of Modules', contents)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001978
1979 def log_message(self, *args): pass
1980
Ka-Ping Yeefd540692001-04-12 12:54:36 +00001981 class DocServer(BaseHTTPServer.HTTPServer):
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001982 def __init__(self, port, callback):
Ka-Ping Yeedb8ed152001-03-02 05:58:17 +00001983 host = (sys.platform == 'mac') and '127.0.0.1' or 'localhost'
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001984 self.address = ('', port)
Ka-Ping Yeedb8ed152001-03-02 05:58:17 +00001985 self.url = 'http://%s:%d/' % (host, port)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001986 self.callback = callback
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001987 self.base.__init__(self, self.address, self.handler)
1988
1989 def serve_until_quit(self):
1990 import select
Guido van Rossum8ca162f2002-04-07 06:36:23 +00001991 self.quit = False
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001992 while not self.quit:
1993 rd, wr, ex = select.select([self.socket.fileno()], [], [], 1)
1994 if rd: self.handle_request()
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001995
1996 def server_activate(self):
1997 self.base.server_activate(self)
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001998 if self.callback: self.callback(self)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001999
2000 DocServer.base = BaseHTTPServer.HTTPServer
2001 DocServer.handler = DocHandler
2002 DocHandler.MessageClass = Message
2003 try:
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00002004 try:
2005 DocServer(port, callback).serve_until_quit()
2006 except (KeyboardInterrupt, select.error):
2007 pass
2008 finally:
Ka-Ping Yee66246962001-04-12 11:59:50 +00002009 if completer: completer()
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002010
2011# ----------------------------------------------------- graphical interface
2012
2013def gui():
2014 """Graphical interface (starts web server and pops up a control window)."""
2015 class GUI:
2016 def __init__(self, window, port=7464):
2017 self.window = window
2018 self.server = None
2019 self.scanner = None
2020
Georg Brandl14fc4272008-05-17 18:39:55 +00002021 import tkinter
2022 self.server_frm = tkinter.Frame(window)
2023 self.title_lbl = tkinter.Label(self.server_frm,
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002024 text='Starting server...\n ')
Georg Brandl14fc4272008-05-17 18:39:55 +00002025 self.open_btn = tkinter.Button(self.server_frm,
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002026 text='open browser', command=self.open, state='disabled')
Georg Brandl14fc4272008-05-17 18:39:55 +00002027 self.quit_btn = tkinter.Button(self.server_frm,
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002028 text='quit serving', command=self.quit, state='disabled')
2029
Georg Brandl14fc4272008-05-17 18:39:55 +00002030 self.search_frm = tkinter.Frame(window)
2031 self.search_lbl = tkinter.Label(self.search_frm, text='Search for')
2032 self.search_ent = tkinter.Entry(self.search_frm)
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002033 self.search_ent.bind('<Return>', self.search)
Georg Brandl14fc4272008-05-17 18:39:55 +00002034 self.stop_btn = tkinter.Button(self.search_frm,
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002035 text='stop', pady=0, command=self.stop, state='disabled')
2036 if sys.platform == 'win32':
Ka-Ping Yeec92cdf72001-03-02 05:54:35 +00002037 # Trying to hide and show this button crashes under Windows.
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002038 self.stop_btn.pack(side='right')
2039
2040 self.window.title('pydoc')
2041 self.window.protocol('WM_DELETE_WINDOW', self.quit)
2042 self.title_lbl.pack(side='top', fill='x')
2043 self.open_btn.pack(side='left', fill='x', expand=1)
2044 self.quit_btn.pack(side='right', fill='x', expand=1)
2045 self.server_frm.pack(side='top', fill='x')
2046
2047 self.search_lbl.pack(side='left')
2048 self.search_ent.pack(side='right', fill='x', expand=1)
2049 self.search_frm.pack(side='top', fill='x')
2050 self.search_ent.focus_set()
2051
Ka-Ping Yeec92cdf72001-03-02 05:54:35 +00002052 font = ('helvetica', sys.platform == 'win32' and 8 or 10)
Georg Brandl14fc4272008-05-17 18:39:55 +00002053 self.result_lst = tkinter.Listbox(window, font=font, height=6)
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002054 self.result_lst.bind('<Button-1>', self.select)
2055 self.result_lst.bind('<Double-Button-1>', self.goto)
Georg Brandl14fc4272008-05-17 18:39:55 +00002056 self.result_scr = tkinter.Scrollbar(window,
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002057 orient='vertical', command=self.result_lst.yview)
2058 self.result_lst.config(yscrollcommand=self.result_scr.set)
2059
Georg Brandl14fc4272008-05-17 18:39:55 +00002060 self.result_frm = tkinter.Frame(window)
2061 self.goto_btn = tkinter.Button(self.result_frm,
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002062 text='go to selected', command=self.goto)
Georg Brandl14fc4272008-05-17 18:39:55 +00002063 self.hide_btn = tkinter.Button(self.result_frm,
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002064 text='hide results', command=self.hide)
2065 self.goto_btn.pack(side='left', fill='x', expand=1)
2066 self.hide_btn.pack(side='right', fill='x', expand=1)
2067
2068 self.window.update()
2069 self.minwidth = self.window.winfo_width()
2070 self.minheight = self.window.winfo_height()
2071 self.bigminheight = (self.server_frm.winfo_reqheight() +
2072 self.search_frm.winfo_reqheight() +
2073 self.result_lst.winfo_reqheight() +
2074 self.result_frm.winfo_reqheight())
2075 self.bigwidth, self.bigheight = self.minwidth, self.bigminheight
2076 self.expanded = 0
2077 self.window.wm_geometry('%dx%d' % (self.minwidth, self.minheight))
2078 self.window.wm_minsize(self.minwidth, self.minheight)
Martin v. Löwis5b26abb2002-12-28 09:23:09 +00002079 self.window.tk.willdispatch()
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002080
2081 import threading
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00002082 threading.Thread(
2083 target=serve, args=(port, self.ready, self.quit)).start()
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002084
2085 def ready(self, server):
2086 self.server = server
2087 self.title_lbl.config(
2088 text='Python documentation server at\n' + server.url)
2089 self.open_btn.config(state='normal')
2090 self.quit_btn.config(state='normal')
2091
Ka-Ping Yee239432a2001-03-02 02:45:08 +00002092 def open(self, event=None, url=None):
2093 url = url or self.server.url
2094 try:
2095 import webbrowser
2096 webbrowser.open(url)
2097 except ImportError: # pre-webbrowser.py compatibility
Ka-Ping Yeec92cdf72001-03-02 05:54:35 +00002098 if sys.platform == 'win32':
Ka-Ping Yee239432a2001-03-02 02:45:08 +00002099 os.system('start "%s"' % url)
2100 elif sys.platform == 'mac':
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +00002101 try: import ic
Ka-Ping Yee239432a2001-03-02 02:45:08 +00002102 except ImportError: pass
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +00002103 else: ic.launchurl(url)
Ka-Ping Yee239432a2001-03-02 02:45:08 +00002104 else:
2105 rc = os.system('netscape -remote "openURL(%s)" &' % url)
2106 if rc: os.system('netscape "%s" &' % url)
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002107
2108 def quit(self, event=None):
2109 if self.server:
2110 self.server.quit = 1
2111 self.window.quit()
2112
2113 def search(self, event=None):
2114 key = self.search_ent.get()
2115 self.stop_btn.pack(side='right')
2116 self.stop_btn.config(state='normal')
2117 self.search_lbl.config(text='Searching for "%s"...' % key)
2118 self.search_ent.forget()
2119 self.search_lbl.pack(side='left')
2120 self.result_lst.delete(0, 'end')
2121 self.goto_btn.config(state='disabled')
2122 self.expand()
2123
2124 import threading
2125 if self.scanner:
2126 self.scanner.quit = 1
2127 self.scanner = ModuleScanner()
2128 threading.Thread(target=self.scanner.run,
Ka-Ping Yee6dcfa382001-04-12 20:27:31 +00002129 args=(self.update, key, self.done)).start()
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002130
2131 def update(self, path, modname, desc):
2132 if modname[-9:] == '.__init__':
2133 modname = modname[:-9] + ' (package)'
2134 self.result_lst.insert('end',
2135 modname + ' - ' + (desc or '(no description)'))
2136
2137 def stop(self, event=None):
2138 if self.scanner:
2139 self.scanner.quit = 1
2140 self.scanner = None
2141
2142 def done(self):
2143 self.scanner = None
2144 self.search_lbl.config(text='Search for')
2145 self.search_lbl.pack(side='left')
2146 self.search_ent.pack(side='right', fill='x', expand=1)
2147 if sys.platform != 'win32': self.stop_btn.forget()
2148 self.stop_btn.config(state='disabled')
2149
2150 def select(self, event=None):
2151 self.goto_btn.config(state='normal')
2152
2153 def goto(self, event=None):
2154 selection = self.result_lst.curselection()
2155 if selection:
Neal Norwitz9d72bb42007-04-17 08:48:32 +00002156 modname = self.result_lst.get(selection[0]).split()[0]
Ka-Ping Yee239432a2001-03-02 02:45:08 +00002157 self.open(url=self.server.url + modname + '.html')
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002158
2159 def collapse(self):
2160 if not self.expanded: return
2161 self.result_frm.forget()
2162 self.result_scr.forget()
2163 self.result_lst.forget()
2164 self.bigwidth = self.window.winfo_width()
2165 self.bigheight = self.window.winfo_height()
2166 self.window.wm_geometry('%dx%d' % (self.minwidth, self.minheight))
2167 self.window.wm_minsize(self.minwidth, self.minheight)
2168 self.expanded = 0
2169
2170 def expand(self):
2171 if self.expanded: return
2172 self.result_frm.pack(side='bottom', fill='x')
2173 self.result_scr.pack(side='right', fill='y')
2174 self.result_lst.pack(side='top', fill='both', expand=1)
2175 self.window.wm_geometry('%dx%d' % (self.bigwidth, self.bigheight))
2176 self.window.wm_minsize(self.minwidth, self.bigminheight)
2177 self.expanded = 1
2178
2179 def hide(self, event=None):
2180 self.stop()
2181 self.collapse()
2182
Georg Brandl14fc4272008-05-17 18:39:55 +00002183 import tkinter
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002184 try:
Georg Brandl14fc4272008-05-17 18:39:55 +00002185 root = tkinter.Tk()
Martin v. Löwise09bd932004-08-22 16:13:26 +00002186 # Tk will crash if pythonw.exe has an XP .manifest
2187 # file and the root has is not destroyed explicitly.
2188 # If the problem is ever fixed in Tk, the explicit
2189 # destroy can go.
2190 try:
2191 gui = GUI(root)
2192 root.mainloop()
2193 finally:
2194 root.destroy()
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002195 except KeyboardInterrupt:
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002196 pass
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002197
2198# -------------------------------------------------- command-line interface
2199
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00002200def ispath(x):
Neal Norwitz9d72bb42007-04-17 08:48:32 +00002201 return isinstance(x, str) and x.find(os.sep) >= 0
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00002202
Ka-Ping Yee1d384632001-03-01 00:24:32 +00002203def cli():
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002204 """Command-line interface (looks at sys.argv to decide what to do)."""
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002205 import getopt
Guido van Rossum756aa932007-04-07 03:04:01 +00002206 class BadUsage(Exception): pass
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002207
Ka-Ping Yee3bda8792001-03-23 13:17:50 +00002208 # Scripts don't get the current directory in their path by default.
Ka-Ping Yeef78a81b2001-03-27 08:13:42 +00002209 scriptdir = os.path.dirname(sys.argv[0])
2210 if scriptdir in sys.path:
2211 sys.path.remove(scriptdir)
Ka-Ping Yee3bda8792001-03-23 13:17:50 +00002212 sys.path.insert(0, '.')
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002213
Ka-Ping Yee3bda8792001-03-23 13:17:50 +00002214 try:
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002215 opts, args = getopt.getopt(sys.argv[1:], 'gk:p:w')
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002216 writing = 0
2217
2218 for opt, val in opts:
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002219 if opt == '-g':
2220 gui()
2221 return
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002222 if opt == '-k':
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002223 apropos(val)
2224 return
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002225 if opt == '-p':
2226 try:
2227 port = int(val)
2228 except ValueError:
2229 raise BadUsage
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002230 def ready(server):
Guido van Rossumbe19ed72007-02-09 05:37:30 +00002231 print('pydoc server ready at %s' % server.url)
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00002232 def stopped():
Guido van Rossumbe19ed72007-02-09 05:37:30 +00002233 print('pydoc server stopped')
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00002234 serve(port, ready, stopped)
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002235 return
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002236 if opt == '-w':
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002237 writing = 1
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002238
2239 if not args: raise BadUsage
2240 for arg in args:
Ka-Ping Yee45daeb02002-08-11 15:11:33 +00002241 if ispath(arg) and not os.path.exists(arg):
Guido van Rossumbe19ed72007-02-09 05:37:30 +00002242 print('file %r does not exist' % arg)
Ka-Ping Yee45daeb02002-08-11 15:11:33 +00002243 break
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002244 try:
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00002245 if ispath(arg) and os.path.isfile(arg):
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002246 arg = importfile(arg)
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00002247 if writing:
2248 if ispath(arg) and os.path.isdir(arg):
2249 writedocs(arg)
2250 else:
2251 writedoc(arg)
2252 else:
Martin v. Löwisb8c084e2003-06-14 09:03:46 +00002253 help.help(arg)
Guido van Rossumb940e112007-01-10 16:19:56 +00002254 except ErrorDuringImport as value:
Guido van Rossumbe19ed72007-02-09 05:37:30 +00002255 print(value)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002256
2257 except (getopt.error, BadUsage):
Martin v. Löwisf9b08b82003-10-31 13:05:21 +00002258 cmd = os.path.basename(sys.argv[0])
Guido van Rossumbe19ed72007-02-09 05:37:30 +00002259 print("""pydoc - the Python documentation tool
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002260
2261%s <name> ...
2262 Show text documentation on something. <name> may be the name of a
Martin v. Löwisb8c084e2003-06-14 09:03:46 +00002263 Python keyword, topic, function, module, or package, or a dotted
2264 reference to a class or function within a module or module in a
2265 package. If <name> contains a '%s', it is used as the path to a
2266 Python source file to document. If name is 'keywords', 'topics',
2267 or 'modules', a listing of these things is displayed.
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002268
2269%s -k <keyword>
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002270 Search for a keyword in the synopsis lines of all available modules.
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002271
2272%s -p <port>
2273 Start an HTTP server on the given port on the local machine.
2274
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002275%s -g
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +00002276 Pop up a graphical interface for finding and serving documentation.
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002277
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002278%s -w <name> ...
2279 Write out the HTML documentation for a module to a file in the current
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +00002280 directory. If <name> contains a '%s', it is treated as a filename; if
2281 it names a directory, documentation is written for all the contents.
Guido van Rossumbe19ed72007-02-09 05:37:30 +00002282""" % (cmd, os.sep, cmd, cmd, cmd, cmd, os.sep))
Ka-Ping Yee1d384632001-03-01 00:24:32 +00002283
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002284if __name__ == '__main__': cli()