blob: 389c635974994f207a56672ef078a1f86fb4ba73 [file] [log] [blame]
Benjamin Peterson90f5ba52010-03-11 22:53:45 +00001#!/usr/bin/env python3
Ka-Ping Yee1d384632001-03-01 00:24:32 +00002"""Generate Python documentation in HTML or text for interactive use.
Ka-Ping Yeedd175342001-02-27 14:43:46 +00003
Ka-Ping Yeedd175342001-02-27 14:43:46 +00004In the Python interpreter, do "from pydoc import help" to provide online
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00005help. Calling help(thing) on a Python object documents the object.
6
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00007Or, at the shell command line outside of Python:
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00008
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00009Run "pydoc <name>" to show documentation on something. <name> may be
10the name of a function, module, package, or a dotted reference to a
11class or function within a module or module in a package. If the
12argument contains a path segment delimiter (e.g. slash on Unix,
13backslash on Windows) it is treated as the path to a Python source file.
Ka-Ping Yee66efbc72001-03-01 13:55:20 +000014
Ka-Ping Yee37f7b382001-03-23 00:12:53 +000015Run "pydoc -k <keyword>" to search for a keyword in the synopsis lines
16of all available modules.
Ka-Ping Yee66efbc72001-03-01 13:55:20 +000017
Ka-Ping Yee37f7b382001-03-23 00:12:53 +000018Run "pydoc -p <port>" to start an HTTP server on a given port on the
19local machine to generate documentation web pages.
Ka-Ping Yee66efbc72001-03-01 13:55:20 +000020
Ka-Ping Yee37f7b382001-03-23 00:12:53 +000021For platforms without a command line, "pydoc -g" starts the HTTP server
22and also pops up a little window for controlling it.
23
24Run "pydoc -w <name>" to write out the HTML documentation for a module
25to a file named "<name>.html".
Skip Montanaro4997a692003-09-10 16:47:51 +000026
27Module docs for core modules are assumed to be in
28
Alexander Belopolskya47bbf52010-11-18 01:52:54 +000029 http://docs.python.org/X.Y/library/
Skip Montanaro4997a692003-09-10 16:47:51 +000030
31This can be overridden by setting the PYTHONDOCS environment variable
32to a different URL or to a local directory containing the Library
33Reference Manual pages.
Ka-Ping Yee66efbc72001-03-01 13:55:20 +000034"""
Alexander Belopolskya47bbf52010-11-18 01:52:54 +000035__all__ = ['help']
Ka-Ping Yeedd175342001-02-27 14:43:46 +000036__author__ = "Ka-Ping Yee <ping@lfw.org>"
Ka-Ping Yee6f3f9a42001-02-27 22:42:36 +000037__date__ = "26 February 2001"
Jeremy Hylton3e0055f2005-10-20 19:59:25 +000038
Ka-Ping Yee09d7d9a2001-02-27 22:43:48 +000039__version__ = "$Revision$"
Martin v. Löwis6fe8f192004-11-14 10:21:04 +000040__credits__ = """Guido van Rossum, for an excellent programming language.
Ka-Ping Yee5e2b1732001-02-27 23:35:09 +000041Tommy Burnette, the original creator of manpy.
Ka-Ping Yee6f3f9a42001-02-27 22:42:36 +000042Paul Prescod, for all his work on onlinehelp.
43Richard Chamberlain, for the first implementation of textdoc.
Raymond Hettingered2dbe32005-01-01 07:51:01 +000044"""
Ka-Ping Yeedd175342001-02-27 14:43:46 +000045
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +000046# Known bugs that can't be fixed here:
47# - imp.load_module() cannot be prevented from clobbering existing
48# loaded modules, so calling synopsis() on a binary module file
49# changes the contents of any existing module with the same name.
50# - If the __file__ attribute on a module is a relative path and
51# the current directory is changed with os.chdir(), an incorrect
52# path will be displayed.
Ka-Ping Yee66efbc72001-03-01 13:55:20 +000053
Georg Brandl1a3284e2007-12-02 09:40:06 +000054import sys, imp, os, re, inspect, builtins, pkgutil
Alexandre Vassalotti1f2ba4b2008-05-16 07:12:44 +000055from reprlib import Repr
Benjamin Peterson0289b152009-06-28 17:22:03 +000056from traceback import extract_tb as _extract_tb
Alexander Belopolskya47bbf52010-11-18 01:52:54 +000057from collections import deque
Ka-Ping Yeedd175342001-02-27 14:43:46 +000058# --------------------------------------------------------- common routines
59
Ka-Ping Yeedd175342001-02-27 14:43:46 +000060def pathdirs():
61 """Convert sys.path into a list of absolute, existing, unique paths."""
62 dirs = []
Ka-Ping Yee1d384632001-03-01 00:24:32 +000063 normdirs = []
Ka-Ping Yeedd175342001-02-27 14:43:46 +000064 for dir in sys.path:
65 dir = os.path.abspath(dir or '.')
Ka-Ping Yee1d384632001-03-01 00:24:32 +000066 normdir = os.path.normcase(dir)
67 if normdir not in normdirs and os.path.isdir(dir):
Ka-Ping Yeedd175342001-02-27 14:43:46 +000068 dirs.append(dir)
Ka-Ping Yee1d384632001-03-01 00:24:32 +000069 normdirs.append(normdir)
Ka-Ping Yeedd175342001-02-27 14:43:46 +000070 return dirs
71
72def getdoc(object):
73 """Get the doc string or comments for an object."""
Ka-Ping Yee3bda8792001-03-23 13:17:50 +000074 result = inspect.getdoc(object) or inspect.getcomments(object)
Neal Norwitz9d72bb42007-04-17 08:48:32 +000075 return result and re.sub('^ *\n', '', result.rstrip()) or ''
Ka-Ping Yeedd175342001-02-27 14:43:46 +000076
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +000077def splitdoc(doc):
78 """Split a doc string into a synopsis line (if any) and the rest."""
Neal Norwitz9d72bb42007-04-17 08:48:32 +000079 lines = doc.strip().split('\n')
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +000080 if len(lines) == 1:
81 return lines[0], ''
Neal Norwitz9d72bb42007-04-17 08:48:32 +000082 elif len(lines) >= 2 and not lines[1].rstrip():
83 return lines[0], '\n'.join(lines[2:])
84 return '', '\n'.join(lines)
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +000085
Ka-Ping Yeedd175342001-02-27 14:43:46 +000086def classname(object, modname):
87 """Get a class name and qualify it with a module name if necessary."""
88 name = object.__name__
89 if object.__module__ != modname:
90 name = object.__module__ + '.' + name
91 return name
92
Ka-Ping Yeedec96e92001-04-13 09:55:49 +000093def isdata(object):
Georg Brandl94432422005-07-22 21:52:25 +000094 """Check if an object is of a type that probably means it's data."""
Ka-Ping Yeedec96e92001-04-13 09:55:49 +000095 return not (inspect.ismodule(object) or inspect.isclass(object) or
96 inspect.isroutine(object) or inspect.isframe(object) or
97 inspect.istraceback(object) or inspect.iscode(object))
Ka-Ping Yeedd175342001-02-27 14:43:46 +000098
99def replace(text, *pairs):
100 """Do a series of global replacements on a string."""
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +0000101 while pairs:
Neal Norwitz9d72bb42007-04-17 08:48:32 +0000102 text = pairs[1].join(text.split(pairs[0]))
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +0000103 pairs = pairs[2:]
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000104 return text
105
106def cram(text, maxlen):
107 """Omit part of a string if needed to make it fit in a maximum length."""
108 if len(text) > maxlen:
Raymond Hettingerfca3bb62002-10-21 04:44:11 +0000109 pre = max(0, (maxlen-3)//2)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000110 post = max(0, maxlen-3-pre)
111 return text[:pre] + '...' + text[len(text)-post:]
112 return text
113
Brett Cannon84601f12004-06-19 01:22:48 +0000114_re_stripid = re.compile(r' at 0x[0-9a-f]{6,16}(>+)$', re.IGNORECASE)
Ka-Ping Yee1d384632001-03-01 00:24:32 +0000115def stripid(text):
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000116 """Remove the hexadecimal id from a Python object representation."""
Brett Cannonc6c1f472004-06-19 01:02:51 +0000117 # The behaviour of %p is implementation-dependent in terms of case.
Ezio Melotti412c95a2010-02-16 23:31:04 +0000118 return _re_stripid.sub(r'\1', text)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000119
Brett Cannonc6c1f472004-06-19 01:02:51 +0000120def _is_some_method(obj):
121 return inspect.ismethod(obj) or inspect.ismethoddescriptor(obj)
Tim Peters536d2262001-09-20 05:13:38 +0000122
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000123def allmethods(cl):
124 methods = {}
Tim Peters536d2262001-09-20 05:13:38 +0000125 for key, value in inspect.getmembers(cl, _is_some_method):
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000126 methods[key] = 1
127 for base in cl.__bases__:
128 methods.update(allmethods(base)) # all your base are belong to us
129 for key in methods.keys():
130 methods[key] = getattr(cl, key)
131 return methods
132
Tim Petersfa26f7c2001-09-24 08:05:11 +0000133def _split_list(s, predicate):
134 """Split sequence s via predicate, and return pair ([true], [false]).
135
136 The return value is a 2-tuple of lists,
137 ([x for x in s if predicate(x)],
138 [x for x in s if not predicate(x)])
139 """
140
Tim Peters28355492001-09-23 21:29:55 +0000141 yes = []
142 no = []
Tim Petersfa26f7c2001-09-24 08:05:11 +0000143 for x in s:
144 if predicate(x):
145 yes.append(x)
Tim Peters28355492001-09-23 21:29:55 +0000146 else:
Tim Petersfa26f7c2001-09-24 08:05:11 +0000147 no.append(x)
Tim Peters28355492001-09-23 21:29:55 +0000148 return yes, no
149
Skip Montanaroa5616d22004-06-11 04:46:12 +0000150def visiblename(name, all=None):
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000151 """Decide whether to show documentation on a variable."""
152 # Certain special names are redundant.
Benjamin Peterson41181742008-07-02 20:22:54 +0000153 _hidden_names = ('__builtins__', '__doc__', '__file__', '__path__',
Barry Warsaw28a691b2010-04-17 00:19:56 +0000154 '__module__', '__name__', '__slots__', '__package__',
Alexander Belopolskya47bbf52010-11-18 01:52:54 +0000155 '__cached__', '__author__', '__credits__', '__date__',
156 '__version__')
Benjamin Peterson41181742008-07-02 20:22:54 +0000157 if name in _hidden_names: return 0
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000158 # Private names are hidden, but special names are displayed.
159 if name.startswith('__') and name.endswith('__'): return 1
Skip Montanaroa5616d22004-06-11 04:46:12 +0000160 if all is not None:
161 # only document that which the programmer exported in __all__
162 return name in all
163 else:
164 return not name.startswith('_')
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000165
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +0000166def classify_class_attrs(object):
167 """Wrap inspect.classify_class_attrs, with fixup for data descriptors."""
Guido van Rossum1bc535d2007-05-15 18:46:22 +0000168 results = []
169 for (name, kind, cls, value) in inspect.classify_class_attrs(object):
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +0000170 if inspect.isdatadescriptor(value):
171 kind = 'data descriptor'
Guido van Rossum1bc535d2007-05-15 18:46:22 +0000172 results.append((name, kind, cls, value))
173 return results
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +0000174
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000175# ----------------------------------------------------- module manipulation
176
177def ispackage(path):
178 """Guess whether a path refers to a package directory."""
179 if os.path.isdir(path):
Raymond Hettingerdbecd932005-02-06 06:57:08 +0000180 for ext in ('.py', '.pyc', '.pyo'):
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000181 if os.path.isfile(os.path.join(path, '__init__' + ext)):
Tim Petersbc0e9102002-04-04 22:55:58 +0000182 return True
183 return False
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000184
Thomas Wouters49fd7fa2006-04-21 10:40:58 +0000185def source_synopsis(file):
186 line = file.readline()
Neal Norwitz9d72bb42007-04-17 08:48:32 +0000187 while line[:1] == '#' or not line.strip():
Thomas Wouters49fd7fa2006-04-21 10:40:58 +0000188 line = file.readline()
189 if not line: break
Neal Norwitz9d72bb42007-04-17 08:48:32 +0000190 line = line.strip()
Thomas Wouters49fd7fa2006-04-21 10:40:58 +0000191 if line[:4] == 'r"""': line = line[1:]
192 if line[:3] == '"""':
193 line = line[3:]
194 if line[-1:] == '\\': line = line[:-1]
Neal Norwitz9d72bb42007-04-17 08:48:32 +0000195 while not line.strip():
Thomas Wouters49fd7fa2006-04-21 10:40:58 +0000196 line = file.readline()
197 if not line: break
Neal Norwitz9d72bb42007-04-17 08:48:32 +0000198 result = line.split('"""')[0].strip()
Thomas Wouters49fd7fa2006-04-21 10:40:58 +0000199 else: result = None
200 return result
201
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000202def synopsis(filename, cache={}):
203 """Get the one-line summary out of a module file."""
Raymond Hettinger32200ae2002-06-01 19:51:15 +0000204 mtime = os.stat(filename).st_mtime
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000205 lastupdate, result = cache.get(filename, (0, None))
206 if lastupdate < mtime:
207 info = inspect.getmoduleinfo(filename)
Georg Brandl26fd2e12006-03-08 09:34:53 +0000208 try:
209 file = open(filename)
210 except IOError:
211 # module can't be opened, so skip it
212 return None
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000213 if info and 'b' in info[2]: # binary modules have to be imported
214 try: module = imp.load_module('__temp__', file, filename, info[1:])
215 except: return None
Thomas Wouters49fd7fa2006-04-21 10:40:58 +0000216 result = (module.__doc__ or '').splitlines()[0]
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000217 del sys.modules['__temp__']
218 else: # text modules can be directly examined
Thomas Wouters49fd7fa2006-04-21 10:40:58 +0000219 result = source_synopsis(file)
220 file.close()
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000221 cache[filename] = (mtime, result)
222 return result
223
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000224class ErrorDuringImport(Exception):
225 """Errors that occurred while trying to import something to document it."""
Guido van Rossum1bc535d2007-05-15 18:46:22 +0000226 def __init__(self, filename, exc_info):
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000227 self.filename = filename
Guido van Rossum1bc535d2007-05-15 18:46:22 +0000228 self.exc, self.value, self.tb = exc_info
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000229
230 def __str__(self):
Guido van Rossuma01a8b62007-05-27 09:20:14 +0000231 exc = self.exc.__name__
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000232 return 'problem in %s - %s: %s' % (self.filename, exc, self.value)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000233
234def importfile(path):
235 """Import a Python source file or compiled file given its path."""
236 magic = imp.get_magic()
237 file = open(path, 'r')
238 if file.read(len(magic)) == magic:
239 kind = imp.PY_COMPILED
240 else:
241 kind = imp.PY_SOURCE
242 file.close()
243 filename = os.path.basename(path)
244 name, ext = os.path.splitext(filename)
245 file = open(path, 'r')
246 try:
247 module = imp.load_module(name, file, path, (ext, 'r', kind))
248 except:
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000249 raise ErrorDuringImport(path, sys.exc_info())
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000250 file.close()
251 return module
252
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000253def safeimport(path, forceload=0, cache={}):
254 """Import a module; handle errors; return None if the module isn't found.
255
256 If the module *is* found but an exception occurs, it's wrapped in an
257 ErrorDuringImport exception and reraised. Unlike __import__, if a
258 package path is specified, the module at the end of the path is returned,
259 not the package at the beginning. If the optional 'forceload' argument
260 is 1, we reload the module from disk (unless it's a dynamic extension)."""
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000261 try:
Ka-Ping Yee9a2dcf82005-11-05 05:04:41 +0000262 # If forceload is 1 and the module has been previously loaded from
263 # disk, we always have to reload the module. Checking the file's
264 # mtime isn't good enough (e.g. the module could contain a class
265 # that inherits from another module that has changed).
266 if forceload and path in sys.modules:
267 if path not in sys.builtin_module_names:
Guido van Rossume7ba4952007-06-06 23:52:48 +0000268 # Remove the module from sys.modules and re-import to try
269 # and avoid problems with partially loaded modules.
270 # Also remove any submodules because they won't appear
271 # in the newly loaded module's namespace if they're already
272 # in sys.modules.
Ka-Ping Yee9a2dcf82005-11-05 05:04:41 +0000273 subs = [m for m in sys.modules if m.startswith(path + '.')]
274 for key in [path] + subs:
275 # Prevent garbage collection.
276 cache[key] = sys.modules[key]
277 del sys.modules[key]
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000278 module = __import__(path)
279 except:
280 # Did the error occur before or after the module was found?
281 (exc, value, tb) = info = sys.exc_info()
Raymond Hettinger54f02222002-06-01 14:18:47 +0000282 if path in sys.modules:
Fred Drakedb390c12005-10-28 14:39:47 +0000283 # An error occurred while executing the imported module.
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000284 raise ErrorDuringImport(sys.modules[path].__file__, info)
285 elif exc is SyntaxError:
286 # A SyntaxError occurred before we could execute the module.
287 raise ErrorDuringImport(value.filename, info)
Benjamin Peterson0289b152009-06-28 17:22:03 +0000288 elif exc is ImportError and _extract_tb(tb)[-1][2]=='safeimport':
289 # The import error occurred directly in this function,
290 # which means there is no such module in the path.
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000291 return None
292 else:
293 # Some other error occurred during the importing process.
294 raise ErrorDuringImport(path, sys.exc_info())
Neal Norwitz9d72bb42007-04-17 08:48:32 +0000295 for part in path.split('.')[1:]:
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000296 try: module = getattr(module, part)
297 except AttributeError: return None
298 return module
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000299
300# ---------------------------------------------------- formatter base class
301
302class Doc:
Alexander Belopolskya47bbf52010-11-18 01:52:54 +0000303
304 PYTHONDOCS = os.environ.get("PYTHONDOCS",
305 "http://docs.python.org/%d.%d/library"
306 % sys.version_info[:2])
307
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000308 def document(self, object, name=None, *args):
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000309 """Generate documentation for an object."""
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000310 args = (object, name) + args
Brett Cannon28a4f0f2003-06-11 23:38:55 +0000311 # 'try' clause is to attempt to handle the possibility that inspect
312 # identifies something in a way that pydoc itself has issues handling;
313 # think 'super' and how it is a descriptor (which raises the exception
314 # by lacking a __name__ attribute) and an instance.
Thomas Wouters0e3f5912006-08-11 14:57:12 +0000315 if inspect.isgetsetdescriptor(object): return self.docdata(*args)
316 if inspect.ismemberdescriptor(object): return self.docdata(*args)
Brett Cannon28a4f0f2003-06-11 23:38:55 +0000317 try:
318 if inspect.ismodule(object): return self.docmodule(*args)
319 if inspect.isclass(object): return self.docclass(*args)
320 if inspect.isroutine(object): return self.docroutine(*args)
321 except AttributeError:
322 pass
Johannes Gijsbers8de645a2004-11-07 19:16:05 +0000323 if isinstance(object, property): return self.docproperty(*args)
Guido van Rossum68468eb2003-02-27 20:14:51 +0000324 return self.docother(*args)
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000325
326 def fail(self, object, name=None, *args):
327 """Raise an exception for unimplemented types."""
328 message = "don't know how to document object%s of type %s" % (
329 name and ' ' + repr(name), type(object).__name__)
Collin Winterce36ad82007-08-30 01:19:48 +0000330 raise TypeError(message)
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000331
Thomas Wouters0e3f5912006-08-11 14:57:12 +0000332 docmodule = docclass = docroutine = docother = docproperty = docdata = fail
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000333
Skip Montanaro4997a692003-09-10 16:47:51 +0000334 def getdocloc(self, object):
335 """Return the location of module docs or None"""
336
337 try:
338 file = inspect.getabsfile(object)
339 except TypeError:
340 file = '(built-in)'
341
Alexander Belopolskya47bbf52010-11-18 01:52:54 +0000342 docloc = os.environ.get("PYTHONDOCS", self.PYTHONDOCS)
343
Skip Montanaro4997a692003-09-10 16:47:51 +0000344 basedir = os.path.join(sys.exec_prefix, "lib",
Alexander Belopolskya47bbf52010-11-18 01:52:54 +0000345 "python%d.%d" % sys.version_info[:2])
Skip Montanaro4997a692003-09-10 16:47:51 +0000346 if (isinstance(object, type(os)) and
347 (object.__name__ in ('errno', 'exceptions', 'gc', 'imp',
348 'marshal', 'posix', 'signal', 'sys',
Georg Brandl2067bfd2008-05-25 13:05:15 +0000349 '_thread', 'zipimport') or
Skip Montanaro4997a692003-09-10 16:47:51 +0000350 (file.startswith(basedir) and
Brian Curtin49c284c2010-03-31 03:19:28 +0000351 not file.startswith(os.path.join(basedir, 'site-packages')))) and
Brian Curtinedef05b2010-04-01 04:05:25 +0000352 object.__name__ not in ('xml.etree', 'test.pydoc_mod')):
Skip Montanaro4997a692003-09-10 16:47:51 +0000353 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....)+)',
Raymond Hettinger95f285c2009-02-24 23:41:47 +0000392 r'<font color="#c040c0">\1</font>',
Ka-Ping Yeea2fe1032001-03-02 01:19:14 +0000393 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."""
Georg Brandl388faac2009-04-10 08:31:48 +0000416 return '''\
Georg Brandle6066942009-04-10 08:28:28 +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>
Georg Brandl388faac2009-04-10 08:31:48 +0000419<meta http-equiv="Content-Type" content="text/html; charset=utf-8">
Raymond Hettinger95f285c2009-02-24 23:41:47 +0000420</head><body bgcolor="#f0f0f8">
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000421%s
422</body></html>''' % (title, contents)
423
424 def heading(self, title, fgcol, bgcol, extras=''):
425 """Format a page heading."""
426 return '''
Tim Peters59ed4482001-10-31 04:20:26 +0000427<table width="100%%" cellspacing=0 cellpadding=2 border=0 summary="heading">
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000428<tr bgcolor="%s">
Tim Peters2306d242001-09-25 03:18:32 +0000429<td valign=bottom>&nbsp;<br>
430<font color="%s" face="helvetica, arial">&nbsp;<br>%s</font></td
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000431><td align=right valign=bottom
Ka-Ping Yee987ec902001-03-23 13:35:45 +0000432><font color="%s" face="helvetica, arial">%s</font></td></tr></table>
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000433 ''' % (bgcol, fgcol, title, fgcol, extras or '&nbsp;')
434
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000435 def section(self, title, fgcol, bgcol, contents, width=6,
436 prelude='', marginalia=None, gap='&nbsp;'):
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000437 """Format a section with a heading."""
438 if marginalia is None:
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +0000439 marginalia = '<tt>' + '&nbsp;' * width + '</tt>'
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000440 result = '''<p>
Tim Peters59ed4482001-10-31 04:20:26 +0000441<table width="100%%" cellspacing=0 cellpadding=2 border=0 summary="section">
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000442<tr bgcolor="%s">
Tim Peters2306d242001-09-25 03:18:32 +0000443<td colspan=3 valign=bottom>&nbsp;<br>
444<font color="%s" face="helvetica, arial">%s</font></td></tr>
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000445 ''' % (bgcol, fgcol, title)
446 if prelude:
447 result = result + '''
Ka-Ping Yee987ec902001-03-23 13:35:45 +0000448<tr bgcolor="%s"><td rowspan=2>%s</td>
449<td colspan=2>%s</td></tr>
450<tr><td>%s</td>''' % (bgcol, marginalia, prelude, gap)
451 else:
452 result = result + '''
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000453<tr><td bgcolor="%s">%s</td><td>%s</td>''' % (bgcol, marginalia, gap)
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000454
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +0000455 return result + '\n<td width="100%%">%s</td></tr></table>' % contents
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000456
457 def bigsection(self, title, *args):
458 """Format a section with a big heading."""
Raymond Hettinger95f285c2009-02-24 23:41:47 +0000459 title = '<big><strong>%s</strong></big>' % title
Guido van Rossum68468eb2003-02-27 20:14:51 +0000460 return self.section(title, *args)
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000461
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000462 def preformat(self, text):
463 """Format literal preformatted text."""
Neal Norwitz9d72bb42007-04-17 08:48:32 +0000464 text = self.escape(text.expandtabs())
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +0000465 return replace(text, '\n\n', '\n \n', '\n\n', '\n \n',
466 ' ', '&nbsp;', '\n', '<br>\n')
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000467
468 def multicolumn(self, list, format, cols=4):
469 """Format a list of items into a multi-column list."""
470 result = ''
Guido van Rossumc1f779c2007-07-03 08:25:58 +0000471 rows = (len(list)+cols-1)//cols
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000472 for col in range(cols):
Guido van Rossumc1f779c2007-07-03 08:25:58 +0000473 result = result + '<td width="%d%%" valign=top>' % (100//cols)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000474 for i in range(rows*col, rows*col+rows):
475 if i < len(list):
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000476 result = result + format(list[i]) + '<br>\n'
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000477 result = result + '</td>'
Tim Peters59ed4482001-10-31 04:20:26 +0000478 return '<table width="100%%" summary="list"><tr>%s</tr></table>' % result
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000479
Raymond Hettinger95f285c2009-02-24 23:41:47 +0000480 def grey(self, text): return '<font color="#909090">%s</font>' % text
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000481
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000482 def namelink(self, name, *dicts):
483 """Make a link for an identifier, given name-to-URL mappings."""
484 for dict in dicts:
Raymond Hettinger54f02222002-06-01 14:18:47 +0000485 if name in dict:
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000486 return '<a href="%s">%s</a>' % (dict[name], name)
487 return name
488
Ka-Ping Yeeb7a48302001-04-12 20:39:14 +0000489 def classlink(self, object, modname):
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000490 """Make a link for a class."""
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000491 name, module = object.__name__, sys.modules.get(object.__module__)
492 if hasattr(module, name) and getattr(module, name) is object:
Ka-Ping Yeeb7a48302001-04-12 20:39:14 +0000493 return '<a href="%s.html#%s">%s</a>' % (
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000494 module.__name__, name, classname(object, modname))
495 return classname(object, modname)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000496
497 def modulelink(self, object):
498 """Make a link for a module."""
499 return '<a href="%s.html">%s</a>' % (object.__name__, object.__name__)
500
Guido van Rossum1bc535d2007-05-15 18:46:22 +0000501 def modpkglink(self, modpkginfo):
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000502 """Make a link for a module or package to display in an index."""
Guido van Rossum1bc535d2007-05-15 18:46:22 +0000503 name, path, ispackage, shadowed = modpkginfo
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000504 if shadowed:
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000505 return self.grey(name)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000506 if path:
507 url = '%s.%s.html' % (path, name)
508 else:
509 url = '%s.html' % name
510 if ispackage:
Raymond Hettinger95f285c2009-02-24 23:41:47 +0000511 text = '<strong>%s</strong>&nbsp;(package)' % name
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000512 else:
513 text = name
514 return '<a href="%s">%s</a>' % (url, text)
515
516 def markup(self, text, escape=None, funcs={}, classes={}, methods={}):
517 """Mark up some plain text, given a context of symbols to look for.
518 Each context dictionary maps object names to anchor names."""
519 escape = escape or self.escape
520 results = []
521 here = 0
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000522 pattern = re.compile(r'\b((http|ftp)://\S+[\w/]|'
523 r'RFC[- ]?(\d+)|'
Ka-Ping Yeef78a81b2001-03-27 08:13:42 +0000524 r'PEP[- ]?(\d+)|'
Neil Schemenauerd69711c2002-03-24 23:02:07 +0000525 r'(self\.)?(\w+))')
Guido van Rossum8ca162f2002-04-07 06:36:23 +0000526 while True:
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000527 match = pattern.search(text, here)
528 if not match: break
529 start, end = match.span()
530 results.append(escape(text[here:start]))
531
Ka-Ping Yeef78a81b2001-03-27 08:13:42 +0000532 all, scheme, rfc, pep, selfdot, name = match.groups()
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000533 if scheme:
Neil Schemenauercddc1a02002-03-24 23:11:21 +0000534 url = escape(all).replace('"', '&quot;')
535 results.append('<a href="%s">%s</a>' % (url, url))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000536 elif rfc:
Ka-Ping Yeef78a81b2001-03-27 08:13:42 +0000537 url = 'http://www.rfc-editor.org/rfc/rfc%d.txt' % int(rfc)
538 results.append('<a href="%s">%s</a>' % (url, escape(all)))
539 elif pep:
Christian Heimes2202f872008-02-06 14:31:34 +0000540 url = 'http://www.python.org/dev/peps/pep-%04d/' % int(pep)
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000541 results.append('<a href="%s">%s</a>' % (url, escape(all)))
542 elif text[end:end+1] == '(':
543 results.append(self.namelink(name, methods, funcs, classes))
544 elif selfdot:
Raymond Hettinger95f285c2009-02-24 23:41:47 +0000545 results.append('self.<strong>%s</strong>' % name)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000546 else:
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000547 results.append(self.namelink(name, classes))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000548 here = end
549 results.append(escape(text[here:]))
Neal Norwitz9d72bb42007-04-17 08:48:32 +0000550 return ''.join(results)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000551
552 # ---------------------------------------------- type-specific routines
553
Ka-Ping Yeeb7a48302001-04-12 20:39:14 +0000554 def formattree(self, tree, modname, parent=None):
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000555 """Produce HTML for a class tree as given by inspect.getclasstree()."""
556 result = ''
557 for entry in tree:
558 if type(entry) is type(()):
559 c, bases = entry
Raymond Hettinger95f285c2009-02-24 23:41:47 +0000560 result = result + '<dt><font face="helvetica, arial">'
Ka-Ping Yeeb7a48302001-04-12 20:39:14 +0000561 result = result + self.classlink(c, modname)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000562 if bases and bases != (parent,):
563 parents = []
564 for base in bases:
Ka-Ping Yeeb7a48302001-04-12 20:39:14 +0000565 parents.append(self.classlink(base, modname))
Neal Norwitz9d72bb42007-04-17 08:48:32 +0000566 result = result + '(' + ', '.join(parents) + ')'
Raymond Hettinger95f285c2009-02-24 23:41:47 +0000567 result = result + '\n</font></dt>'
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000568 elif type(entry) is type([]):
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000569 result = result + '<dd>\n%s</dd>\n' % self.formattree(
Ka-Ping Yeeb7a48302001-04-12 20:39:14 +0000570 entry, modname, c)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000571 return '<dl>\n%s</dl>\n' % result
572
Tim Peters8dd7ade2001-10-18 19:56:17 +0000573 def docmodule(self, object, name=None, mod=None, *ignored):
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000574 """Produce HTML documentation for a module object."""
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000575 name = object.__name__ # ignore the passed-in name
Skip Montanaroa5616d22004-06-11 04:46:12 +0000576 try:
577 all = object.__all__
578 except AttributeError:
579 all = None
Neal Norwitz9d72bb42007-04-17 08:48:32 +0000580 parts = name.split('.')
Ka-Ping Yee66efbc72001-03-01 13:55:20 +0000581 links = []
582 for i in range(len(parts)-1):
583 links.append(
Raymond Hettinger95f285c2009-02-24 23:41:47 +0000584 '<a href="%s.html"><font color="#ffffff">%s</font></a>' %
Neal Norwitz9d72bb42007-04-17 08:48:32 +0000585 ('.'.join(parts[:i+1]), parts[i]))
586 linkedname = '.'.join(links + parts[-1:])
Raymond Hettinger95f285c2009-02-24 23:41:47 +0000587 head = '<big><big><strong>%s</strong></big></big>' % linkedname
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000588 try:
Ka-Ping Yee239432a2001-03-02 02:45:08 +0000589 path = inspect.getabsfile(object)
Ka-Ping Yee6191a232001-04-13 15:00:27 +0000590 url = path
591 if sys.platform == 'win32':
592 import nturl2path
593 url = nturl2path.pathname2url(path)
594 filelink = '<a href="file:%s">%s</a>' % (url, path)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000595 except TypeError:
596 filelink = '(built-in)'
Ka-Ping Yee6f3f9a42001-02-27 22:42:36 +0000597 info = []
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000598 if hasattr(object, '__version__'):
Ka-Ping Yee6f3f9a42001-02-27 22:42:36 +0000599 version = str(object.__version__)
Ka-Ping Yee40c49912001-02-27 22:46:01 +0000600 if version[:11] == '$' + 'Revision: ' and version[-1:] == '$':
Neal Norwitz9d72bb42007-04-17 08:48:32 +0000601 version = version[11:-1].strip()
Ka-Ping Yee1d384632001-03-01 00:24:32 +0000602 info.append('version %s' % self.escape(version))
Ka-Ping Yee6f3f9a42001-02-27 22:42:36 +0000603 if hasattr(object, '__date__'):
604 info.append(self.escape(str(object.__date__)))
605 if info:
Neal Norwitz9d72bb42007-04-17 08:48:32 +0000606 head = head + ' (%s)' % ', '.join(info)
Skip Montanaro4997a692003-09-10 16:47:51 +0000607 docloc = self.getdocloc(object)
608 if docloc is not None:
Alexander Belopolskya47bbf52010-11-18 01:52:54 +0000609 docloc = '<br><a href="%(docloc)s">Module Reference</a>' % locals()
Skip Montanaro4997a692003-09-10 16:47:51 +0000610 else:
611 docloc = ''
Ka-Ping Yee66efbc72001-03-01 13:55:20 +0000612 result = self.heading(
Skip Montanaro4997a692003-09-10 16:47:51 +0000613 head, '#ffffff', '#7799ee',
614 '<a href=".">index</a><br>' + filelink + docloc)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000615
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000616 modules = inspect.getmembers(object, inspect.ismodule)
617
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000618 classes, cdict = [], {}
619 for key, value in inspect.getmembers(object, inspect.isclass):
Johannes Gijsbers4c11f602004-08-30 14:13:04 +0000620 # if __all__ exists, believe it. Otherwise use old heuristic.
621 if (all is not None or
622 (inspect.getmodule(value) or object) is object):
Skip Montanaroa5616d22004-06-11 04:46:12 +0000623 if visiblename(key, all):
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000624 classes.append((key, value))
625 cdict[key] = cdict[value] = '#' + key
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000626 for key, value in classes:
627 for base in value.__bases__:
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000628 key, modname = base.__name__, base.__module__
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000629 module = sys.modules.get(modname)
630 if modname != name and module and hasattr(module, key):
631 if getattr(module, key) is base:
Raymond Hettinger54f02222002-06-01 14:18:47 +0000632 if not key in cdict:
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000633 cdict[key] = cdict[base] = modname + '.html#' + key
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000634 funcs, fdict = [], {}
635 for key, value in inspect.getmembers(object, inspect.isroutine):
Johannes Gijsbers4c11f602004-08-30 14:13:04 +0000636 # if __all__ exists, believe it. Otherwise use old heuristic.
637 if (all is not None or
638 inspect.isbuiltin(value) or inspect.getmodule(value) is object):
Skip Montanaroa5616d22004-06-11 04:46:12 +0000639 if visiblename(key, all):
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000640 funcs.append((key, value))
641 fdict[key] = '#-' + key
642 if inspect.isfunction(value): fdict[value] = fdict[key]
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000643 data = []
644 for key, value in inspect.getmembers(object, isdata):
Skip Montanaroa5616d22004-06-11 04:46:12 +0000645 if visiblename(key, all):
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000646 data.append((key, value))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000647
648 doc = self.markup(getdoc(object), self.preformat, fdict, cdict)
649 doc = doc and '<tt>%s</tt>' % doc
Tim Peters2306d242001-09-25 03:18:32 +0000650 result = result + '<p>%s</p>\n' % doc
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000651
652 if hasattr(object, '__path__'):
653 modpkgs = []
Thomas Wouters49fd7fa2006-04-21 10:40:58 +0000654 for importer, modname, ispkg in pkgutil.iter_modules(object.__path__):
655 modpkgs.append((modname, name, ispkg, 0))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000656 modpkgs.sort()
657 contents = self.multicolumn(modpkgs, self.modpkglink)
658 result = result + self.bigsection(
659 'Package Contents', '#ffffff', '#aa55cc', contents)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000660 elif modules:
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000661 contents = self.multicolumn(
Guido van Rossum1bc535d2007-05-15 18:46:22 +0000662 modules, lambda t: self.modulelink(t[1]))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000663 result = result + self.bigsection(
Christian Heimes7131fd92008-02-19 14:21:46 +0000664 'Modules', '#ffffff', '#aa55cc', contents)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000665
666 if classes:
Guido van Rossum1bc535d2007-05-15 18:46:22 +0000667 classlist = [value for (key, value) in classes]
Ka-Ping Yeeb7a48302001-04-12 20:39:14 +0000668 contents = [
669 self.formattree(inspect.getclasstree(classlist, 1), name)]
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000670 for key, value in classes:
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +0000671 contents.append(self.document(value, key, name, fdict, cdict))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000672 result = result + self.bigsection(
Neal Norwitz9d72bb42007-04-17 08:48:32 +0000673 'Classes', '#ffffff', '#ee77aa', ' '.join(contents))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000674 if funcs:
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000675 contents = []
676 for key, value in funcs:
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +0000677 contents.append(self.document(value, key, name, fdict, cdict))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000678 result = result + self.bigsection(
Neal Norwitz9d72bb42007-04-17 08:48:32 +0000679 'Functions', '#ffffff', '#eeaa77', ' '.join(contents))
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000680 if data:
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000681 contents = []
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000682 for key, value in data:
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000683 contents.append(self.document(value, key))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000684 result = result + self.bigsection(
Neal Norwitz9d72bb42007-04-17 08:48:32 +0000685 'Data', '#ffffff', '#55aa55', '<br>\n'.join(contents))
Ka-Ping Yee6f3f9a42001-02-27 22:42:36 +0000686 if hasattr(object, '__author__'):
687 contents = self.markup(str(object.__author__), self.preformat)
688 result = result + self.bigsection(
689 'Author', '#ffffff', '#7799ee', contents)
Ka-Ping Yee6f3f9a42001-02-27 22:42:36 +0000690 if hasattr(object, '__credits__'):
691 contents = self.markup(str(object.__credits__), self.preformat)
692 result = result + self.bigsection(
693 'Credits', '#ffffff', '#7799ee', contents)
694
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000695 return result
696
Tim Peters8dd7ade2001-10-18 19:56:17 +0000697 def docclass(self, object, name=None, mod=None, funcs={}, classes={},
698 *ignored):
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000699 """Produce HTML documentation for a class object."""
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000700 realname = object.__name__
701 name = name or realname
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000702 bases = object.__bases__
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000703
Tim Petersb47879b2001-09-24 04:47:19 +0000704 contents = []
705 push = contents.append
706
Tim Petersfa26f7c2001-09-24 08:05:11 +0000707 # Cute little class to pump out a horizontal rule between sections.
708 class HorizontalRule:
709 def __init__(self):
710 self.needone = 0
711 def maybe(self):
712 if self.needone:
713 push('<hr>\n')
714 self.needone = 1
715 hr = HorizontalRule()
716
Tim Petersc86f6ca2001-09-26 21:31:51 +0000717 # List the mro, if non-trivial.
Raymond Hettinger756b3f32004-01-29 06:37:52 +0000718 mro = deque(inspect.getmro(object))
Tim Petersc86f6ca2001-09-26 21:31:51 +0000719 if len(mro) > 2:
720 hr.maybe()
721 push('<dl><dt>Method resolution order:</dt>\n')
722 for base in mro:
723 push('<dd>%s</dd>\n' % self.classlink(base,
724 object.__module__))
725 push('</dl>\n')
726
Tim Petersb47879b2001-09-24 04:47:19 +0000727 def spill(msg, attrs, predicate):
Tim Petersfa26f7c2001-09-24 08:05:11 +0000728 ok, attrs = _split_list(attrs, predicate)
Tim Petersb47879b2001-09-24 04:47:19 +0000729 if ok:
Tim Petersfa26f7c2001-09-24 08:05:11 +0000730 hr.maybe()
Tim Petersb47879b2001-09-24 04:47:19 +0000731 push(msg)
732 for name, kind, homecls, value in ok:
733 push(self.document(getattr(object, name), name, mod,
734 funcs, classes, mdict, object))
735 push('\n')
736 return attrs
737
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +0000738 def spilldescriptors(msg, attrs, predicate):
Tim Petersfa26f7c2001-09-24 08:05:11 +0000739 ok, attrs = _split_list(attrs, predicate)
Tim Petersb47879b2001-09-24 04:47:19 +0000740 if ok:
Tim Petersfa26f7c2001-09-24 08:05:11 +0000741 hr.maybe()
Tim Petersb47879b2001-09-24 04:47:19 +0000742 push(msg)
743 for name, kind, homecls, value in ok:
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +0000744 push(self._docdescriptor(name, value, mod))
Tim Petersb47879b2001-09-24 04:47:19 +0000745 return attrs
746
Tim Petersfa26f7c2001-09-24 08:05:11 +0000747 def spilldata(msg, attrs, predicate):
748 ok, attrs = _split_list(attrs, predicate)
Tim Petersb47879b2001-09-24 04:47:19 +0000749 if ok:
Tim Petersfa26f7c2001-09-24 08:05:11 +0000750 hr.maybe()
Tim Petersb47879b2001-09-24 04:47:19 +0000751 push(msg)
752 for name, kind, homecls, value in ok:
753 base = self.docother(getattr(object, name), name, mod)
Guido van Rossumd59da4b2007-05-22 18:11:13 +0000754 if hasattr(value, '__call__') or inspect.isdatadescriptor(value):
Guido van Rossum5e355b22002-05-21 20:56:15 +0000755 doc = getattr(value, "__doc__", None)
756 else:
757 doc = None
Tim Petersb47879b2001-09-24 04:47:19 +0000758 if doc is None:
759 push('<dl><dt>%s</dl>\n' % base)
760 else:
761 doc = self.markup(getdoc(value), self.preformat,
762 funcs, classes, mdict)
Tim Peters2306d242001-09-25 03:18:32 +0000763 doc = '<dd><tt>%s</tt>' % doc
Tim Petersb47879b2001-09-24 04:47:19 +0000764 push('<dl><dt>%s%s</dl>\n' % (base, doc))
765 push('\n')
766 return attrs
767
Guido van Rossum1bc535d2007-05-15 18:46:22 +0000768 attrs = [(name, kind, cls, value)
769 for name, kind, cls, value in classify_class_attrs(object)
770 if visiblename(name)]
771
Tim Petersb47879b2001-09-24 04:47:19 +0000772 mdict = {}
773 for key, kind, homecls, value in attrs:
774 mdict[key] = anchor = '#' + name + '-' + key
775 value = getattr(object, key)
776 try:
777 # The value may not be hashable (e.g., a data attr with
778 # a dict or list value).
779 mdict[value] = anchor
780 except TypeError:
781 pass
782
Tim Petersfa26f7c2001-09-24 08:05:11 +0000783 while attrs:
Tim Peters351e3622001-09-27 03:29:51 +0000784 if mro:
Raymond Hettinger756b3f32004-01-29 06:37:52 +0000785 thisclass = mro.popleft()
Tim Peters351e3622001-09-27 03:29:51 +0000786 else:
787 thisclass = attrs[0][2]
Tim Petersfa26f7c2001-09-24 08:05:11 +0000788 attrs, inherited = _split_list(attrs, lambda t: t[2] is thisclass)
789
Georg Brandl1a3284e2007-12-02 09:40:06 +0000790 if thisclass is builtins.object:
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000791 attrs = inherited
792 continue
793 elif thisclass is object:
794 tag = 'defined here'
Tim Petersb47879b2001-09-24 04:47:19 +0000795 else:
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000796 tag = 'inherited from %s' % self.classlink(thisclass,
797 object.__module__)
Tim Petersb47879b2001-09-24 04:47:19 +0000798 tag += ':<br>\n'
799
800 # Sort attrs by name.
Raymond Hettingerd4cb56d2008-01-30 02:55:10 +0000801 attrs.sort(key=lambda t: t[0])
Tim Petersb47879b2001-09-24 04:47:19 +0000802
803 # Pump out the attrs, segregated by kind.
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000804 attrs = spill('Methods %s' % tag, attrs,
Tim Petersb47879b2001-09-24 04:47:19 +0000805 lambda t: t[1] == 'method')
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000806 attrs = spill('Class methods %s' % tag, attrs,
Tim Petersb47879b2001-09-24 04:47:19 +0000807 lambda t: t[1] == 'class method')
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000808 attrs = spill('Static methods %s' % tag, attrs,
Tim Petersb47879b2001-09-24 04:47:19 +0000809 lambda t: t[1] == 'static method')
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +0000810 attrs = spilldescriptors('Data descriptors %s' % tag, attrs,
811 lambda t: t[1] == 'data descriptor')
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000812 attrs = spilldata('Data and other attributes %s' % tag, attrs,
Tim Petersfa26f7c2001-09-24 08:05:11 +0000813 lambda t: t[1] == 'data')
Tim Petersb47879b2001-09-24 04:47:19 +0000814 assert attrs == []
Tim Peters351e3622001-09-27 03:29:51 +0000815 attrs = inherited
Tim Petersb47879b2001-09-24 04:47:19 +0000816
817 contents = ''.join(contents)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000818
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000819 if name == realname:
820 title = '<a name="%s">class <strong>%s</strong></a>' % (
821 name, realname)
822 else:
823 title = '<strong>%s</strong> = <a name="%s">class %s</a>' % (
824 name, name, realname)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000825 if bases:
826 parents = []
827 for base in bases:
Ka-Ping Yeeb7a48302001-04-12 20:39:14 +0000828 parents.append(self.classlink(base, object.__module__))
Neal Norwitz9d72bb42007-04-17 08:48:32 +0000829 title = title + '(%s)' % ', '.join(parents)
Tim Peters2306d242001-09-25 03:18:32 +0000830 doc = self.markup(getdoc(object), self.preformat, funcs, classes, mdict)
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000831 doc = doc and '<tt>%s<br>&nbsp;</tt>' % doc
Tim Petersc86f6ca2001-09-26 21:31:51 +0000832
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000833 return self.section(title, '#000000', '#ffc8d8', contents, 3, doc)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000834
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000835 def formatvalue(self, object):
836 """Format an argument default value as text."""
Tim Peters2306d242001-09-25 03:18:32 +0000837 return self.grey('=' + self.repr(object))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000838
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +0000839 def docroutine(self, object, name=None, mod=None,
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000840 funcs={}, classes={}, methods={}, cl=None):
Ka-Ping Yee66efbc72001-03-01 13:55:20 +0000841 """Produce HTML documentation for a function or method object."""
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000842 realname = object.__name__
843 name = name or realname
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000844 anchor = (cl and cl.__name__ or '') + '-' + name
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000845 note = ''
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000846 skipdocs = 0
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000847 if inspect.ismethod(object):
Christian Heimesff737952007-11-27 10:40:20 +0000848 imclass = object.__self__.__class__
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000849 if cl:
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +0000850 if imclass is not cl:
Ka-Ping Yeeb7a48302001-04-12 20:39:14 +0000851 note = ' from ' + self.classlink(imclass, mod)
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000852 else:
Christian Heimesff737952007-11-27 10:40:20 +0000853 if object.__self__ is not None:
Ka-Ping Yeeb7a48302001-04-12 20:39:14 +0000854 note = ' method of %s instance' % self.classlink(
Christian Heimesff737952007-11-27 10:40:20 +0000855 object.__self__.__class__, mod)
Ka-Ping Yeeb7a48302001-04-12 20:39:14 +0000856 else:
857 note = ' unbound %s method' % self.classlink(imclass,mod)
Christian Heimesff737952007-11-27 10:40:20 +0000858 object = object.__func__
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000859
860 if name == realname:
861 title = '<a name="%s"><strong>%s</strong></a>' % (anchor, realname)
862 else:
Raymond Hettinger54f02222002-06-01 14:18:47 +0000863 if (cl and realname in cl.__dict__ and
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000864 cl.__dict__[realname] is object):
Ka-Ping Yeee280c062001-03-23 14:05:53 +0000865 reallink = '<a href="#%s">%s</a>' % (
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000866 cl.__name__ + '-' + realname, realname)
867 skipdocs = 1
868 else:
869 reallink = realname
870 title = '<a name="%s"><strong>%s</strong></a> = %s' % (
871 anchor, name, reallink)
Tim Peters4bcfa312001-09-20 06:08:24 +0000872 if inspect.isfunction(object):
Guido van Rossum2e65f892007-02-28 22:03:49 +0000873 args, varargs, kwonlyargs, kwdefaults, varkw, defaults, ann = \
874 inspect.getfullargspec(object)
Ka-Ping Yee66efbc72001-03-01 13:55:20 +0000875 argspec = inspect.formatargspec(
Guido van Rossum2e65f892007-02-28 22:03:49 +0000876 args, varargs, kwonlyargs, kwdefaults, varkw, defaults, ann,
877 formatvalue=self.formatvalue,
878 formatannotation=inspect.formatannotationrelativeto(object))
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000879 if realname == '<lambda>':
Tim Peters59ed4482001-10-31 04:20:26 +0000880 title = '<strong>%s</strong> <em>lambda</em> ' % name
Guido van Rossum2e65f892007-02-28 22:03:49 +0000881 # XXX lambda's won't usually have func_annotations['return']
882 # since the syntax doesn't support but it is possible.
883 # So removing parentheses isn't truly safe.
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000884 argspec = argspec[1:-1] # remove parentheses
Tim Peters4bcfa312001-09-20 06:08:24 +0000885 else:
886 argspec = '(...)'
Ka-Ping Yee66efbc72001-03-01 13:55:20 +0000887
Tim Peters2306d242001-09-25 03:18:32 +0000888 decl = title + argspec + (note and self.grey(
889 '<font face="helvetica, arial">%s</font>' % note))
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000890
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000891 if skipdocs:
Tim Peters2306d242001-09-25 03:18:32 +0000892 return '<dl><dt>%s</dt></dl>\n' % decl
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000893 else:
894 doc = self.markup(
895 getdoc(object), self.preformat, funcs, classes, methods)
Tim Peters2306d242001-09-25 03:18:32 +0000896 doc = doc and '<dd><tt>%s</tt></dd>' % doc
897 return '<dl><dt>%s</dt>%s</dl>\n' % (decl, doc)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000898
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +0000899 def _docdescriptor(self, name, value, mod):
Johannes Gijsbers8de645a2004-11-07 19:16:05 +0000900 results = []
901 push = results.append
902
903 if name:
904 push('<dl><dt><strong>%s</strong></dt>\n' % name)
905 if value.__doc__ is not None:
Ka-Ping Yeebba6acc2005-02-19 22:58:26 +0000906 doc = self.markup(getdoc(value), self.preformat)
Johannes Gijsbers8de645a2004-11-07 19:16:05 +0000907 push('<dd><tt>%s</tt></dd>\n' % doc)
Johannes Gijsbers8de645a2004-11-07 19:16:05 +0000908 push('</dl>\n')
909
910 return ''.join(results)
911
912 def docproperty(self, object, name=None, mod=None, cl=None):
913 """Produce html documentation for a property."""
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +0000914 return self._docdescriptor(name, object, mod)
Johannes Gijsbers8de645a2004-11-07 19:16:05 +0000915
Tim Peters8dd7ade2001-10-18 19:56:17 +0000916 def docother(self, object, name=None, mod=None, *ignored):
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000917 """Produce HTML documentation for a data object."""
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +0000918 lhs = name and '<strong>%s</strong> = ' % name or ''
919 return lhs + self.repr(object)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000920
Thomas Wouters0e3f5912006-08-11 14:57:12 +0000921 def docdata(self, object, name=None, mod=None, cl=None):
922 """Produce html documentation for a data descriptor."""
923 return self._docdescriptor(name, object, mod)
924
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000925 def index(self, dir, shadowed=None):
926 """Generate an HTML index for a directory of modules."""
927 modpkgs = []
928 if shadowed is None: shadowed = {}
Thomas Wouters49fd7fa2006-04-21 10:40:58 +0000929 for importer, name, ispkg in pkgutil.iter_modules([dir]):
930 modpkgs.append((name, '', ispkg, name in shadowed))
931 shadowed[name] = 1
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000932
933 modpkgs.sort()
934 contents = self.multicolumn(modpkgs, self.modpkglink)
935 return self.bigsection(dir, '#ffffff', '#ee77aa', contents)
936
937# -------------------------------------------- text documentation generator
938
939class TextRepr(Repr):
940 """Class for safely making a text representation of a Python object."""
941 def __init__(self):
942 Repr.__init__(self)
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000943 self.maxlist = self.maxtuple = 20
944 self.maxdict = 10
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000945 self.maxstring = self.maxother = 100
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000946
947 def repr1(self, x, level):
Skip Montanaro0fe8fce2003-06-27 15:45:41 +0000948 if hasattr(type(x), '__name__'):
Neal Norwitz9d72bb42007-04-17 08:48:32 +0000949 methodname = 'repr_' + '_'.join(type(x).__name__.split())
Skip Montanaro0fe8fce2003-06-27 15:45:41 +0000950 if hasattr(self, methodname):
951 return getattr(self, methodname)(x, level)
952 return cram(stripid(repr(x)), self.maxother)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000953
Ka-Ping Yeea2fe1032001-03-02 01:19:14 +0000954 def repr_string(self, x, level):
955 test = cram(x, self.maxstring)
956 testrepr = repr(test)
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +0000957 if '\\' in test and '\\' not in replace(testrepr, r'\\', ''):
Ka-Ping Yeea2fe1032001-03-02 01:19:14 +0000958 # Backslashes are only literal in the string and are never
959 # needed to make any special characters, so show a raw string.
960 return 'r' + testrepr[0] + test + testrepr[0]
961 return testrepr
962
Skip Montanarodf708782002-03-07 22:58:02 +0000963 repr_str = repr_string
964
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000965 def repr_instance(self, x, level):
966 try:
Ka-Ping Yee1d384632001-03-01 00:24:32 +0000967 return cram(stripid(repr(x)), self.maxstring)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000968 except:
969 return '<%s instance>' % x.__class__.__name__
970
971class TextDoc(Doc):
972 """Formatter class for text documentation."""
973
974 # ------------------------------------------- text formatting utilities
975
976 _repr_instance = TextRepr()
977 repr = _repr_instance.repr
978
979 def bold(self, text):
980 """Format a string in bold by overstriking."""
Neal Norwitz9d72bb42007-04-17 08:48:32 +0000981 return ''.join(map(lambda ch: ch + '\b' + ch, text))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000982
983 def indent(self, text, prefix=' '):
984 """Indent text by prepending a given prefix to each line."""
985 if not text: return ''
Collin Winter72e110c2007-07-17 00:27:30 +0000986 lines = [prefix + line for line in text.split('\n')]
Neal Norwitz9d72bb42007-04-17 08:48:32 +0000987 if lines: lines[-1] = lines[-1].rstrip()
988 return '\n'.join(lines)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000989
990 def section(self, title, contents):
991 """Format a section with a given heading."""
Neal Norwitz9d72bb42007-04-17 08:48:32 +0000992 clean_contents = self.indent(contents).rstrip()
993 return self.bold(title) + '\n' + clean_contents + '\n\n'
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000994
995 # ---------------------------------------------- type-specific routines
996
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000997 def formattree(self, tree, modname, parent=None, prefix=''):
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000998 """Render in text a class tree as returned by inspect.getclasstree()."""
999 result = ''
1000 for entry in tree:
1001 if type(entry) is type(()):
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001002 c, bases = entry
1003 result = result + prefix + classname(c, modname)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001004 if bases and bases != (parent,):
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001005 parents = map(lambda c, m=modname: classname(c, m), bases)
Neal Norwitz9d72bb42007-04-17 08:48:32 +00001006 result = result + '(%s)' % ', '.join(parents)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001007 result = result + '\n'
1008 elif type(entry) is type([]):
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001009 result = result + self.formattree(
1010 entry, modname, c, prefix + ' ')
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001011 return result
1012
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001013 def docmodule(self, object, name=None, mod=None):
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001014 """Produce text documentation for a given module object."""
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001015 name = object.__name__ # ignore the passed-in name
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +00001016 synop, desc = splitdoc(getdoc(object))
1017 result = self.section('NAME', name + (synop and ' - ' + synop))
Alexander Belopolskya47bbf52010-11-18 01:52:54 +00001018 all = getattr(object, '__all__', None)
Skip Montanaro4997a692003-09-10 16:47:51 +00001019 docloc = self.getdocloc(object)
1020 if docloc is not None:
Alexander Belopolskya47bbf52010-11-18 01:52:54 +00001021 result = result + self.section('MODULE REFERENCE', docloc + """
1022
1023The following documentation is automatically generated from the Python source
1024files. It may be incomplete, incorrect or include features that are considered
1025implementation detail and may vary between Python implementations. When in
1026doubt, consult the module reference at the location listed above.
1027""")
Skip Montanaro4997a692003-09-10 16:47:51 +00001028
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +00001029 if desc:
1030 result = result + self.section('DESCRIPTION', desc)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001031
1032 classes = []
1033 for key, value in inspect.getmembers(object, inspect.isclass):
Johannes Gijsbers4c11f602004-08-30 14:13:04 +00001034 # if __all__ exists, believe it. Otherwise use old heuristic.
1035 if (all is not None
1036 or (inspect.getmodule(value) or object) is object):
Skip Montanaroa5616d22004-06-11 04:46:12 +00001037 if visiblename(key, all):
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +00001038 classes.append((key, value))
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001039 funcs = []
1040 for key, value in inspect.getmembers(object, inspect.isroutine):
Johannes Gijsbers4c11f602004-08-30 14:13:04 +00001041 # if __all__ exists, believe it. Otherwise use old heuristic.
1042 if (all is not None or
1043 inspect.isbuiltin(value) or inspect.getmodule(value) is object):
Skip Montanaroa5616d22004-06-11 04:46:12 +00001044 if visiblename(key, all):
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +00001045 funcs.append((key, value))
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001046 data = []
1047 for key, value in inspect.getmembers(object, isdata):
Skip Montanaroa5616d22004-06-11 04:46:12 +00001048 if visiblename(key, all):
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001049 data.append((key, value))
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001050
Christian Heimes1af737c2008-01-23 08:24:23 +00001051 modpkgs = []
1052 modpkgs_names = set()
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001053 if hasattr(object, '__path__'):
Thomas Wouters49fd7fa2006-04-21 10:40:58 +00001054 for importer, modname, ispkg in pkgutil.iter_modules(object.__path__):
Christian Heimes1af737c2008-01-23 08:24:23 +00001055 modpkgs_names.add(modname)
Thomas Wouters49fd7fa2006-04-21 10:40:58 +00001056 if ispkg:
1057 modpkgs.append(modname + ' (package)')
1058 else:
1059 modpkgs.append(modname)
1060
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001061 modpkgs.sort()
1062 result = result + self.section(
Neal Norwitz9d72bb42007-04-17 08:48:32 +00001063 'PACKAGE CONTENTS', '\n'.join(modpkgs))
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001064
Christian Heimes1af737c2008-01-23 08:24:23 +00001065 # Detect submodules as sometimes created by C extensions
1066 submodules = []
1067 for key, value in inspect.getmembers(object, inspect.ismodule):
1068 if value.__name__.startswith(name + '.') and key not in modpkgs_names:
1069 submodules.append(key)
1070 if submodules:
1071 submodules.sort()
1072 result = result + self.section(
Amaury Forgeot d'Arc768db922008-04-24 21:00:04 +00001073 'SUBMODULES', '\n'.join(submodules))
Christian Heimes1af737c2008-01-23 08:24:23 +00001074
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001075 if classes:
Guido van Rossum1bc535d2007-05-15 18:46:22 +00001076 classlist = [value for key, value in classes]
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001077 contents = [self.formattree(
1078 inspect.getclasstree(classlist, 1), name)]
1079 for key, value in classes:
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001080 contents.append(self.document(value, key, name))
Neal Norwitz9d72bb42007-04-17 08:48:32 +00001081 result = result + self.section('CLASSES', '\n'.join(contents))
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001082
1083 if funcs:
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001084 contents = []
1085 for key, value in funcs:
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001086 contents.append(self.document(value, key, name))
Neal Norwitz9d72bb42007-04-17 08:48:32 +00001087 result = result + self.section('FUNCTIONS', '\n'.join(contents))
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001088
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001089 if data:
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001090 contents = []
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001091 for key, value in data:
Georg Brandl8b813db2005-10-01 16:32:31 +00001092 contents.append(self.docother(value, key, name, maxlen=70))
Neal Norwitz9d72bb42007-04-17 08:48:32 +00001093 result = result + self.section('DATA', '\n'.join(contents))
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001094
1095 if hasattr(object, '__version__'):
1096 version = str(object.__version__)
Ka-Ping Yee1d384632001-03-01 00:24:32 +00001097 if version[:11] == '$' + 'Revision: ' and version[-1:] == '$':
Neal Norwitz9d72bb42007-04-17 08:48:32 +00001098 version = version[11:-1].strip()
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001099 result = result + self.section('VERSION', version)
Ka-Ping Yee6f3f9a42001-02-27 22:42:36 +00001100 if hasattr(object, '__date__'):
1101 result = result + self.section('DATE', str(object.__date__))
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001102 if hasattr(object, '__author__'):
Ka-Ping Yee6f3f9a42001-02-27 22:42:36 +00001103 result = result + self.section('AUTHOR', str(object.__author__))
1104 if hasattr(object, '__credits__'):
1105 result = result + self.section('CREDITS', str(object.__credits__))
Alexander Belopolskya47bbf52010-11-18 01:52:54 +00001106 try:
1107 file = inspect.getabsfile(object)
1108 except TypeError:
1109 file = '(built-in)'
1110 result = result + self.section('FILE', file)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001111 return result
1112
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001113 def docclass(self, object, name=None, mod=None):
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001114 """Produce text documentation for a given class object."""
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001115 realname = object.__name__
1116 name = name or realname
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001117 bases = object.__bases__
1118
Tim Petersc86f6ca2001-09-26 21:31:51 +00001119 def makename(c, m=object.__module__):
1120 return classname(c, m)
1121
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001122 if name == realname:
1123 title = 'class ' + self.bold(realname)
1124 else:
1125 title = self.bold(name) + ' = class ' + realname
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001126 if bases:
Ka-Ping Yee3bda8792001-03-23 13:17:50 +00001127 parents = map(makename, bases)
Neal Norwitz9d72bb42007-04-17 08:48:32 +00001128 title = title + '(%s)' % ', '.join(parents)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001129
1130 doc = getdoc(object)
Tim Peters28355492001-09-23 21:29:55 +00001131 contents = doc and [doc + '\n'] or []
1132 push = contents.append
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001133
Tim Petersc86f6ca2001-09-26 21:31:51 +00001134 # List the mro, if non-trivial.
Raymond Hettinger756b3f32004-01-29 06:37:52 +00001135 mro = deque(inspect.getmro(object))
Tim Petersc86f6ca2001-09-26 21:31:51 +00001136 if len(mro) > 2:
1137 push("Method resolution order:")
1138 for base in mro:
1139 push(' ' + makename(base))
1140 push('')
1141
Tim Petersf4aad8e2001-09-24 22:40:47 +00001142 # Cute little class to pump out a horizontal rule between sections.
1143 class HorizontalRule:
1144 def __init__(self):
1145 self.needone = 0
1146 def maybe(self):
1147 if self.needone:
1148 push('-' * 70)
1149 self.needone = 1
1150 hr = HorizontalRule()
1151
Tim Peters28355492001-09-23 21:29:55 +00001152 def spill(msg, attrs, predicate):
Tim Petersfa26f7c2001-09-24 08:05:11 +00001153 ok, attrs = _split_list(attrs, predicate)
Tim Peters28355492001-09-23 21:29:55 +00001154 if ok:
Tim Petersf4aad8e2001-09-24 22:40:47 +00001155 hr.maybe()
Tim Peters28355492001-09-23 21:29:55 +00001156 push(msg)
1157 for name, kind, homecls, value in ok:
1158 push(self.document(getattr(object, name),
1159 name, mod, object))
1160 return attrs
1161
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +00001162 def spilldescriptors(msg, attrs, predicate):
Tim Petersfa26f7c2001-09-24 08:05:11 +00001163 ok, attrs = _split_list(attrs, predicate)
Tim Peters28355492001-09-23 21:29:55 +00001164 if ok:
Tim Petersf4aad8e2001-09-24 22:40:47 +00001165 hr.maybe()
Tim Peters28355492001-09-23 21:29:55 +00001166 push(msg)
1167 for name, kind, homecls, value in ok:
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +00001168 push(self._docdescriptor(name, value, mod))
Tim Peters28355492001-09-23 21:29:55 +00001169 return attrs
Tim Petersb47879b2001-09-24 04:47:19 +00001170
Tim Petersfa26f7c2001-09-24 08:05:11 +00001171 def spilldata(msg, attrs, predicate):
1172 ok, attrs = _split_list(attrs, predicate)
Tim Peters28355492001-09-23 21:29:55 +00001173 if ok:
Tim Petersf4aad8e2001-09-24 22:40:47 +00001174 hr.maybe()
Tim Peters28355492001-09-23 21:29:55 +00001175 push(msg)
1176 for name, kind, homecls, value in ok:
Guido van Rossumd59da4b2007-05-22 18:11:13 +00001177 if hasattr(value, '__call__') or inspect.isdatadescriptor(value):
Ka-Ping Yeebba6acc2005-02-19 22:58:26 +00001178 doc = getdoc(value)
Guido van Rossum5e355b22002-05-21 20:56:15 +00001179 else:
1180 doc = None
Tim Peters28355492001-09-23 21:29:55 +00001181 push(self.docother(getattr(object, name),
Georg Brandl8b813db2005-10-01 16:32:31 +00001182 name, mod, maxlen=70, doc=doc) + '\n')
Tim Peters28355492001-09-23 21:29:55 +00001183 return attrs
1184
Guido van Rossum1bc535d2007-05-15 18:46:22 +00001185 attrs = [(name, kind, cls, value)
1186 for name, kind, cls, value in classify_class_attrs(object)
1187 if visiblename(name)]
1188
Tim Petersfa26f7c2001-09-24 08:05:11 +00001189 while attrs:
Tim Peters351e3622001-09-27 03:29:51 +00001190 if mro:
Raymond Hettinger756b3f32004-01-29 06:37:52 +00001191 thisclass = mro.popleft()
Tim Peters351e3622001-09-27 03:29:51 +00001192 else:
1193 thisclass = attrs[0][2]
Tim Petersfa26f7c2001-09-24 08:05:11 +00001194 attrs, inherited = _split_list(attrs, lambda t: t[2] is thisclass)
1195
Georg Brandl1a3284e2007-12-02 09:40:06 +00001196 if thisclass is builtins.object:
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +00001197 attrs = inherited
1198 continue
1199 elif thisclass is object:
Tim Peters28355492001-09-23 21:29:55 +00001200 tag = "defined here"
1201 else:
Tim Petersfa26f7c2001-09-24 08:05:11 +00001202 tag = "inherited from %s" % classname(thisclass,
1203 object.__module__)
Tim Peters28355492001-09-23 21:29:55 +00001204
1205 # Sort attrs by name.
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +00001206 attrs.sort()
Tim Peters28355492001-09-23 21:29:55 +00001207
1208 # Pump out the attrs, segregated by kind.
Tim Petersf4aad8e2001-09-24 22:40:47 +00001209 attrs = spill("Methods %s:\n" % tag, attrs,
Tim Peters28355492001-09-23 21:29:55 +00001210 lambda t: t[1] == 'method')
Tim Petersf4aad8e2001-09-24 22:40:47 +00001211 attrs = spill("Class methods %s:\n" % tag, attrs,
Tim Peters28355492001-09-23 21:29:55 +00001212 lambda t: t[1] == 'class method')
Tim Petersf4aad8e2001-09-24 22:40:47 +00001213 attrs = spill("Static methods %s:\n" % tag, attrs,
Tim Peters28355492001-09-23 21:29:55 +00001214 lambda t: t[1] == 'static method')
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +00001215 attrs = spilldescriptors("Data descriptors %s:\n" % tag, attrs,
1216 lambda t: t[1] == 'data descriptor')
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +00001217 attrs = spilldata("Data and other attributes %s:\n" % tag, attrs,
1218 lambda t: t[1] == 'data')
Tim Peters28355492001-09-23 21:29:55 +00001219 assert attrs == []
Tim Peters351e3622001-09-27 03:29:51 +00001220 attrs = inherited
Tim Peters28355492001-09-23 21:29:55 +00001221
1222 contents = '\n'.join(contents)
1223 if not contents:
1224 return title + '\n'
Neal Norwitz9d72bb42007-04-17 08:48:32 +00001225 return title + '\n' + self.indent(contents.rstrip(), ' | ') + '\n'
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001226
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001227 def formatvalue(self, object):
1228 """Format an argument default value as text."""
1229 return '=' + self.repr(object)
1230
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001231 def docroutine(self, object, name=None, mod=None, cl=None):
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001232 """Produce text documentation for a function or method object."""
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001233 realname = object.__name__
1234 name = name or realname
1235 note = ''
Ka-Ping Yee3bda8792001-03-23 13:17:50 +00001236 skipdocs = 0
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001237 if inspect.ismethod(object):
Christian Heimesff737952007-11-27 10:40:20 +00001238 imclass = object.__self__.__class__
Ka-Ping Yee3bda8792001-03-23 13:17:50 +00001239 if cl:
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001240 if imclass is not cl:
1241 note = ' from ' + classname(imclass, mod)
Ka-Ping Yee3bda8792001-03-23 13:17:50 +00001242 else:
Christian Heimesff737952007-11-27 10:40:20 +00001243 if object.__self__ is not None:
Ka-Ping Yeeb7a48302001-04-12 20:39:14 +00001244 note = ' method of %s instance' % classname(
Christian Heimesff737952007-11-27 10:40:20 +00001245 object.__self__.__class__, mod)
Ka-Ping Yeeb7a48302001-04-12 20:39:14 +00001246 else:
1247 note = ' unbound %s method' % classname(imclass,mod)
Christian Heimesff737952007-11-27 10:40:20 +00001248 object = object.__func__
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001249
1250 if name == realname:
1251 title = self.bold(realname)
1252 else:
Raymond Hettinger54f02222002-06-01 14:18:47 +00001253 if (cl and realname in cl.__dict__ and
Ka-Ping Yee3bda8792001-03-23 13:17:50 +00001254 cl.__dict__[realname] is object):
1255 skipdocs = 1
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001256 title = self.bold(name) + ' = ' + realname
Tim Peters4bcfa312001-09-20 06:08:24 +00001257 if inspect.isfunction(object):
Guido van Rossum2e65f892007-02-28 22:03:49 +00001258 args, varargs, varkw, defaults, kwonlyargs, kwdefaults, ann = \
1259 inspect.getfullargspec(object)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001260 argspec = inspect.formatargspec(
Guido van Rossum2e65f892007-02-28 22:03:49 +00001261 args, varargs, varkw, defaults, kwonlyargs, kwdefaults, ann,
1262 formatvalue=self.formatvalue,
1263 formatannotation=inspect.formatannotationrelativeto(object))
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001264 if realname == '<lambda>':
Georg Brandl501dd0d2006-02-17 09:45:40 +00001265 title = self.bold(name) + ' lambda '
Guido van Rossum2e65f892007-02-28 22:03:49 +00001266 # XXX lambda's won't usually have func_annotations['return']
1267 # since the syntax doesn't support but it is possible.
1268 # So removing parentheses isn't truly safe.
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001269 argspec = argspec[1:-1] # remove parentheses
Tim Peters4bcfa312001-09-20 06:08:24 +00001270 else:
1271 argspec = '(...)'
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001272 decl = title + argspec + note
1273
Ka-Ping Yee3bda8792001-03-23 13:17:50 +00001274 if skipdocs:
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001275 return decl + '\n'
Ka-Ping Yee3bda8792001-03-23 13:17:50 +00001276 else:
1277 doc = getdoc(object) or ''
Neal Norwitz9d72bb42007-04-17 08:48:32 +00001278 return decl + '\n' + (doc and self.indent(doc).rstrip() + '\n')
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001279
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +00001280 def _docdescriptor(self, name, value, mod):
Johannes Gijsbers8de645a2004-11-07 19:16:05 +00001281 results = []
1282 push = results.append
1283
1284 if name:
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +00001285 push(self.bold(name))
1286 push('\n')
Johannes Gijsbers8de645a2004-11-07 19:16:05 +00001287 doc = getdoc(value) or ''
1288 if doc:
1289 push(self.indent(doc))
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +00001290 push('\n')
1291 return ''.join(results)
Johannes Gijsbers8de645a2004-11-07 19:16:05 +00001292
1293 def docproperty(self, object, name=None, mod=None, cl=None):
1294 """Produce text documentation for a property."""
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +00001295 return self._docdescriptor(name, object, mod)
Johannes Gijsbers8de645a2004-11-07 19:16:05 +00001296
Thomas Wouters0e3f5912006-08-11 14:57:12 +00001297 def docdata(self, object, name=None, mod=None, cl=None):
1298 """Produce text documentation for a data descriptor."""
1299 return self._docdescriptor(name, object, mod)
1300
Georg Brandl8b813db2005-10-01 16:32:31 +00001301 def docother(self, object, name=None, mod=None, parent=None, maxlen=None, doc=None):
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001302 """Produce text documentation for a data object."""
1303 repr = self.repr(object)
1304 if maxlen:
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001305 line = (name and name + ' = ' or '') + repr
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001306 chop = maxlen - len(line)
1307 if chop < 0: repr = repr[:chop] + '...'
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001308 line = (name and self.bold(name) + ' = ' or '') + repr
Tim Peters28355492001-09-23 21:29:55 +00001309 if doc is not None:
1310 line += '\n' + self.indent(str(doc))
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001311 return line
1312
Georg Brandld80d5f42010-12-03 07:47:22 +00001313class _PlainTextDoc(TextDoc):
1314 """Subclass of TextDoc which overrides string styling"""
1315 def bold(self, text):
1316 return text
1317
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001318# --------------------------------------------------------- user interfaces
1319
1320def pager(text):
1321 """The first time this is called, determine what kind of pager to use."""
1322 global pager
1323 pager = getpager()
1324 pager(text)
1325
1326def getpager():
1327 """Decide what method to use for paging through text."""
Guido van Rossuma01a8b62007-05-27 09:20:14 +00001328 if not hasattr(sys.stdout, "isatty"):
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001329 return plainpager
1330 if not sys.stdin.isatty() or not sys.stdout.isatty():
1331 return plainpager
Raymond Hettinger54f02222002-06-01 14:18:47 +00001332 if 'PAGER' in os.environ:
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +00001333 if sys.platform == 'win32': # pipes completely broken in Windows
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001334 return lambda text: tempfilepager(plain(text), os.environ['PAGER'])
Raymond Hettingerdbecd932005-02-06 06:57:08 +00001335 elif os.environ.get('TERM') in ('dumb', 'emacs'):
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001336 return lambda text: pipepager(plain(text), os.environ['PAGER'])
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +00001337 else:
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001338 return lambda text: pipepager(text, os.environ['PAGER'])
Ka-Ping Yeea487e4e2005-11-05 04:49:18 +00001339 if os.environ.get('TERM') in ('dumb', 'emacs'):
1340 return plainpager
Andrew MacIntyre54e0eab2002-03-03 03:12:30 +00001341 if sys.platform == 'win32' or sys.platform.startswith('os2'):
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001342 return lambda text: tempfilepager(plain(text), 'more <')
Skip Montanarod404bee2002-09-26 21:44:57 +00001343 if hasattr(os, 'system') and os.system('(less) 2>/dev/null') == 0:
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001344 return lambda text: pipepager(text, 'less')
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001345
1346 import tempfile
Guido van Rossum3b0a3292002-08-09 16:38:32 +00001347 (fd, filename) = tempfile.mkstemp()
1348 os.close(fd)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001349 try:
Georg Brandl3dbca812008-07-23 16:10:53 +00001350 if hasattr(os, 'system') and os.system('more "%s"' % filename) == 0:
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001351 return lambda text: pipepager(text, 'more')
1352 else:
1353 return ttypager
1354 finally:
1355 os.unlink(filename)
1356
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001357def plain(text):
1358 """Remove boldface formatting from text."""
1359 return re.sub('.\b', '', text)
1360
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001361def pipepager(text, cmd):
1362 """Page through text by feeding it to another program."""
1363 pipe = os.popen(cmd, 'w')
1364 try:
1365 pipe.write(text)
1366 pipe.close()
1367 except IOError:
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001368 pass # Ignore broken pipes caused by quitting the pager program.
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001369
1370def tempfilepager(text, cmd):
1371 """Page through text by invoking a program on a temporary file."""
1372 import tempfile
Tim Peters550e4e52003-02-07 01:53:46 +00001373 filename = tempfile.mktemp()
1374 file = open(filename, 'w')
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001375 file.write(text)
1376 file.close()
1377 try:
Georg Brandl3dbca812008-07-23 16:10:53 +00001378 os.system(cmd + ' "' + filename + '"')
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001379 finally:
1380 os.unlink(filename)
1381
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001382def ttypager(text):
1383 """Page through text on a text terminal."""
Neal Norwitz9d72bb42007-04-17 08:48:32 +00001384 lines = plain(text).split('\n')
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001385 try:
1386 import tty
1387 fd = sys.stdin.fileno()
1388 old = tty.tcgetattr(fd)
1389 tty.setcbreak(fd)
1390 getchar = lambda: sys.stdin.read(1)
Ka-Ping Yee457aab22001-02-27 23:36:29 +00001391 except (ImportError, AttributeError):
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001392 tty = None
1393 getchar = lambda: sys.stdin.readline()[:-1][:1]
1394
1395 try:
1396 r = inc = os.environ.get('LINES', 25) - 1
Neal Norwitz9d72bb42007-04-17 08:48:32 +00001397 sys.stdout.write('\n'.join(lines[:inc]) + '\n')
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001398 while lines[r:]:
1399 sys.stdout.write('-- more --')
1400 sys.stdout.flush()
1401 c = getchar()
1402
Raymond Hettingerdbecd932005-02-06 06:57:08 +00001403 if c in ('q', 'Q'):
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001404 sys.stdout.write('\r \r')
1405 break
Raymond Hettingerdbecd932005-02-06 06:57:08 +00001406 elif c in ('\r', '\n'):
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001407 sys.stdout.write('\r \r' + lines[r] + '\n')
1408 r = r + 1
1409 continue
Raymond Hettingerdbecd932005-02-06 06:57:08 +00001410 if c in ('b', 'B', '\x1b'):
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001411 r = r - inc - inc
1412 if r < 0: r = 0
Neal Norwitz9d72bb42007-04-17 08:48:32 +00001413 sys.stdout.write('\n' + '\n'.join(lines[r:r+inc]) + '\n')
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001414 r = r + inc
1415
1416 finally:
1417 if tty:
1418 tty.tcsetattr(fd, tty.TCSAFLUSH, old)
1419
1420def plainpager(text):
1421 """Simply print unformatted text. This is the ultimate fallback."""
1422 sys.stdout.write(plain(text))
1423
1424def describe(thing):
Ka-Ping Yee3bda8792001-03-23 13:17:50 +00001425 """Produce a short description of the given thing."""
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001426 if inspect.ismodule(thing):
1427 if thing.__name__ in sys.builtin_module_names:
1428 return 'built-in module ' + thing.__name__
1429 if hasattr(thing, '__path__'):
1430 return 'package ' + thing.__name__
1431 else:
1432 return 'module ' + thing.__name__
1433 if inspect.isbuiltin(thing):
1434 return 'built-in function ' + thing.__name__
Thomas Wouters0e3f5912006-08-11 14:57:12 +00001435 if inspect.isgetsetdescriptor(thing):
1436 return 'getset descriptor %s.%s.%s' % (
1437 thing.__objclass__.__module__, thing.__objclass__.__name__,
1438 thing.__name__)
1439 if inspect.ismemberdescriptor(thing):
1440 return 'member descriptor %s.%s.%s' % (
1441 thing.__objclass__.__module__, thing.__objclass__.__name__,
1442 thing.__name__)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001443 if inspect.isclass(thing):
1444 return 'class ' + thing.__name__
1445 if inspect.isfunction(thing):
1446 return 'function ' + thing.__name__
1447 if inspect.ismethod(thing):
1448 return 'method ' + thing.__name__
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001449 return type(thing).__name__
1450
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001451def locate(path, forceload=0):
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001452 """Locate an object by name or dotted path, importing as necessary."""
Neal Norwitz9d72bb42007-04-17 08:48:32 +00001453 parts = [part for part in path.split('.') if part]
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001454 module, n = None, 0
1455 while n < len(parts):
Neal Norwitz9d72bb42007-04-17 08:48:32 +00001456 nextmodule = safeimport('.'.join(parts[:n+1]), forceload)
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001457 if nextmodule: module, n = nextmodule, n + 1
1458 else: break
1459 if module:
1460 object = module
1461 for part in parts[n:]:
1462 try: object = getattr(object, part)
1463 except AttributeError: return None
1464 return object
1465 else:
Georg Brandl1a3284e2007-12-02 09:40:06 +00001466 if hasattr(builtins, path):
1467 return getattr(builtins, path)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001468
1469# --------------------------------------- interactive interpreter interface
1470
1471text = TextDoc()
Georg Brandld80d5f42010-12-03 07:47:22 +00001472plaintext = _PlainTextDoc()
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001473html = HTMLDoc()
1474
Ka-Ping Yee45daeb02002-08-11 15:11:33 +00001475def resolve(thing, forceload=0):
1476 """Given an object or a path to an object, get the object and its name."""
1477 if isinstance(thing, str):
1478 object = locate(thing, forceload)
1479 if not object:
Collin Winterce36ad82007-08-30 01:19:48 +00001480 raise ImportError('no Python documentation found for %r' % thing)
Ka-Ping Yee45daeb02002-08-11 15:11:33 +00001481 return object, thing
1482 else:
1483 return thing, getattr(thing, '__name__', None)
1484
Georg Brandld80d5f42010-12-03 07:47:22 +00001485def render_doc(thing, title='Python Library Documentation: %s', forceload=0,
1486 renderer=None):
Guido van Rossumd8faa362007-04-27 19:54:29 +00001487 """Render text documentation, given an object or a path to an object."""
Georg Brandld80d5f42010-12-03 07:47:22 +00001488 if renderer is None:
1489 renderer = text
Guido van Rossumd8faa362007-04-27 19:54:29 +00001490 object, name = resolve(thing, forceload)
1491 desc = describe(object)
1492 module = inspect.getmodule(object)
1493 if name and '.' in name:
1494 desc += ' in ' + name[:name.rfind('.')]
1495 elif module and module is not object:
1496 desc += ' in module ' + module.__name__
Amaury Forgeot d'Arc768db922008-04-24 21:00:04 +00001497
1498 if not (inspect.ismodule(object) or
Guido van Rossumd8faa362007-04-27 19:54:29 +00001499 inspect.isclass(object) or
1500 inspect.isroutine(object) or
1501 inspect.isgetsetdescriptor(object) or
1502 inspect.ismemberdescriptor(object) or
1503 isinstance(object, property)):
1504 # If the passed object is a piece of data or an instance,
1505 # document its available methods instead of its value.
1506 object = type(object)
1507 desc += ' object'
Georg Brandld80d5f42010-12-03 07:47:22 +00001508 return title % desc + '\n\n' + renderer.document(object, name)
Guido van Rossumd8faa362007-04-27 19:54:29 +00001509
Georg Brandld80d5f42010-12-03 07:47:22 +00001510def doc(thing, title='Python Library Documentation: %s', forceload=0,
1511 output=None):
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001512 """Display text documentation, given an object or a path to an object."""
Ka-Ping Yee45daeb02002-08-11 15:11:33 +00001513 try:
Georg Brandld80d5f42010-12-03 07:47:22 +00001514 if output is None:
1515 pager(render_doc(thing, title, forceload))
1516 else:
1517 output.write(render_doc(thing, title, forceload, plaintext))
Guido van Rossumb940e112007-01-10 16:19:56 +00001518 except (ImportError, ErrorDuringImport) as value:
Guido van Rossumbe19ed72007-02-09 05:37:30 +00001519 print(value)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001520
Ka-Ping Yee45daeb02002-08-11 15:11:33 +00001521def writedoc(thing, forceload=0):
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001522 """Write HTML documentation to a file in the current directory."""
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001523 try:
Ka-Ping Yee45daeb02002-08-11 15:11:33 +00001524 object, name = resolve(thing, forceload)
1525 page = html.page(describe(object), html.document(object, name))
Georg Brandl388faac2009-04-10 08:31:48 +00001526 file = open(name + '.html', 'w', encoding='utf-8')
Ka-Ping Yee45daeb02002-08-11 15:11:33 +00001527 file.write(page)
1528 file.close()
Guido van Rossumbe19ed72007-02-09 05:37:30 +00001529 print('wrote', name + '.html')
Guido van Rossumb940e112007-01-10 16:19:56 +00001530 except (ImportError, ErrorDuringImport) as value:
Guido van Rossumbe19ed72007-02-09 05:37:30 +00001531 print(value)
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001532
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001533def writedocs(dir, pkgpath='', done=None):
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001534 """Write out HTML documentation for all modules in a directory tree."""
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001535 if done is None: done = {}
Thomas Wouters49fd7fa2006-04-21 10:40:58 +00001536 for importer, modname, ispkg in pkgutil.walk_packages([dir], pkgpath):
1537 writedoc(modname)
1538 return
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001539
1540class Helper:
Georg Brandl6b38daa2008-06-01 21:05:17 +00001541
1542 # These dictionaries map a topic name to either an alias, or a tuple
1543 # (label, seealso-items). The "label" is the label of the corresponding
1544 # section in the .rst file under Doc/ and an index into the dictionary
Georg Brandl5617db82009-04-27 16:28:57 +00001545 # in pydoc_data/topics.py.
Georg Brandl6b38daa2008-06-01 21:05:17 +00001546 #
1547 # CAUTION: if you change one of these dictionaries, be sure to adapt the
1548 # list of needed labels in Doc/tools/sphinxext/pyspecific.py and
Georg Brandl5617db82009-04-27 16:28:57 +00001549 # regenerate the pydoc_data/topics.py file by running
Georg Brandl6b38daa2008-06-01 21:05:17 +00001550 # make pydoc-topics
1551 # in Doc/ and copying the output file into the Lib/ directory.
1552
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001553 keywords = {
1554 'and': 'BOOLEAN',
Guido van Rossumd8faa362007-04-27 19:54:29 +00001555 'as': 'with',
Georg Brandl6b38daa2008-06-01 21:05:17 +00001556 'assert': ('assert', ''),
1557 'break': ('break', 'while for'),
1558 'class': ('class', 'CLASSES SPECIALMETHODS'),
1559 'continue': ('continue', 'while for'),
1560 'def': ('function', ''),
1561 'del': ('del', 'BASICMETHODS'),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001562 'elif': 'if',
Georg Brandl6b38daa2008-06-01 21:05:17 +00001563 'else': ('else', 'while for'),
Georg Brandl0d855392008-08-30 19:53:05 +00001564 'except': 'try',
1565 'finally': 'try',
Georg Brandl6b38daa2008-06-01 21:05:17 +00001566 'for': ('for', 'break continue while'),
Georg Brandl0d855392008-08-30 19:53:05 +00001567 'from': 'import',
Georg Brandl74abf6f2010-11-20 19:54:36 +00001568 'global': ('global', 'nonlocal NAMESPACES'),
Georg Brandl6b38daa2008-06-01 21:05:17 +00001569 'if': ('if', 'TRUTHVALUE'),
1570 'import': ('import', 'MODULES'),
Georg Brandl395ed242009-09-04 08:07:32 +00001571 'in': ('in', 'SEQUENCEMETHODS'),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001572 'is': 'COMPARISON',
Georg Brandl6b38daa2008-06-01 21:05:17 +00001573 'lambda': ('lambda', 'FUNCTIONS'),
Georg Brandl74abf6f2010-11-20 19:54:36 +00001574 'nonlocal': ('nonlocal', 'global NAMESPACES'),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001575 'not': 'BOOLEAN',
1576 'or': 'BOOLEAN',
Georg Brandl6b38daa2008-06-01 21:05:17 +00001577 'pass': ('pass', ''),
1578 'raise': ('raise', 'EXCEPTIONS'),
1579 'return': ('return', 'FUNCTIONS'),
1580 'try': ('try', 'EXCEPTIONS'),
1581 'while': ('while', 'break continue if TRUTHVALUE'),
1582 'with': ('with', 'CONTEXTMANAGERS EXCEPTIONS yield'),
1583 'yield': ('yield', ''),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001584 }
Georg Brandldb7b6b92009-01-01 15:53:14 +00001585 # Either add symbols to this dictionary or to the symbols dictionary
1586 # directly: Whichever is easier. They are merged later.
1587 _symbols_inverse = {
1588 'STRINGS' : ("'", "'''", "r'", "b'", '"""', '"', 'r"', 'b"'),
1589 'OPERATORS' : ('+', '-', '*', '**', '/', '//', '%', '<<', '>>', '&',
1590 '|', '^', '~', '<', '>', '<=', '>=', '==', '!=', '<>'),
1591 'COMPARISON' : ('<', '>', '<=', '>=', '==', '!=', '<>'),
1592 'UNARY' : ('-', '~'),
1593 'AUGMENTEDASSIGNMENT' : ('+=', '-=', '*=', '/=', '%=', '&=', '|=',
1594 '^=', '<<=', '>>=', '**=', '//='),
1595 'BITWISE' : ('<<', '>>', '&', '|', '^', '~'),
1596 'COMPLEX' : ('j', 'J')
1597 }
1598 symbols = {
1599 '%': 'OPERATORS FORMATTING',
1600 '**': 'POWER',
1601 ',': 'TUPLES LISTS FUNCTIONS',
1602 '.': 'ATTRIBUTES FLOAT MODULES OBJECTS',
1603 '...': 'ELLIPSIS',
1604 ':': 'SLICINGS DICTIONARYLITERALS',
1605 '@': 'def class',
1606 '\\': 'STRINGS',
1607 '_': 'PRIVATENAMES',
1608 '__': 'PRIVATENAMES SPECIALMETHODS',
1609 '`': 'BACKQUOTES',
1610 '(': 'TUPLES FUNCTIONS CALLS',
1611 ')': 'TUPLES FUNCTIONS CALLS',
1612 '[': 'LISTS SUBSCRIPTS SLICINGS',
1613 ']': 'LISTS SUBSCRIPTS SLICINGS'
1614 }
1615 for topic, symbols_ in _symbols_inverse.items():
1616 for symbol in symbols_:
1617 topics = symbols.get(symbol, topic)
1618 if topic not in topics:
1619 topics = topics + ' ' + topic
1620 symbols[symbol] = topics
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001621
1622 topics = {
Georg Brandl6b38daa2008-06-01 21:05:17 +00001623 'TYPES': ('types', 'STRINGS UNICODE NUMBERS SEQUENCES MAPPINGS '
1624 'FUNCTIONS CLASSES MODULES FILES inspect'),
1625 'STRINGS': ('strings', 'str UNICODE SEQUENCES STRINGMETHODS '
1626 'FORMATTING TYPES'),
1627 'STRINGMETHODS': ('string-methods', 'STRINGS FORMATTING'),
1628 'FORMATTING': ('formatstrings', 'OPERATORS'),
1629 'UNICODE': ('strings', 'encodings unicode SEQUENCES STRINGMETHODS '
1630 'FORMATTING TYPES'),
1631 'NUMBERS': ('numbers', 'INTEGER FLOAT COMPLEX TYPES'),
1632 'INTEGER': ('integers', 'int range'),
1633 'FLOAT': ('floating', 'float math'),
1634 'COMPLEX': ('imaginary', 'complex cmath'),
1635 'SEQUENCES': ('typesseq', 'STRINGMETHODS FORMATTING range LISTS'),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001636 'MAPPINGS': 'DICTIONARIES',
Georg Brandl6b38daa2008-06-01 21:05:17 +00001637 'FUNCTIONS': ('typesfunctions', 'def TYPES'),
1638 'METHODS': ('typesmethods', 'class def CLASSES TYPES'),
1639 'CODEOBJECTS': ('bltin-code-objects', 'compile FUNCTIONS TYPES'),
1640 'TYPEOBJECTS': ('bltin-type-objects', 'types TYPES'),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001641 'FRAMEOBJECTS': 'TYPES',
1642 'TRACEBACKS': 'TYPES',
Georg Brandl6b38daa2008-06-01 21:05:17 +00001643 'NONE': ('bltin-null-object', ''),
1644 'ELLIPSIS': ('bltin-ellipsis-object', 'SLICINGS'),
1645 'FILES': ('bltin-file-objects', ''),
1646 'SPECIALATTRIBUTES': ('specialattrs', ''),
1647 'CLASSES': ('types', 'class SPECIALMETHODS PRIVATENAMES'),
1648 'MODULES': ('typesmodules', 'import'),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001649 'PACKAGES': 'import',
Georg Brandl6b38daa2008-06-01 21:05:17 +00001650 'EXPRESSIONS': ('operator-summary', 'lambda or and not in is BOOLEAN '
1651 'COMPARISON BITWISE SHIFTING BINARY FORMATTING POWER '
1652 'UNARY ATTRIBUTES SUBSCRIPTS SLICINGS CALLS TUPLES '
1653 'LISTS DICTIONARIES'),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001654 'OPERATORS': 'EXPRESSIONS',
1655 'PRECEDENCE': 'EXPRESSIONS',
Georg Brandl6b38daa2008-06-01 21:05:17 +00001656 'OBJECTS': ('objects', 'TYPES'),
1657 'SPECIALMETHODS': ('specialnames', 'BASICMETHODS ATTRIBUTEMETHODS '
Georg Brandl395ed242009-09-04 08:07:32 +00001658 'CALLABLEMETHODS SEQUENCEMETHODS MAPPINGMETHODS '
1659 'NUMBERMETHODS CLASSES'),
Mark Dickinsona56c4672009-01-27 18:17:45 +00001660 'BASICMETHODS': ('customization', 'hash repr str SPECIALMETHODS'),
Georg Brandl6b38daa2008-06-01 21:05:17 +00001661 'ATTRIBUTEMETHODS': ('attribute-access', 'ATTRIBUTES SPECIALMETHODS'),
1662 'CALLABLEMETHODS': ('callable-types', 'CALLS SPECIALMETHODS'),
Georg Brandl395ed242009-09-04 08:07:32 +00001663 'SEQUENCEMETHODS': ('sequence-types', 'SEQUENCES SEQUENCEMETHODS '
Georg Brandl6b38daa2008-06-01 21:05:17 +00001664 'SPECIALMETHODS'),
1665 'MAPPINGMETHODS': ('sequence-types', 'MAPPINGS SPECIALMETHODS'),
1666 'NUMBERMETHODS': ('numeric-types', 'NUMBERS AUGMENTEDASSIGNMENT '
1667 'SPECIALMETHODS'),
1668 'EXECUTION': ('execmodel', 'NAMESPACES DYNAMICFEATURES EXCEPTIONS'),
Georg Brandl74abf6f2010-11-20 19:54:36 +00001669 'NAMESPACES': ('naming', 'global nonlocal ASSIGNMENT DELETION DYNAMICFEATURES'),
Georg Brandl6b38daa2008-06-01 21:05:17 +00001670 'DYNAMICFEATURES': ('dynamic-features', ''),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001671 'SCOPING': 'NAMESPACES',
1672 'FRAMES': 'NAMESPACES',
Georg Brandl6b38daa2008-06-01 21:05:17 +00001673 'EXCEPTIONS': ('exceptions', 'try except finally raise'),
1674 'CONVERSIONS': ('conversions', ''),
1675 'IDENTIFIERS': ('identifiers', 'keywords SPECIALIDENTIFIERS'),
1676 'SPECIALIDENTIFIERS': ('id-classes', ''),
1677 'PRIVATENAMES': ('atom-identifiers', ''),
1678 'LITERALS': ('atom-literals', 'STRINGS NUMBERS TUPLELITERALS '
1679 'LISTLITERALS DICTIONARYLITERALS'),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001680 'TUPLES': 'SEQUENCES',
Georg Brandl6b38daa2008-06-01 21:05:17 +00001681 'TUPLELITERALS': ('exprlists', 'TUPLES LITERALS'),
1682 'LISTS': ('typesseq-mutable', 'LISTLITERALS'),
1683 'LISTLITERALS': ('lists', 'LISTS LITERALS'),
1684 'DICTIONARIES': ('typesmapping', 'DICTIONARYLITERALS'),
1685 'DICTIONARYLITERALS': ('dict', 'DICTIONARIES LITERALS'),
1686 'ATTRIBUTES': ('attribute-references', 'getattr hasattr setattr ATTRIBUTEMETHODS'),
Georg Brandl395ed242009-09-04 08:07:32 +00001687 'SUBSCRIPTS': ('subscriptions', 'SEQUENCEMETHODS'),
1688 'SLICINGS': ('slicings', 'SEQUENCEMETHODS'),
Georg Brandl6b38daa2008-06-01 21:05:17 +00001689 'CALLS': ('calls', 'EXPRESSIONS'),
1690 'POWER': ('power', 'EXPRESSIONS'),
1691 'UNARY': ('unary', 'EXPRESSIONS'),
1692 'BINARY': ('binary', 'EXPRESSIONS'),
1693 'SHIFTING': ('shifting', 'EXPRESSIONS'),
1694 'BITWISE': ('bitwise', 'EXPRESSIONS'),
1695 'COMPARISON': ('comparisons', 'EXPRESSIONS BASICMETHODS'),
1696 'BOOLEAN': ('booleans', 'EXPRESSIONS TRUTHVALUE'),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001697 'ASSERTION': 'assert',
Georg Brandl6b38daa2008-06-01 21:05:17 +00001698 'ASSIGNMENT': ('assignment', 'AUGMENTEDASSIGNMENT'),
1699 'AUGMENTEDASSIGNMENT': ('augassign', 'NUMBERMETHODS'),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001700 'DELETION': 'del',
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001701 'RETURNING': 'return',
1702 'IMPORTING': 'import',
1703 'CONDITIONAL': 'if',
Georg Brandl6b38daa2008-06-01 21:05:17 +00001704 'LOOPING': ('compound', 'for while break continue'),
1705 'TRUTHVALUE': ('truth', 'if while and or not BASICMETHODS'),
1706 'DEBUGGING': ('debugger', 'pdb'),
1707 'CONTEXTMANAGERS': ('context-managers', 'with'),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001708 }
1709
Georg Brandl78aa3962010-07-31 21:51:48 +00001710 def __init__(self, input=None, output=None):
1711 self._input = input
1712 self._output = output
1713
Georg Brandl76ae3972010-08-01 06:32:55 +00001714 input = property(lambda self: self._input or sys.stdin)
1715 output = property(lambda self: self._output or sys.stdout)
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001716
Ka-Ping Yee79c009d2001-04-13 10:53:25 +00001717 def __repr__(self):
Ka-Ping Yee9bc576b2001-04-13 13:57:31 +00001718 if inspect.stack()[1][3] == '?':
Ka-Ping Yee79c009d2001-04-13 10:53:25 +00001719 self()
1720 return ''
Ka-Ping Yee9bc576b2001-04-13 13:57:31 +00001721 return '<pydoc.Helper instance>'
Ka-Ping Yee79c009d2001-04-13 10:53:25 +00001722
Alexander Belopolsky2e733c92010-07-04 17:00:20 +00001723 _GoInteractive = object()
1724 def __call__(self, request=_GoInteractive):
1725 if request is not self._GoInteractive:
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001726 self.help(request)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001727 else:
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001728 self.intro()
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001729 self.interact()
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001730 self.output.write('''
Fred Drakee61967f2001-05-10 18:41:02 +00001731You are now leaving help and returning to the Python interpreter.
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001732If you want to ask for help on a particular object directly from the
1733interpreter, you can type "help(object)". Executing "help('string')"
1734has the same effect as typing a particular string at the help> prompt.
1735''')
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001736
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001737 def interact(self):
1738 self.output.write('\n')
Guido van Rossum8ca162f2002-04-07 06:36:23 +00001739 while True:
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001740 try:
Johannes Gijsberse7691d32004-08-17 13:21:53 +00001741 request = self.getline('help> ')
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001742 if not request: break
Johannes Gijsberse7691d32004-08-17 13:21:53 +00001743 except (KeyboardInterrupt, EOFError):
1744 break
Neal Norwitz9d72bb42007-04-17 08:48:32 +00001745 request = replace(request, '"', '', "'", '').strip()
1746 if request.lower() in ('q', 'quit'): break
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001747 self.help(request)
1748
Johannes Gijsberse7691d32004-08-17 13:21:53 +00001749 def getline(self, prompt):
Guido van Rossum7c086c02008-01-02 03:52:38 +00001750 """Read one line, using input() when appropriate."""
Johannes Gijsberse7691d32004-08-17 13:21:53 +00001751 if self.input is sys.stdin:
Guido van Rossum7c086c02008-01-02 03:52:38 +00001752 return input(prompt)
Johannes Gijsberse7691d32004-08-17 13:21:53 +00001753 else:
1754 self.output.write(prompt)
1755 self.output.flush()
1756 return self.input.readline()
1757
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001758 def help(self, request):
1759 if type(request) is type(''):
R. David Murray1f1b9d32009-05-27 20:56:59 +00001760 request = request.strip()
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001761 if request == 'help': self.intro()
1762 elif request == 'keywords': self.listkeywords()
Georg Brandldb7b6b92009-01-01 15:53:14 +00001763 elif request == 'symbols': self.listsymbols()
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001764 elif request == 'topics': self.listtopics()
1765 elif request == 'modules': self.listmodules()
1766 elif request[:8] == 'modules ':
Neal Norwitz9d72bb42007-04-17 08:48:32 +00001767 self.listmodules(request.split()[1])
Georg Brandldb7b6b92009-01-01 15:53:14 +00001768 elif request in self.symbols: self.showsymbol(request)
Raymond Hettinger54f02222002-06-01 14:18:47 +00001769 elif request in self.keywords: self.showtopic(request)
1770 elif request in self.topics: self.showtopic(request)
Georg Brandld80d5f42010-12-03 07:47:22 +00001771 elif request: doc(request, 'Help on %s:', output=self._output)
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001772 elif isinstance(request, Helper): self()
Georg Brandld80d5f42010-12-03 07:47:22 +00001773 else: doc(request, 'Help on %s:', output=self._output)
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001774 self.output.write('\n')
1775
1776 def intro(self):
1777 self.output.write('''
1778Welcome to Python %s! This is the online help utility.
1779
1780If this is your first time using Python, you should definitely check out
Georg Brandl86def6c2008-01-21 20:36:10 +00001781the tutorial on the Internet at http://docs.python.org/tutorial/.
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001782
1783Enter the name of any module, keyword, or topic to get help on writing
1784Python programs and using Python modules. To quit this help utility and
1785return to the interpreter, just type "quit".
1786
1787To get a list of available modules, keywords, or topics, type "modules",
1788"keywords", or "topics". Each module also comes with a one-line summary
1789of what it does; to list the modules whose summaries contain a given word
1790such as "spam", type "modules spam".
1791''' % sys.version[:3])
1792
1793 def list(self, items, columns=4, width=80):
Guido van Rossum486364b2007-06-30 05:01:58 +00001794 items = list(sorted(items))
1795 colw = width // columns
1796 rows = (len(items) + columns - 1) // columns
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001797 for row in range(rows):
1798 for col in range(columns):
1799 i = col * rows + row
1800 if i < len(items):
1801 self.output.write(items[i])
1802 if col < columns - 1:
Guido van Rossum486364b2007-06-30 05:01:58 +00001803 self.output.write(' ' + ' ' * (colw - 1 - len(items[i])))
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001804 self.output.write('\n')
1805
1806 def listkeywords(self):
1807 self.output.write('''
1808Here is a list of the Python keywords. Enter any keyword to get more help.
1809
1810''')
1811 self.list(self.keywords.keys())
1812
Georg Brandldb7b6b92009-01-01 15:53:14 +00001813 def listsymbols(self):
1814 self.output.write('''
1815Here is a list of the punctuation symbols which Python assigns special meaning
1816to. Enter any symbol to get more help.
1817
1818''')
1819 self.list(self.symbols.keys())
1820
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001821 def listtopics(self):
1822 self.output.write('''
1823Here is a list of available topics. Enter any topic name to get more help.
1824
1825''')
1826 self.list(self.topics.keys())
1827
Georg Brandldb7b6b92009-01-01 15:53:14 +00001828 def showtopic(self, topic, more_xrefs=''):
Georg Brandl6b38daa2008-06-01 21:05:17 +00001829 try:
Georg Brandl5617db82009-04-27 16:28:57 +00001830 import pydoc_data.topics
Georg Brandl6b38daa2008-06-01 21:05:17 +00001831 except ImportError:
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001832 self.output.write('''
Georg Brandl6b38daa2008-06-01 21:05:17 +00001833Sorry, topic and keyword documentation is not available because the
Georg Brandl5617db82009-04-27 16:28:57 +00001834module "pydoc_data.topics" could not be found.
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001835''')
1836 return
1837 target = self.topics.get(topic, self.keywords.get(topic))
1838 if not target:
1839 self.output.write('no documentation found for %s\n' % repr(topic))
1840 return
1841 if type(target) is type(''):
Georg Brandldb7b6b92009-01-01 15:53:14 +00001842 return self.showtopic(target, more_xrefs)
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001843
Georg Brandl6b38daa2008-06-01 21:05:17 +00001844 label, xrefs = target
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001845 try:
Georg Brandl5617db82009-04-27 16:28:57 +00001846 doc = pydoc_data.topics.topics[label]
Georg Brandl6b38daa2008-06-01 21:05:17 +00001847 except KeyError:
1848 self.output.write('no documentation found for %s\n' % repr(topic))
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001849 return
Georg Brandl6b38daa2008-06-01 21:05:17 +00001850 pager(doc.strip() + '\n')
Georg Brandldb7b6b92009-01-01 15:53:14 +00001851 if more_xrefs:
1852 xrefs = (xrefs or '') + ' ' + more_xrefs
Ka-Ping Yeeda793892001-04-13 11:02:51 +00001853 if xrefs:
Georg Brandl6b38daa2008-06-01 21:05:17 +00001854 import io, formatter
Guido van Rossum34d19282007-08-09 01:03:29 +00001855 buffer = io.StringIO()
Ka-Ping Yeeda793892001-04-13 11:02:51 +00001856 formatter.DumbWriter(buffer).send_flowing_data(
Neal Norwitz9d72bb42007-04-17 08:48:32 +00001857 'Related help topics: ' + ', '.join(xrefs.split()) + '\n')
Ka-Ping Yeeda793892001-04-13 11:02:51 +00001858 self.output.write('\n%s\n' % buffer.getvalue())
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001859
Georg Brandldb7b6b92009-01-01 15:53:14 +00001860 def showsymbol(self, symbol):
1861 target = self.symbols[symbol]
1862 topic, _, xrefs = target.partition(' ')
1863 self.showtopic(topic, xrefs)
1864
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001865 def listmodules(self, key=''):
1866 if key:
1867 self.output.write('''
1868Here is a list of matching modules. Enter any module name to get more help.
1869
1870''')
1871 apropos(key)
1872 else:
1873 self.output.write('''
1874Please wait a moment while I gather a list of all available modules...
1875
1876''')
1877 modules = {}
1878 def callback(path, modname, desc, modules=modules):
1879 if modname and modname[-9:] == '.__init__':
1880 modname = modname[:-9] + ' (package)'
Neal Norwitz9d72bb42007-04-17 08:48:32 +00001881 if modname.find('.') < 0:
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001882 modules[modname] = 1
Christian Heimesd32ed6f2008-01-14 18:49:24 +00001883 def onerror(modname):
1884 callback(None, modname, None)
1885 ModuleScanner().run(callback, onerror=onerror)
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001886 self.list(modules.keys())
1887 self.output.write('''
1888Enter any module name to get more help. Or, type "modules spam" to search
1889for modules whose descriptions contain the word "spam".
1890''')
1891
Georg Brandl78aa3962010-07-31 21:51:48 +00001892help = Helper()
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001893
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001894class Scanner:
1895 """A generic tree iterator."""
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001896 def __init__(self, roots, children, descendp):
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001897 self.roots = roots[:]
1898 self.state = []
1899 self.children = children
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001900 self.descendp = descendp
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001901
1902 def next(self):
1903 if not self.state:
1904 if not self.roots:
1905 return None
1906 root = self.roots.pop(0)
1907 self.state = [(root, self.children(root))]
1908 node, children = self.state[-1]
1909 if not children:
1910 self.state.pop()
1911 return self.next()
1912 child = children.pop(0)
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001913 if self.descendp(child):
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001914 self.state.append((child, self.children(child)))
1915 return child
1916
Thomas Wouters49fd7fa2006-04-21 10:40:58 +00001917
1918class ModuleScanner:
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001919 """An interruptible scanner that searches module synopses."""
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001920
Christian Heimesd32ed6f2008-01-14 18:49:24 +00001921 def run(self, callback, key=None, completer=None, onerror=None):
Neal Norwitz9d72bb42007-04-17 08:48:32 +00001922 if key: key = key.lower()
Guido van Rossum8ca162f2002-04-07 06:36:23 +00001923 self.quit = False
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001924 seen = {}
1925
1926 for modname in sys.builtin_module_names:
Ka-Ping Yee239432a2001-03-02 02:45:08 +00001927 if modname != '__main__':
1928 seen[modname] = 1
Ka-Ping Yee66246962001-04-12 11:59:50 +00001929 if key is None:
1930 callback(None, modname, '')
1931 else:
Neal Norwitz9d72bb42007-04-17 08:48:32 +00001932 name = __import__(modname).__doc__ or ''
1933 desc = name.split('\n')[0]
1934 name = modname + ' - ' + desc
1935 if name.lower().find(key) >= 0:
Ka-Ping Yee66246962001-04-12 11:59:50 +00001936 callback(None, modname, desc)
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001937
Christian Heimesd32ed6f2008-01-14 18:49:24 +00001938 for importer, modname, ispkg in pkgutil.walk_packages(onerror=onerror):
Thomas Wouters49fd7fa2006-04-21 10:40:58 +00001939 if self.quit:
1940 break
1941 if key is None:
1942 callback(None, modname, '')
1943 else:
Georg Brandl126c8792009-04-05 15:05:48 +00001944 try:
1945 loader = importer.find_module(modname)
1946 except SyntaxError:
1947 # raised by tests for bad coding cookies or BOM
1948 continue
1949 if hasattr(loader, 'get_source'):
Amaury Forgeot d'Arc9196dc62008-06-19 20:54:32 +00001950 try:
1951 source = loader.get_source(modname)
1952 except UnicodeDecodeError:
1953 if onerror:
1954 onerror(modname)
1955 continue
Guido van Rossum34d19282007-08-09 01:03:29 +00001956 import io
Amaury Forgeot d'Arc9196dc62008-06-19 20:54:32 +00001957 desc = source_synopsis(io.StringIO(source)) or ''
Georg Brandl126c8792009-04-05 15:05:48 +00001958 if hasattr(loader, 'get_filename'):
Thomas Wouters49fd7fa2006-04-21 10:40:58 +00001959 path = loader.get_filename(modname)
Ka-Ping Yee66246962001-04-12 11:59:50 +00001960 else:
Thomas Wouters49fd7fa2006-04-21 10:40:58 +00001961 path = None
1962 else:
Amaury Forgeot d'Arc9196dc62008-06-19 20:54:32 +00001963 try:
1964 module = loader.load_module(modname)
1965 except ImportError:
1966 if onerror:
1967 onerror(modname)
1968 continue
Thomas Wouters49fd7fa2006-04-21 10:40:58 +00001969 desc = (module.__doc__ or '').splitlines()[0]
1970 path = getattr(module,'__file__',None)
Neal Norwitz9d72bb42007-04-17 08:48:32 +00001971 name = modname + ' - ' + desc
1972 if name.lower().find(key) >= 0:
Thomas Wouters49fd7fa2006-04-21 10:40:58 +00001973 callback(path, modname, desc)
1974
1975 if completer:
1976 completer()
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001977
1978def apropos(key):
1979 """Print all the one-line module summaries that contain a substring."""
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001980 def callback(path, modname, desc):
1981 if modname[-9:] == '.__init__':
1982 modname = modname[:-9] + ' (package)'
Guido van Rossumbe19ed72007-02-09 05:37:30 +00001983 print(modname, desc and '- ' + desc)
Amaury Forgeot d'Arc9196dc62008-06-19 20:54:32 +00001984 def onerror(modname):
1985 pass
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +00001986 try: import warnings
1987 except ImportError: pass
1988 else: warnings.filterwarnings('ignore') # ignore problems during import
Amaury Forgeot d'Arc9196dc62008-06-19 20:54:32 +00001989 ModuleScanner().run(callback, key, onerror=onerror)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001990
1991# --------------------------------------------------- web browser interface
1992
Ka-Ping Yee66246962001-04-12 11:59:50 +00001993def serve(port, callback=None, completer=None):
Georg Brandl83e9f4c2008-06-12 18:52:31 +00001994 import http.server, email.message, select
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001995
Georg Brandl24420152008-05-26 16:32:26 +00001996 class DocHandler(http.server.BaseHTTPRequestHandler):
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001997 def send_document(self, title, contents):
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001998 try:
1999 self.send_response(200)
Georg Brandl825fc8b2008-01-19 20:44:32 +00002000 self.send_header('Content-Type', 'text/html; charset=UTF-8')
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002001 self.end_headers()
Georg Brandl825fc8b2008-01-19 20:44:32 +00002002 self.wfile.write(html.page(title, contents).encode('utf-8'))
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002003 except IOError: pass
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002004
2005 def do_GET(self):
2006 path = self.path
2007 if path[-5:] == '.html': path = path[:-5]
2008 if path[:1] == '/': path = path[1:]
2009 if path and path != '.':
2010 try:
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00002011 obj = locate(path, forceload=1)
Guido van Rossumb940e112007-01-10 16:19:56 +00002012 except ErrorDuringImport as value:
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00002013 self.send_document(path, html.escape(str(value)))
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002014 return
Ka-Ping Yee3bda8792001-03-23 13:17:50 +00002015 if obj:
2016 self.send_document(describe(obj), html.document(obj, path))
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002017 else:
2018 self.send_document(path,
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00002019'no Python documentation found for %s' % repr(path))
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002020 else:
2021 heading = html.heading(
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002022'<big><big><strong>Python: Index of Modules</strong></big></big>',
2023'#ffffff', '#7799ee')
Ka-Ping Yee239432a2001-03-02 02:45:08 +00002024 def bltinlink(name):
2025 return '<a href="%s.html">%s</a>' % (name, name)
Georg Brandl661b0a12008-01-27 18:16:00 +00002026 names = [x for x in sys.builtin_module_names if x != '__main__']
2027 contents = html.multicolumn(names, bltinlink)
Ka-Ping Yee239432a2001-03-02 02:45:08 +00002028 indices = ['<p>' + html.bigsection(
2029 'Built-in Modules', '#ffffff', '#ee77aa', contents)]
2030
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002031 seen = {}
Thomas Wouters49fd7fa2006-04-21 10:40:58 +00002032 for dir in sys.path:
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002033 indices.append(html.index(dir, seen))
Neal Norwitz9d72bb42007-04-17 08:48:32 +00002034 contents = heading + ' '.join(indices) + '''<p align=right>
Tim Peters2306d242001-09-25 03:18:32 +00002035<font color="#909090" face="helvetica, arial"><strong>
2036pydoc</strong> by Ka-Ping Yee &lt;ping@lfw.org&gt;</font>'''
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002037 self.send_document('Index of Modules', contents)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002038
2039 def log_message(self, *args): pass
2040
Georg Brandl24420152008-05-26 16:32:26 +00002041 class DocServer(http.server.HTTPServer):
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002042 def __init__(self, port, callback):
Ronald Oussoren94f25282010-05-05 19:11:21 +00002043 host = 'localhost'
Senthil Kumaran7ff59132010-08-18 19:32:21 +00002044 self.address = (host, port)
Ka-Ping Yeedb8ed152001-03-02 05:58:17 +00002045 self.url = 'http://%s:%d/' % (host, port)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002046 self.callback = callback
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002047 self.base.__init__(self, self.address, self.handler)
2048
2049 def serve_until_quit(self):
2050 import select
Guido van Rossum8ca162f2002-04-07 06:36:23 +00002051 self.quit = False
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002052 while not self.quit:
2053 rd, wr, ex = select.select([self.socket.fileno()], [], [], 1)
2054 if rd: self.handle_request()
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002055
2056 def server_activate(self):
2057 self.base.server_activate(self)
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002058 if self.callback: self.callback(self)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002059
Georg Brandl24420152008-05-26 16:32:26 +00002060 DocServer.base = http.server.HTTPServer
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002061 DocServer.handler = DocHandler
Georg Brandl83e9f4c2008-06-12 18:52:31 +00002062 DocHandler.MessageClass = email.message.Message
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002063 try:
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00002064 try:
2065 DocServer(port, callback).serve_until_quit()
2066 except (KeyboardInterrupt, select.error):
2067 pass
2068 finally:
Ka-Ping Yee66246962001-04-12 11:59:50 +00002069 if completer: completer()
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002070
2071# ----------------------------------------------------- graphical interface
2072
2073def gui():
2074 """Graphical interface (starts web server and pops up a control window)."""
2075 class GUI:
2076 def __init__(self, window, port=7464):
2077 self.window = window
2078 self.server = None
2079 self.scanner = None
2080
Georg Brandl14fc4272008-05-17 18:39:55 +00002081 import tkinter
2082 self.server_frm = tkinter.Frame(window)
2083 self.title_lbl = tkinter.Label(self.server_frm,
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002084 text='Starting server...\n ')
Georg Brandl14fc4272008-05-17 18:39:55 +00002085 self.open_btn = tkinter.Button(self.server_frm,
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002086 text='open browser', command=self.open, state='disabled')
Georg Brandl14fc4272008-05-17 18:39:55 +00002087 self.quit_btn = tkinter.Button(self.server_frm,
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002088 text='quit serving', command=self.quit, state='disabled')
2089
Georg Brandl14fc4272008-05-17 18:39:55 +00002090 self.search_frm = tkinter.Frame(window)
2091 self.search_lbl = tkinter.Label(self.search_frm, text='Search for')
2092 self.search_ent = tkinter.Entry(self.search_frm)
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002093 self.search_ent.bind('<Return>', self.search)
Georg Brandl14fc4272008-05-17 18:39:55 +00002094 self.stop_btn = tkinter.Button(self.search_frm,
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002095 text='stop', pady=0, command=self.stop, state='disabled')
2096 if sys.platform == 'win32':
Ka-Ping Yeec92cdf72001-03-02 05:54:35 +00002097 # Trying to hide and show this button crashes under Windows.
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002098 self.stop_btn.pack(side='right')
2099
2100 self.window.title('pydoc')
2101 self.window.protocol('WM_DELETE_WINDOW', self.quit)
2102 self.title_lbl.pack(side='top', fill='x')
2103 self.open_btn.pack(side='left', fill='x', expand=1)
2104 self.quit_btn.pack(side='right', fill='x', expand=1)
2105 self.server_frm.pack(side='top', fill='x')
2106
2107 self.search_lbl.pack(side='left')
2108 self.search_ent.pack(side='right', fill='x', expand=1)
2109 self.search_frm.pack(side='top', fill='x')
2110 self.search_ent.focus_set()
2111
Ka-Ping Yeec92cdf72001-03-02 05:54:35 +00002112 font = ('helvetica', sys.platform == 'win32' and 8 or 10)
Georg Brandl14fc4272008-05-17 18:39:55 +00002113 self.result_lst = tkinter.Listbox(window, font=font, height=6)
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002114 self.result_lst.bind('<Button-1>', self.select)
2115 self.result_lst.bind('<Double-Button-1>', self.goto)
Georg Brandl14fc4272008-05-17 18:39:55 +00002116 self.result_scr = tkinter.Scrollbar(window,
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002117 orient='vertical', command=self.result_lst.yview)
2118 self.result_lst.config(yscrollcommand=self.result_scr.set)
2119
Georg Brandl14fc4272008-05-17 18:39:55 +00002120 self.result_frm = tkinter.Frame(window)
2121 self.goto_btn = tkinter.Button(self.result_frm,
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002122 text='go to selected', command=self.goto)
Georg Brandl14fc4272008-05-17 18:39:55 +00002123 self.hide_btn = tkinter.Button(self.result_frm,
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002124 text='hide results', command=self.hide)
2125 self.goto_btn.pack(side='left', fill='x', expand=1)
2126 self.hide_btn.pack(side='right', fill='x', expand=1)
2127
2128 self.window.update()
2129 self.minwidth = self.window.winfo_width()
2130 self.minheight = self.window.winfo_height()
2131 self.bigminheight = (self.server_frm.winfo_reqheight() +
2132 self.search_frm.winfo_reqheight() +
2133 self.result_lst.winfo_reqheight() +
2134 self.result_frm.winfo_reqheight())
2135 self.bigwidth, self.bigheight = self.minwidth, self.bigminheight
2136 self.expanded = 0
2137 self.window.wm_geometry('%dx%d' % (self.minwidth, self.minheight))
2138 self.window.wm_minsize(self.minwidth, self.minheight)
Martin v. Löwis5b26abb2002-12-28 09:23:09 +00002139 self.window.tk.willdispatch()
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002140
2141 import threading
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00002142 threading.Thread(
2143 target=serve, args=(port, self.ready, self.quit)).start()
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002144
2145 def ready(self, server):
2146 self.server = server
2147 self.title_lbl.config(
2148 text='Python documentation server at\n' + server.url)
2149 self.open_btn.config(state='normal')
2150 self.quit_btn.config(state='normal')
2151
Ka-Ping Yee239432a2001-03-02 02:45:08 +00002152 def open(self, event=None, url=None):
2153 url = url or self.server.url
2154 try:
2155 import webbrowser
2156 webbrowser.open(url)
2157 except ImportError: # pre-webbrowser.py compatibility
Ka-Ping Yeec92cdf72001-03-02 05:54:35 +00002158 if sys.platform == 'win32':
Ka-Ping Yee239432a2001-03-02 02:45:08 +00002159 os.system('start "%s"' % url)
Ka-Ping Yee239432a2001-03-02 02:45:08 +00002160 else:
2161 rc = os.system('netscape -remote "openURL(%s)" &' % url)
2162 if rc: os.system('netscape "%s" &' % url)
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002163
2164 def quit(self, event=None):
2165 if self.server:
2166 self.server.quit = 1
2167 self.window.quit()
2168
2169 def search(self, event=None):
2170 key = self.search_ent.get()
2171 self.stop_btn.pack(side='right')
2172 self.stop_btn.config(state='normal')
2173 self.search_lbl.config(text='Searching for "%s"...' % key)
2174 self.search_ent.forget()
2175 self.search_lbl.pack(side='left')
2176 self.result_lst.delete(0, 'end')
2177 self.goto_btn.config(state='disabled')
2178 self.expand()
2179
2180 import threading
2181 if self.scanner:
2182 self.scanner.quit = 1
2183 self.scanner = ModuleScanner()
2184 threading.Thread(target=self.scanner.run,
Ka-Ping Yee6dcfa382001-04-12 20:27:31 +00002185 args=(self.update, key, self.done)).start()
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002186
2187 def update(self, path, modname, desc):
2188 if modname[-9:] == '.__init__':
2189 modname = modname[:-9] + ' (package)'
2190 self.result_lst.insert('end',
2191 modname + ' - ' + (desc or '(no description)'))
2192
2193 def stop(self, event=None):
2194 if self.scanner:
2195 self.scanner.quit = 1
2196 self.scanner = None
2197
2198 def done(self):
2199 self.scanner = None
2200 self.search_lbl.config(text='Search for')
2201 self.search_lbl.pack(side='left')
2202 self.search_ent.pack(side='right', fill='x', expand=1)
2203 if sys.platform != 'win32': self.stop_btn.forget()
2204 self.stop_btn.config(state='disabled')
2205
2206 def select(self, event=None):
2207 self.goto_btn.config(state='normal')
2208
2209 def goto(self, event=None):
2210 selection = self.result_lst.curselection()
2211 if selection:
Neal Norwitz9d72bb42007-04-17 08:48:32 +00002212 modname = self.result_lst.get(selection[0]).split()[0]
Ka-Ping Yee239432a2001-03-02 02:45:08 +00002213 self.open(url=self.server.url + modname + '.html')
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002214
2215 def collapse(self):
2216 if not self.expanded: return
2217 self.result_frm.forget()
2218 self.result_scr.forget()
2219 self.result_lst.forget()
2220 self.bigwidth = self.window.winfo_width()
2221 self.bigheight = self.window.winfo_height()
2222 self.window.wm_geometry('%dx%d' % (self.minwidth, self.minheight))
2223 self.window.wm_minsize(self.minwidth, self.minheight)
2224 self.expanded = 0
2225
2226 def expand(self):
2227 if self.expanded: return
2228 self.result_frm.pack(side='bottom', fill='x')
2229 self.result_scr.pack(side='right', fill='y')
2230 self.result_lst.pack(side='top', fill='both', expand=1)
2231 self.window.wm_geometry('%dx%d' % (self.bigwidth, self.bigheight))
2232 self.window.wm_minsize(self.minwidth, self.bigminheight)
2233 self.expanded = 1
2234
2235 def hide(self, event=None):
2236 self.stop()
2237 self.collapse()
2238
Georg Brandl14fc4272008-05-17 18:39:55 +00002239 import tkinter
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002240 try:
Georg Brandl14fc4272008-05-17 18:39:55 +00002241 root = tkinter.Tk()
Martin v. Löwise09bd932004-08-22 16:13:26 +00002242 # Tk will crash if pythonw.exe has an XP .manifest
2243 # file and the root has is not destroyed explicitly.
2244 # If the problem is ever fixed in Tk, the explicit
2245 # destroy can go.
2246 try:
2247 gui = GUI(root)
2248 root.mainloop()
2249 finally:
2250 root.destroy()
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002251 except KeyboardInterrupt:
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002252 pass
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002253
2254# -------------------------------------------------- command-line interface
2255
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00002256def ispath(x):
Neal Norwitz9d72bb42007-04-17 08:48:32 +00002257 return isinstance(x, str) and x.find(os.sep) >= 0
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00002258
Ka-Ping Yee1d384632001-03-01 00:24:32 +00002259def cli():
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002260 """Command-line interface (looks at sys.argv to decide what to do)."""
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002261 import getopt
Guido van Rossum756aa932007-04-07 03:04:01 +00002262 class BadUsage(Exception): pass
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002263
Nick Coghlan106274b2009-11-15 23:04:33 +00002264 # Scripts don't get the current directory in their path by default
2265 # unless they are run with the '-m' switch
2266 if '' not in sys.path:
2267 scriptdir = os.path.dirname(sys.argv[0])
2268 if scriptdir in sys.path:
2269 sys.path.remove(scriptdir)
2270 sys.path.insert(0, '.')
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002271
Ka-Ping Yee3bda8792001-03-23 13:17:50 +00002272 try:
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002273 opts, args = getopt.getopt(sys.argv[1:], 'gk:p:w')
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002274 writing = 0
2275
2276 for opt, val in opts:
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002277 if opt == '-g':
2278 gui()
2279 return
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002280 if opt == '-k':
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002281 apropos(val)
2282 return
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002283 if opt == '-p':
2284 try:
2285 port = int(val)
2286 except ValueError:
2287 raise BadUsage
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002288 def ready(server):
Guido van Rossumbe19ed72007-02-09 05:37:30 +00002289 print('pydoc server ready at %s' % server.url)
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00002290 def stopped():
Guido van Rossumbe19ed72007-02-09 05:37:30 +00002291 print('pydoc server stopped')
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00002292 serve(port, ready, stopped)
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002293 return
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002294 if opt == '-w':
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002295 writing = 1
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002296
2297 if not args: raise BadUsage
2298 for arg in args:
Ka-Ping Yee45daeb02002-08-11 15:11:33 +00002299 if ispath(arg) and not os.path.exists(arg):
Guido van Rossumbe19ed72007-02-09 05:37:30 +00002300 print('file %r does not exist' % arg)
Ka-Ping Yee45daeb02002-08-11 15:11:33 +00002301 break
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002302 try:
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00002303 if ispath(arg) and os.path.isfile(arg):
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002304 arg = importfile(arg)
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00002305 if writing:
2306 if ispath(arg) and os.path.isdir(arg):
2307 writedocs(arg)
2308 else:
2309 writedoc(arg)
2310 else:
Martin v. Löwisb8c084e2003-06-14 09:03:46 +00002311 help.help(arg)
Guido van Rossumb940e112007-01-10 16:19:56 +00002312 except ErrorDuringImport as value:
Guido van Rossumbe19ed72007-02-09 05:37:30 +00002313 print(value)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002314
2315 except (getopt.error, BadUsage):
Martin v. Löwisf9b08b82003-10-31 13:05:21 +00002316 cmd = os.path.basename(sys.argv[0])
Guido van Rossumbe19ed72007-02-09 05:37:30 +00002317 print("""pydoc - the Python documentation tool
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002318
2319%s <name> ...
2320 Show text documentation on something. <name> may be the name of a
Martin v. Löwisb8c084e2003-06-14 09:03:46 +00002321 Python keyword, topic, function, module, or package, or a dotted
2322 reference to a class or function within a module or module in a
2323 package. If <name> contains a '%s', it is used as the path to a
2324 Python source file to document. If name is 'keywords', 'topics',
2325 or 'modules', a listing of these things is displayed.
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002326
2327%s -k <keyword>
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002328 Search for a keyword in the synopsis lines of all available modules.
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002329
2330%s -p <port>
2331 Start an HTTP server on the given port on the local machine.
2332
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002333%s -g
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +00002334 Pop up a graphical interface for finding and serving documentation.
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002335
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002336%s -w <name> ...
2337 Write out the HTML documentation for a module to a file in the current
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +00002338 directory. If <name> contains a '%s', it is treated as a filename; if
2339 it names a directory, documentation is written for all the contents.
Guido van Rossumbe19ed72007-02-09 05:37:30 +00002340""" % (cmd, os.sep, cmd, cmd, cmd, cmd, os.sep))
Ka-Ping Yee1d384632001-03-01 00:24:32 +00002341
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002342if __name__ == '__main__': cli()