blob: 5e91ad21ee0cd562fc125e39deee63e8a20b6430 [file] [log] [blame]
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001#!/usr/bin/env python
Guido van Rossumfce538c2002-08-06 17:29:38 +00002# -*- coding: Latin-1 -*-
Ka-Ping Yee1d384632001-03-01 00:24:32 +00003"""Generate Python documentation in HTML or text for interactive use.
Ka-Ping Yeedd175342001-02-27 14:43:46 +00004
Ka-Ping Yeedd175342001-02-27 14:43:46 +00005In the Python interpreter, do "from pydoc import help" to provide online
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00006help. Calling help(thing) on a Python object documents the object.
7
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00008Or, at the shell command line outside of Python:
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00009
Ka-Ping Yee37f7b382001-03-23 00:12:53 +000010Run "pydoc <name>" to show documentation on something. <name> may be
11the name of a function, module, package, or a dotted reference to a
12class or function within a module or module in a package. If the
13argument contains a path segment delimiter (e.g. slash on Unix,
14backslash on Windows) it is treated as the path to a Python source file.
Ka-Ping Yee66efbc72001-03-01 13:55:20 +000015
Ka-Ping Yee37f7b382001-03-23 00:12:53 +000016Run "pydoc -k <keyword>" to search for a keyword in the synopsis lines
17of all available modules.
Ka-Ping Yee66efbc72001-03-01 13:55:20 +000018
Ka-Ping Yee37f7b382001-03-23 00:12:53 +000019Run "pydoc -p <port>" to start an HTTP server on a given port on the
20local machine to generate documentation web pages.
Ka-Ping Yee66efbc72001-03-01 13:55:20 +000021
Ka-Ping Yee37f7b382001-03-23 00:12:53 +000022For platforms without a command line, "pydoc -g" starts the HTTP server
23and also pops up a little window for controlling it.
24
25Run "pydoc -w <name>" to write out the HTML documentation for a module
26to a file named "<name>.html".
Skip Montanaro4997a692003-09-10 16:47:51 +000027
28Module docs for core modules are assumed to be in
29
30 http://www.python.org/doc/current/lib/
31
32This can be overridden by setting the PYTHONDOCS environment variable
33to a different URL or to a local directory containing the Library
34Reference Manual pages.
Ka-Ping Yee66efbc72001-03-01 13:55:20 +000035"""
Ka-Ping Yeedd175342001-02-27 14:43:46 +000036
37__author__ = "Ka-Ping Yee <ping@lfw.org>"
Ka-Ping Yee6f3f9a42001-02-27 22:42:36 +000038__date__ = "26 February 2001"
Jeremy Hylton3e0055f2005-10-20 19:59:25 +000039
Ka-Ping Yee09d7d9a2001-02-27 22:43:48 +000040__version__ = "$Revision$"
Martin v. Löwis6fe8f192004-11-14 10:21:04 +000041__credits__ = """Guido van Rossum, for an excellent programming language.
Ka-Ping Yee5e2b1732001-02-27 23:35:09 +000042Tommy Burnette, the original creator of manpy.
Ka-Ping Yee6f3f9a42001-02-27 22:42:36 +000043Paul Prescod, for all his work on onlinehelp.
44Richard Chamberlain, for the first implementation of textdoc.
Raymond Hettingered2dbe32005-01-01 07:51:01 +000045"""
Ka-Ping Yeedd175342001-02-27 14:43:46 +000046
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +000047# Known bugs that can't be fixed here:
48# - imp.load_module() cannot be prevented from clobbering existing
49# loaded modules, so calling synopsis() on a binary module file
50# changes the contents of any existing module with the same name.
51# - If the __file__ attribute on a module is a relative path and
52# the current directory is changed with os.chdir(), an incorrect
53# path will be displayed.
Ka-Ping Yee66efbc72001-03-01 13:55:20 +000054
Thomas Wouters49fd7fa2006-04-21 10:40:58 +000055import sys, imp, os, re, types, inspect, __builtin__, pkgutil
Ka-Ping Yeedd175342001-02-27 14:43:46 +000056from repr import Repr
Thomas Wouters49fd7fa2006-04-21 10:40:58 +000057try:
58 from collections import deque
59except ImportError:
60 # Python 2.3 compatibility
61 class deque(list):
62 def popleft(self):
63 return self.pop(0)
Ka-Ping Yeedd175342001-02-27 14:43:46 +000064
65# --------------------------------------------------------- common routines
66
Ka-Ping Yeedd175342001-02-27 14:43:46 +000067def pathdirs():
68 """Convert sys.path into a list of absolute, existing, unique paths."""
69 dirs = []
Ka-Ping Yee1d384632001-03-01 00:24:32 +000070 normdirs = []
Ka-Ping Yeedd175342001-02-27 14:43:46 +000071 for dir in sys.path:
72 dir = os.path.abspath(dir or '.')
Ka-Ping Yee1d384632001-03-01 00:24:32 +000073 normdir = os.path.normcase(dir)
74 if normdir not in normdirs and os.path.isdir(dir):
Ka-Ping Yeedd175342001-02-27 14:43:46 +000075 dirs.append(dir)
Ka-Ping Yee1d384632001-03-01 00:24:32 +000076 normdirs.append(normdir)
Ka-Ping Yeedd175342001-02-27 14:43:46 +000077 return dirs
78
79def getdoc(object):
80 """Get the doc string or comments for an object."""
Ka-Ping Yee3bda8792001-03-23 13:17:50 +000081 result = inspect.getdoc(object) or inspect.getcomments(object)
Neal Norwitz9d72bb42007-04-17 08:48:32 +000082 return result and re.sub('^ *\n', '', result.rstrip()) or ''
Ka-Ping Yeedd175342001-02-27 14:43:46 +000083
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +000084def splitdoc(doc):
85 """Split a doc string into a synopsis line (if any) and the rest."""
Neal Norwitz9d72bb42007-04-17 08:48:32 +000086 lines = doc.strip().split('\n')
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +000087 if len(lines) == 1:
88 return lines[0], ''
Neal Norwitz9d72bb42007-04-17 08:48:32 +000089 elif len(lines) >= 2 and not lines[1].rstrip():
90 return lines[0], '\n'.join(lines[2:])
91 return '', '\n'.join(lines)
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +000092
Ka-Ping Yeedd175342001-02-27 14:43:46 +000093def classname(object, modname):
94 """Get a class name and qualify it with a module name if necessary."""
95 name = object.__name__
96 if object.__module__ != modname:
97 name = object.__module__ + '.' + name
98 return name
99
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000100def isdata(object):
Georg Brandl94432422005-07-22 21:52:25 +0000101 """Check if an object is of a type that probably means it's data."""
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000102 return not (inspect.ismodule(object) or inspect.isclass(object) or
103 inspect.isroutine(object) or inspect.isframe(object) or
104 inspect.istraceback(object) or inspect.iscode(object))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000105
106def replace(text, *pairs):
107 """Do a series of global replacements on a string."""
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +0000108 while pairs:
Neal Norwitz9d72bb42007-04-17 08:48:32 +0000109 text = pairs[1].join(text.split(pairs[0]))
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +0000110 pairs = pairs[2:]
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000111 return text
112
113def cram(text, maxlen):
114 """Omit part of a string if needed to make it fit in a maximum length."""
115 if len(text) > maxlen:
Raymond Hettingerfca3bb62002-10-21 04:44:11 +0000116 pre = max(0, (maxlen-3)//2)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000117 post = max(0, maxlen-3-pre)
118 return text[:pre] + '...' + text[len(text)-post:]
119 return text
120
Brett Cannon84601f12004-06-19 01:22:48 +0000121_re_stripid = re.compile(r' at 0x[0-9a-f]{6,16}(>+)$', re.IGNORECASE)
Ka-Ping Yee1d384632001-03-01 00:24:32 +0000122def stripid(text):
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000123 """Remove the hexadecimal id from a Python object representation."""
Brett Cannonc6c1f472004-06-19 01:02:51 +0000124 # The behaviour of %p is implementation-dependent in terms of case.
125 if _re_stripid.search(repr(Exception)):
126 return _re_stripid.sub(r'\1', text)
Ka-Ping Yee1d384632001-03-01 00:24:32 +0000127 return text
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000128
Brett Cannonc6c1f472004-06-19 01:02:51 +0000129def _is_some_method(obj):
130 return inspect.ismethod(obj) or inspect.ismethoddescriptor(obj)
Tim Peters536d2262001-09-20 05:13:38 +0000131
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000132def allmethods(cl):
133 methods = {}
Tim Peters536d2262001-09-20 05:13:38 +0000134 for key, value in inspect.getmembers(cl, _is_some_method):
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000135 methods[key] = 1
136 for base in cl.__bases__:
137 methods.update(allmethods(base)) # all your base are belong to us
138 for key in methods.keys():
139 methods[key] = getattr(cl, key)
140 return methods
141
Tim Petersfa26f7c2001-09-24 08:05:11 +0000142def _split_list(s, predicate):
143 """Split sequence s via predicate, and return pair ([true], [false]).
144
145 The return value is a 2-tuple of lists,
146 ([x for x in s if predicate(x)],
147 [x for x in s if not predicate(x)])
148 """
149
Tim Peters28355492001-09-23 21:29:55 +0000150 yes = []
151 no = []
Tim Petersfa26f7c2001-09-24 08:05:11 +0000152 for x in s:
153 if predicate(x):
154 yes.append(x)
Tim Peters28355492001-09-23 21:29:55 +0000155 else:
Tim Petersfa26f7c2001-09-24 08:05:11 +0000156 no.append(x)
Tim Peters28355492001-09-23 21:29:55 +0000157 return yes, no
158
Skip Montanaroa5616d22004-06-11 04:46:12 +0000159def visiblename(name, all=None):
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000160 """Decide whether to show documentation on a variable."""
161 # Certain special names are redundant.
Raymond Hettingerdbecd932005-02-06 06:57:08 +0000162 if name in ('__builtins__', '__doc__', '__file__', '__path__',
163 '__module__', '__name__', '__slots__'): return 0
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000164 # Private names are hidden, but special names are displayed.
165 if name.startswith('__') and name.endswith('__'): return 1
Skip Montanaroa5616d22004-06-11 04:46:12 +0000166 if all is not None:
167 # only document that which the programmer exported in __all__
168 return name in all
169 else:
170 return not name.startswith('_')
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000171
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +0000172def classify_class_attrs(object):
173 """Wrap inspect.classify_class_attrs, with fixup for data descriptors."""
174 def fixup((name, kind, cls, value)):
175 if inspect.isdatadescriptor(value):
176 kind = 'data descriptor'
177 return name, kind, cls, value
178 return map(fixup, inspect.classify_class_attrs(object))
179
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000180# ----------------------------------------------------- module manipulation
181
182def ispackage(path):
183 """Guess whether a path refers to a package directory."""
184 if os.path.isdir(path):
Raymond Hettingerdbecd932005-02-06 06:57:08 +0000185 for ext in ('.py', '.pyc', '.pyo'):
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000186 if os.path.isfile(os.path.join(path, '__init__' + ext)):
Tim Petersbc0e9102002-04-04 22:55:58 +0000187 return True
188 return False
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000189
Thomas Wouters49fd7fa2006-04-21 10:40:58 +0000190def source_synopsis(file):
191 line = file.readline()
Neal Norwitz9d72bb42007-04-17 08:48:32 +0000192 while line[:1] == '#' or not line.strip():
Thomas Wouters49fd7fa2006-04-21 10:40:58 +0000193 line = file.readline()
194 if not line: break
Neal Norwitz9d72bb42007-04-17 08:48:32 +0000195 line = line.strip()
Thomas Wouters49fd7fa2006-04-21 10:40:58 +0000196 if line[:4] == 'r"""': line = line[1:]
197 if line[:3] == '"""':
198 line = line[3:]
199 if line[-1:] == '\\': line = line[:-1]
Neal Norwitz9d72bb42007-04-17 08:48:32 +0000200 while not line.strip():
Thomas Wouters49fd7fa2006-04-21 10:40:58 +0000201 line = file.readline()
202 if not line: break
Neal Norwitz9d72bb42007-04-17 08:48:32 +0000203 result = line.split('"""')[0].strip()
Thomas Wouters49fd7fa2006-04-21 10:40:58 +0000204 else: result = None
205 return result
206
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000207def synopsis(filename, cache={}):
208 """Get the one-line summary out of a module file."""
Raymond Hettinger32200ae2002-06-01 19:51:15 +0000209 mtime = os.stat(filename).st_mtime
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000210 lastupdate, result = cache.get(filename, (0, None))
211 if lastupdate < mtime:
212 info = inspect.getmoduleinfo(filename)
Georg Brandl26fd2e12006-03-08 09:34:53 +0000213 try:
214 file = open(filename)
215 except IOError:
216 # module can't be opened, so skip it
217 return None
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000218 if info and 'b' in info[2]: # binary modules have to be imported
219 try: module = imp.load_module('__temp__', file, filename, info[1:])
220 except: return None
Thomas Wouters49fd7fa2006-04-21 10:40:58 +0000221 result = (module.__doc__ or '').splitlines()[0]
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000222 del sys.modules['__temp__']
223 else: # text modules can be directly examined
Thomas Wouters49fd7fa2006-04-21 10:40:58 +0000224 result = source_synopsis(file)
225 file.close()
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000226 cache[filename] = (mtime, result)
227 return result
228
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000229class ErrorDuringImport(Exception):
230 """Errors that occurred while trying to import something to document it."""
231 def __init__(self, filename, (exc, value, tb)):
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000232 self.filename = filename
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000233 self.exc = exc
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000234 self.value = value
235 self.tb = tb
236
237 def __str__(self):
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000238 exc = self.exc
239 if type(exc) is types.ClassType:
240 exc = exc.__name__
241 return 'problem in %s - %s: %s' % (self.filename, exc, self.value)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000242
243def importfile(path):
244 """Import a Python source file or compiled file given its path."""
245 magic = imp.get_magic()
246 file = open(path, 'r')
247 if file.read(len(magic)) == magic:
248 kind = imp.PY_COMPILED
249 else:
250 kind = imp.PY_SOURCE
251 file.close()
252 filename = os.path.basename(path)
253 name, ext = os.path.splitext(filename)
254 file = open(path, 'r')
255 try:
256 module = imp.load_module(name, file, path, (ext, 'r', kind))
257 except:
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000258 raise ErrorDuringImport(path, sys.exc_info())
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000259 file.close()
260 return module
261
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000262def safeimport(path, forceload=0, cache={}):
263 """Import a module; handle errors; return None if the module isn't found.
264
265 If the module *is* found but an exception occurs, it's wrapped in an
266 ErrorDuringImport exception and reraised. Unlike __import__, if a
267 package path is specified, the module at the end of the path is returned,
268 not the package at the beginning. If the optional 'forceload' argument
269 is 1, we reload the module from disk (unless it's a dynamic extension)."""
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000270 try:
Ka-Ping Yee9a2dcf82005-11-05 05:04:41 +0000271 # If forceload is 1 and the module has been previously loaded from
272 # disk, we always have to reload the module. Checking the file's
273 # mtime isn't good enough (e.g. the module could contain a class
274 # that inherits from another module that has changed).
275 if forceload and path in sys.modules:
276 if path not in sys.builtin_module_names:
277 # Avoid simply calling reload() because it leaves names in
278 # the currently loaded module lying around if they're not
279 # defined in the new source file. Instead, remove the
280 # module from sys.modules and re-import. Also remove any
281 # submodules because they won't appear in the newly loaded
282 # module's namespace if they're already in sys.modules.
283 subs = [m for m in sys.modules if m.startswith(path + '.')]
284 for key in [path] + subs:
285 # Prevent garbage collection.
286 cache[key] = sys.modules[key]
287 del sys.modules[key]
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000288 module = __import__(path)
289 except:
290 # Did the error occur before or after the module was found?
291 (exc, value, tb) = info = sys.exc_info()
Raymond Hettinger54f02222002-06-01 14:18:47 +0000292 if path in sys.modules:
Fred Drakedb390c12005-10-28 14:39:47 +0000293 # An error occurred while executing the imported module.
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000294 raise ErrorDuringImport(sys.modules[path].__file__, info)
295 elif exc is SyntaxError:
296 # A SyntaxError occurred before we could execute the module.
297 raise ErrorDuringImport(value.filename, info)
298 elif exc is ImportError and \
Neal Norwitz9d72bb42007-04-17 08:48:32 +0000299 str(value).lower().split()[:2] == ['no', 'module']:
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000300 # The module was not found.
301 return None
302 else:
303 # Some other error occurred during the importing process.
304 raise ErrorDuringImport(path, sys.exc_info())
Neal Norwitz9d72bb42007-04-17 08:48:32 +0000305 for part in path.split('.')[1:]:
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000306 try: module = getattr(module, part)
307 except AttributeError: return None
308 return module
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000309
310# ---------------------------------------------------- formatter base class
311
312class Doc:
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000313 def document(self, object, name=None, *args):
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000314 """Generate documentation for an object."""
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000315 args = (object, name) + args
Brett Cannon28a4f0f2003-06-11 23:38:55 +0000316 # 'try' clause is to attempt to handle the possibility that inspect
317 # identifies something in a way that pydoc itself has issues handling;
318 # think 'super' and how it is a descriptor (which raises the exception
319 # by lacking a __name__ attribute) and an instance.
Thomas Wouters0e3f5912006-08-11 14:57:12 +0000320 if inspect.isgetsetdescriptor(object): return self.docdata(*args)
321 if inspect.ismemberdescriptor(object): return self.docdata(*args)
Brett Cannon28a4f0f2003-06-11 23:38:55 +0000322 try:
323 if inspect.ismodule(object): return self.docmodule(*args)
324 if inspect.isclass(object): return self.docclass(*args)
325 if inspect.isroutine(object): return self.docroutine(*args)
326 except AttributeError:
327 pass
Johannes Gijsbers8de645a2004-11-07 19:16:05 +0000328 if isinstance(object, property): return self.docproperty(*args)
Guido van Rossum68468eb2003-02-27 20:14:51 +0000329 return self.docother(*args)
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000330
331 def fail(self, object, name=None, *args):
332 """Raise an exception for unimplemented types."""
333 message = "don't know how to document object%s of type %s" % (
334 name and ' ' + repr(name), type(object).__name__)
335 raise TypeError, message
336
Thomas Wouters0e3f5912006-08-11 14:57:12 +0000337 docmodule = docclass = docroutine = docother = docproperty = docdata = fail
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000338
Skip Montanaro4997a692003-09-10 16:47:51 +0000339 def getdocloc(self, object):
340 """Return the location of module docs or None"""
341
342 try:
343 file = inspect.getabsfile(object)
344 except TypeError:
345 file = '(built-in)'
346
347 docloc = os.environ.get("PYTHONDOCS",
348 "http://www.python.org/doc/current/lib")
349 basedir = os.path.join(sys.exec_prefix, "lib",
350 "python"+sys.version[0:3])
351 if (isinstance(object, type(os)) and
352 (object.__name__ in ('errno', 'exceptions', 'gc', 'imp',
353 'marshal', 'posix', 'signal', 'sys',
354 'thread', 'zipimport') or
355 (file.startswith(basedir) and
356 not file.startswith(os.path.join(basedir, 'site-packages'))))):
Skip Montanarof2134842004-06-07 02:40:05 +0000357 htmlfile = "module-%s.html" % object.__name__
Skip Montanaro4997a692003-09-10 16:47:51 +0000358 if docloc.startswith("http://"):
Skip Montanarof2134842004-06-07 02:40:05 +0000359 docloc = "%s/%s" % (docloc.rstrip("/"), htmlfile)
Skip Montanaro4997a692003-09-10 16:47:51 +0000360 else:
Skip Montanarof2134842004-06-07 02:40:05 +0000361 docloc = os.path.join(docloc, htmlfile)
Skip Montanaro4997a692003-09-10 16:47:51 +0000362 else:
363 docloc = None
364 return docloc
365
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000366# -------------------------------------------- HTML documentation generator
367
368class HTMLRepr(Repr):
369 """Class for safely making an HTML representation of a Python object."""
370 def __init__(self):
371 Repr.__init__(self)
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000372 self.maxlist = self.maxtuple = 20
373 self.maxdict = 10
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000374 self.maxstring = self.maxother = 100
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000375
376 def escape(self, text):
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +0000377 return replace(text, '&', '&amp;', '<', '&lt;', '>', '&gt;')
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000378
379 def repr(self, object):
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000380 return Repr.repr(self, object)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000381
382 def repr1(self, x, level):
Skip Montanaro0fe8fce2003-06-27 15:45:41 +0000383 if hasattr(type(x), '__name__'):
Neal Norwitz9d72bb42007-04-17 08:48:32 +0000384 methodname = 'repr_' + '_'.join(type(x).__name__.split())
Skip Montanaro0fe8fce2003-06-27 15:45:41 +0000385 if hasattr(self, methodname):
386 return getattr(self, methodname)(x, level)
387 return self.escape(cram(stripid(repr(x)), self.maxother))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000388
389 def repr_string(self, x, level):
Ka-Ping Yeea2fe1032001-03-02 01:19:14 +0000390 test = cram(x, self.maxstring)
391 testrepr = repr(test)
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +0000392 if '\\' in test and '\\' not in replace(testrepr, r'\\', ''):
Ka-Ping Yeea2fe1032001-03-02 01:19:14 +0000393 # Backslashes are only literal in the string and are never
394 # needed to make any special characters, so show a raw string.
395 return 'r' + testrepr[0] + self.escape(test) + testrepr[0]
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +0000396 return re.sub(r'((\\[\\abfnrtv\'"]|\\[0-9]..|\\x..|\\u....)+)',
Ka-Ping Yeea2fe1032001-03-02 01:19:14 +0000397 r'<font color="#c040c0">\1</font>',
398 self.escape(testrepr))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000399
Skip Montanarodf708782002-03-07 22:58:02 +0000400 repr_str = repr_string
401
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000402 def repr_instance(self, x, level):
403 try:
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000404 return self.escape(cram(stripid(repr(x)), self.maxstring))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000405 except:
406 return self.escape('<%s instance>' % x.__class__.__name__)
407
408 repr_unicode = repr_string
409
410class HTMLDoc(Doc):
411 """Formatter class for HTML documentation."""
412
413 # ------------------------------------------- HTML formatting utilities
414
415 _repr_instance = HTMLRepr()
416 repr = _repr_instance.repr
417 escape = _repr_instance.escape
418
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000419 def page(self, title, contents):
420 """Format an HTML page."""
421 return '''
Tim Peters59ed4482001-10-31 04:20:26 +0000422<!doctype html PUBLIC "-//W3C//DTD HTML 4.0 Transitional//EN">
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000423<html><head><title>Python: %s</title>
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000424</head><body bgcolor="#f0f0f8">
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000425%s
426</body></html>''' % (title, contents)
427
428 def heading(self, title, fgcol, bgcol, extras=''):
429 """Format a page heading."""
430 return '''
Tim Peters59ed4482001-10-31 04:20:26 +0000431<table width="100%%" cellspacing=0 cellpadding=2 border=0 summary="heading">
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000432<tr bgcolor="%s">
Tim Peters2306d242001-09-25 03:18:32 +0000433<td valign=bottom>&nbsp;<br>
434<font color="%s" face="helvetica, arial">&nbsp;<br>%s</font></td
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000435><td align=right valign=bottom
Ka-Ping Yee987ec902001-03-23 13:35:45 +0000436><font color="%s" face="helvetica, arial">%s</font></td></tr></table>
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000437 ''' % (bgcol, fgcol, title, fgcol, extras or '&nbsp;')
438
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000439 def section(self, title, fgcol, bgcol, contents, width=6,
440 prelude='', marginalia=None, gap='&nbsp;'):
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000441 """Format a section with a heading."""
442 if marginalia is None:
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +0000443 marginalia = '<tt>' + '&nbsp;' * width + '</tt>'
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000444 result = '''<p>
Tim Peters59ed4482001-10-31 04:20:26 +0000445<table width="100%%" cellspacing=0 cellpadding=2 border=0 summary="section">
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000446<tr bgcolor="%s">
Tim Peters2306d242001-09-25 03:18:32 +0000447<td colspan=3 valign=bottom>&nbsp;<br>
448<font color="%s" face="helvetica, arial">%s</font></td></tr>
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000449 ''' % (bgcol, fgcol, title)
450 if prelude:
451 result = result + '''
Ka-Ping Yee987ec902001-03-23 13:35:45 +0000452<tr bgcolor="%s"><td rowspan=2>%s</td>
453<td colspan=2>%s</td></tr>
454<tr><td>%s</td>''' % (bgcol, marginalia, prelude, gap)
455 else:
456 result = result + '''
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000457<tr><td bgcolor="%s">%s</td><td>%s</td>''' % (bgcol, marginalia, gap)
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000458
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +0000459 return result + '\n<td width="100%%">%s</td></tr></table>' % contents
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000460
461 def bigsection(self, title, *args):
462 """Format a section with a big heading."""
463 title = '<big><strong>%s</strong></big>' % title
Guido van Rossum68468eb2003-02-27 20:14:51 +0000464 return self.section(title, *args)
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000465
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000466 def preformat(self, text):
467 """Format literal preformatted text."""
Neal Norwitz9d72bb42007-04-17 08:48:32 +0000468 text = self.escape(text.expandtabs())
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +0000469 return replace(text, '\n\n', '\n \n', '\n\n', '\n \n',
470 ' ', '&nbsp;', '\n', '<br>\n')
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000471
472 def multicolumn(self, list, format, cols=4):
473 """Format a list of items into a multi-column list."""
474 result = ''
475 rows = (len(list)+cols-1)/cols
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000476 for col in range(cols):
477 result = result + '<td width="%d%%" valign=top>' % (100/cols)
478 for i in range(rows*col, rows*col+rows):
479 if i < len(list):
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000480 result = result + format(list[i]) + '<br>\n'
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000481 result = result + '</td>'
Tim Peters59ed4482001-10-31 04:20:26 +0000482 return '<table width="100%%" summary="list"><tr>%s</tr></table>' % result
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000483
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000484 def grey(self, text): return '<font color="#909090">%s</font>' % text
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000485
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000486 def namelink(self, name, *dicts):
487 """Make a link for an identifier, given name-to-URL mappings."""
488 for dict in dicts:
Raymond Hettinger54f02222002-06-01 14:18:47 +0000489 if name in dict:
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000490 return '<a href="%s">%s</a>' % (dict[name], name)
491 return name
492
Ka-Ping Yeeb7a48302001-04-12 20:39:14 +0000493 def classlink(self, object, modname):
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000494 """Make a link for a class."""
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000495 name, module = object.__name__, sys.modules.get(object.__module__)
496 if hasattr(module, name) and getattr(module, name) is object:
Ka-Ping Yeeb7a48302001-04-12 20:39:14 +0000497 return '<a href="%s.html#%s">%s</a>' % (
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000498 module.__name__, name, classname(object, modname))
499 return classname(object, modname)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000500
501 def modulelink(self, object):
502 """Make a link for a module."""
503 return '<a href="%s.html">%s</a>' % (object.__name__, object.__name__)
504
505 def modpkglink(self, (name, path, ispackage, shadowed)):
506 """Make a link for a module or package to display in an index."""
507 if shadowed:
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000508 return self.grey(name)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000509 if path:
510 url = '%s.%s.html' % (path, name)
511 else:
512 url = '%s.html' % name
513 if ispackage:
514 text = '<strong>%s</strong>&nbsp;(package)' % name
515 else:
516 text = name
517 return '<a href="%s">%s</a>' % (url, text)
518
519 def markup(self, text, escape=None, funcs={}, classes={}, methods={}):
520 """Mark up some plain text, given a context of symbols to look for.
521 Each context dictionary maps object names to anchor names."""
522 escape = escape or self.escape
523 results = []
524 here = 0
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000525 pattern = re.compile(r'\b((http|ftp)://\S+[\w/]|'
526 r'RFC[- ]?(\d+)|'
Ka-Ping Yeef78a81b2001-03-27 08:13:42 +0000527 r'PEP[- ]?(\d+)|'
Neil Schemenauerd69711c2002-03-24 23:02:07 +0000528 r'(self\.)?(\w+))')
Guido van Rossum8ca162f2002-04-07 06:36:23 +0000529 while True:
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000530 match = pattern.search(text, here)
531 if not match: break
532 start, end = match.span()
533 results.append(escape(text[here:start]))
534
Ka-Ping Yeef78a81b2001-03-27 08:13:42 +0000535 all, scheme, rfc, pep, selfdot, name = match.groups()
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000536 if scheme:
Neil Schemenauercddc1a02002-03-24 23:11:21 +0000537 url = escape(all).replace('"', '&quot;')
538 results.append('<a href="%s">%s</a>' % (url, url))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000539 elif rfc:
Ka-Ping Yeef78a81b2001-03-27 08:13:42 +0000540 url = 'http://www.rfc-editor.org/rfc/rfc%d.txt' % int(rfc)
541 results.append('<a href="%s">%s</a>' % (url, escape(all)))
542 elif pep:
543 url = 'http://www.python.org/peps/pep-%04d.html' % int(pep)
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000544 results.append('<a href="%s">%s</a>' % (url, escape(all)))
545 elif text[end:end+1] == '(':
546 results.append(self.namelink(name, methods, funcs, classes))
547 elif selfdot:
548 results.append('self.<strong>%s</strong>' % name)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000549 else:
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000550 results.append(self.namelink(name, classes))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000551 here = end
552 results.append(escape(text[here:]))
Neal Norwitz9d72bb42007-04-17 08:48:32 +0000553 return ''.join(results)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000554
555 # ---------------------------------------------- type-specific routines
556
Ka-Ping Yeeb7a48302001-04-12 20:39:14 +0000557 def formattree(self, tree, modname, parent=None):
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000558 """Produce HTML for a class tree as given by inspect.getclasstree()."""
559 result = ''
560 for entry in tree:
561 if type(entry) is type(()):
562 c, bases = entry
Tim Peters2306d242001-09-25 03:18:32 +0000563 result = result + '<dt><font face="helvetica, arial">'
Ka-Ping Yeeb7a48302001-04-12 20:39:14 +0000564 result = result + self.classlink(c, modname)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000565 if bases and bases != (parent,):
566 parents = []
567 for base in bases:
Ka-Ping Yeeb7a48302001-04-12 20:39:14 +0000568 parents.append(self.classlink(base, modname))
Neal Norwitz9d72bb42007-04-17 08:48:32 +0000569 result = result + '(' + ', '.join(parents) + ')'
Tim Peters2306d242001-09-25 03:18:32 +0000570 result = result + '\n</font></dt>'
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000571 elif type(entry) is type([]):
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000572 result = result + '<dd>\n%s</dd>\n' % self.formattree(
Ka-Ping Yeeb7a48302001-04-12 20:39:14 +0000573 entry, modname, c)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000574 return '<dl>\n%s</dl>\n' % result
575
Tim Peters8dd7ade2001-10-18 19:56:17 +0000576 def docmodule(self, object, name=None, mod=None, *ignored):
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000577 """Produce HTML documentation for a module object."""
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000578 name = object.__name__ # ignore the passed-in name
Skip Montanaroa5616d22004-06-11 04:46:12 +0000579 try:
580 all = object.__all__
581 except AttributeError:
582 all = None
Neal Norwitz9d72bb42007-04-17 08:48:32 +0000583 parts = name.split('.')
Ka-Ping Yee66efbc72001-03-01 13:55:20 +0000584 links = []
585 for i in range(len(parts)-1):
586 links.append(
587 '<a href="%s.html"><font color="#ffffff">%s</font></a>' %
Neal Norwitz9d72bb42007-04-17 08:48:32 +0000588 ('.'.join(parts[:i+1]), parts[i]))
589 linkedname = '.'.join(links + parts[-1:])
Ka-Ping Yee66efbc72001-03-01 13:55:20 +0000590 head = '<big><big><strong>%s</strong></big></big>' % linkedname
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000591 try:
Ka-Ping Yee239432a2001-03-02 02:45:08 +0000592 path = inspect.getabsfile(object)
Ka-Ping Yee6191a232001-04-13 15:00:27 +0000593 url = path
594 if sys.platform == 'win32':
595 import nturl2path
596 url = nturl2path.pathname2url(path)
597 filelink = '<a href="file:%s">%s</a>' % (url, path)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000598 except TypeError:
599 filelink = '(built-in)'
Ka-Ping Yee6f3f9a42001-02-27 22:42:36 +0000600 info = []
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000601 if hasattr(object, '__version__'):
Ka-Ping Yee6f3f9a42001-02-27 22:42:36 +0000602 version = str(object.__version__)
Ka-Ping Yee40c49912001-02-27 22:46:01 +0000603 if version[:11] == '$' + 'Revision: ' and version[-1:] == '$':
Neal Norwitz9d72bb42007-04-17 08:48:32 +0000604 version = version[11:-1].strip()
Ka-Ping Yee1d384632001-03-01 00:24:32 +0000605 info.append('version %s' % self.escape(version))
Ka-Ping Yee6f3f9a42001-02-27 22:42:36 +0000606 if hasattr(object, '__date__'):
607 info.append(self.escape(str(object.__date__)))
608 if info:
Neal Norwitz9d72bb42007-04-17 08:48:32 +0000609 head = head + ' (%s)' % ', '.join(info)
Skip Montanaro4997a692003-09-10 16:47:51 +0000610 docloc = self.getdocloc(object)
611 if docloc is not None:
612 docloc = '<br><a href="%(docloc)s">Module Docs</a>' % locals()
613 else:
614 docloc = ''
Ka-Ping Yee66efbc72001-03-01 13:55:20 +0000615 result = self.heading(
Skip Montanaro4997a692003-09-10 16:47:51 +0000616 head, '#ffffff', '#7799ee',
617 '<a href=".">index</a><br>' + filelink + docloc)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000618
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000619 modules = inspect.getmembers(object, inspect.ismodule)
620
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000621 classes, cdict = [], {}
622 for key, value in inspect.getmembers(object, inspect.isclass):
Johannes Gijsbers4c11f602004-08-30 14:13:04 +0000623 # if __all__ exists, believe it. Otherwise use old heuristic.
624 if (all is not None or
625 (inspect.getmodule(value) or object) is object):
Skip Montanaroa5616d22004-06-11 04:46:12 +0000626 if visiblename(key, all):
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000627 classes.append((key, value))
628 cdict[key] = cdict[value] = '#' + key
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000629 for key, value in classes:
630 for base in value.__bases__:
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000631 key, modname = base.__name__, base.__module__
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000632 module = sys.modules.get(modname)
633 if modname != name and module and hasattr(module, key):
634 if getattr(module, key) is base:
Raymond Hettinger54f02222002-06-01 14:18:47 +0000635 if not key in cdict:
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000636 cdict[key] = cdict[base] = modname + '.html#' + key
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000637 funcs, fdict = [], {}
638 for key, value in inspect.getmembers(object, inspect.isroutine):
Johannes Gijsbers4c11f602004-08-30 14:13:04 +0000639 # if __all__ exists, believe it. Otherwise use old heuristic.
640 if (all is not None or
641 inspect.isbuiltin(value) or inspect.getmodule(value) is object):
Skip Montanaroa5616d22004-06-11 04:46:12 +0000642 if visiblename(key, all):
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000643 funcs.append((key, value))
644 fdict[key] = '#-' + key
645 if inspect.isfunction(value): fdict[value] = fdict[key]
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000646 data = []
647 for key, value in inspect.getmembers(object, isdata):
Skip Montanaroa5616d22004-06-11 04:46:12 +0000648 if visiblename(key, all):
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000649 data.append((key, value))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000650
651 doc = self.markup(getdoc(object), self.preformat, fdict, cdict)
652 doc = doc and '<tt>%s</tt>' % doc
Tim Peters2306d242001-09-25 03:18:32 +0000653 result = result + '<p>%s</p>\n' % doc
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000654
655 if hasattr(object, '__path__'):
656 modpkgs = []
Thomas Wouters49fd7fa2006-04-21 10:40:58 +0000657 for importer, modname, ispkg in pkgutil.iter_modules(object.__path__):
658 modpkgs.append((modname, name, ispkg, 0))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000659 modpkgs.sort()
660 contents = self.multicolumn(modpkgs, self.modpkglink)
661 result = result + self.bigsection(
662 'Package Contents', '#ffffff', '#aa55cc', contents)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000663 elif modules:
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000664 contents = self.multicolumn(
665 modules, lambda (key, value), s=self: s.modulelink(value))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000666 result = result + self.bigsection(
667 'Modules', '#fffff', '#aa55cc', contents)
668
669 if classes:
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000670 classlist = map(lambda (key, value): value, classes)
Ka-Ping Yeeb7a48302001-04-12 20:39:14 +0000671 contents = [
672 self.formattree(inspect.getclasstree(classlist, 1), name)]
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000673 for key, value in classes:
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +0000674 contents.append(self.document(value, key, name, fdict, cdict))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000675 result = result + self.bigsection(
Neal Norwitz9d72bb42007-04-17 08:48:32 +0000676 'Classes', '#ffffff', '#ee77aa', ' '.join(contents))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000677 if funcs:
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000678 contents = []
679 for key, value in funcs:
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +0000680 contents.append(self.document(value, key, name, fdict, cdict))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000681 result = result + self.bigsection(
Neal Norwitz9d72bb42007-04-17 08:48:32 +0000682 'Functions', '#ffffff', '#eeaa77', ' '.join(contents))
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000683 if data:
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000684 contents = []
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000685 for key, value in data:
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000686 contents.append(self.document(value, key))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000687 result = result + self.bigsection(
Neal Norwitz9d72bb42007-04-17 08:48:32 +0000688 'Data', '#ffffff', '#55aa55', '<br>\n'.join(contents))
Ka-Ping Yee6f3f9a42001-02-27 22:42:36 +0000689 if hasattr(object, '__author__'):
690 contents = self.markup(str(object.__author__), self.preformat)
691 result = result + self.bigsection(
692 'Author', '#ffffff', '#7799ee', contents)
Ka-Ping Yee6f3f9a42001-02-27 22:42:36 +0000693 if hasattr(object, '__credits__'):
694 contents = self.markup(str(object.__credits__), self.preformat)
695 result = result + self.bigsection(
696 'Credits', '#ffffff', '#7799ee', contents)
697
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000698 return result
699
Tim Peters8dd7ade2001-10-18 19:56:17 +0000700 def docclass(self, object, name=None, mod=None, funcs={}, classes={},
701 *ignored):
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000702 """Produce HTML documentation for a class object."""
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000703 realname = object.__name__
704 name = name or realname
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000705 bases = object.__bases__
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000706
Tim Petersb47879b2001-09-24 04:47:19 +0000707 contents = []
708 push = contents.append
709
Tim Petersfa26f7c2001-09-24 08:05:11 +0000710 # Cute little class to pump out a horizontal rule between sections.
711 class HorizontalRule:
712 def __init__(self):
713 self.needone = 0
714 def maybe(self):
715 if self.needone:
716 push('<hr>\n')
717 self.needone = 1
718 hr = HorizontalRule()
719
Tim Petersc86f6ca2001-09-26 21:31:51 +0000720 # List the mro, if non-trivial.
Raymond Hettinger756b3f32004-01-29 06:37:52 +0000721 mro = deque(inspect.getmro(object))
Tim Petersc86f6ca2001-09-26 21:31:51 +0000722 if len(mro) > 2:
723 hr.maybe()
724 push('<dl><dt>Method resolution order:</dt>\n')
725 for base in mro:
726 push('<dd>%s</dd>\n' % self.classlink(base,
727 object.__module__))
728 push('</dl>\n')
729
Tim Petersb47879b2001-09-24 04:47:19 +0000730 def spill(msg, attrs, predicate):
Tim Petersfa26f7c2001-09-24 08:05:11 +0000731 ok, attrs = _split_list(attrs, predicate)
Tim Petersb47879b2001-09-24 04:47:19 +0000732 if ok:
Tim Petersfa26f7c2001-09-24 08:05:11 +0000733 hr.maybe()
Tim Petersb47879b2001-09-24 04:47:19 +0000734 push(msg)
735 for name, kind, homecls, value in ok:
736 push(self.document(getattr(object, name), name, mod,
737 funcs, classes, mdict, object))
738 push('\n')
739 return attrs
740
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +0000741 def spilldescriptors(msg, attrs, predicate):
Tim Petersfa26f7c2001-09-24 08:05:11 +0000742 ok, attrs = _split_list(attrs, predicate)
Tim Petersb47879b2001-09-24 04:47:19 +0000743 if ok:
Tim Petersfa26f7c2001-09-24 08:05:11 +0000744 hr.maybe()
Tim Petersb47879b2001-09-24 04:47:19 +0000745 push(msg)
746 for name, kind, homecls, value in ok:
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +0000747 push(self._docdescriptor(name, value, mod))
Tim Petersb47879b2001-09-24 04:47:19 +0000748 return attrs
749
Tim Petersfa26f7c2001-09-24 08:05:11 +0000750 def spilldata(msg, attrs, predicate):
751 ok, attrs = _split_list(attrs, predicate)
Tim Petersb47879b2001-09-24 04:47:19 +0000752 if ok:
Tim Petersfa26f7c2001-09-24 08:05:11 +0000753 hr.maybe()
Tim Petersb47879b2001-09-24 04:47:19 +0000754 push(msg)
755 for name, kind, homecls, value in ok:
756 base = self.docother(getattr(object, name), name, mod)
Martin v. Löwise59e2ba2003-05-03 09:09:02 +0000757 if callable(value) or inspect.isdatadescriptor(value):
Guido van Rossum5e355b22002-05-21 20:56:15 +0000758 doc = getattr(value, "__doc__", None)
759 else:
760 doc = None
Tim Petersb47879b2001-09-24 04:47:19 +0000761 if doc is None:
762 push('<dl><dt>%s</dl>\n' % base)
763 else:
764 doc = self.markup(getdoc(value), self.preformat,
765 funcs, classes, mdict)
Tim Peters2306d242001-09-25 03:18:32 +0000766 doc = '<dd><tt>%s</tt>' % doc
Tim Petersb47879b2001-09-24 04:47:19 +0000767 push('<dl><dt>%s%s</dl>\n' % (base, doc))
768 push('\n')
769 return attrs
770
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000771 attrs = filter(lambda (name, kind, cls, value): visiblename(name),
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +0000772 classify_class_attrs(object))
Tim Petersb47879b2001-09-24 04:47:19 +0000773 mdict = {}
774 for key, kind, homecls, value in attrs:
775 mdict[key] = anchor = '#' + name + '-' + key
776 value = getattr(object, key)
777 try:
778 # The value may not be hashable (e.g., a data attr with
779 # a dict or list value).
780 mdict[value] = anchor
781 except TypeError:
782 pass
783
Tim Petersfa26f7c2001-09-24 08:05:11 +0000784 while attrs:
Tim Peters351e3622001-09-27 03:29:51 +0000785 if mro:
Raymond Hettinger756b3f32004-01-29 06:37:52 +0000786 thisclass = mro.popleft()
Tim Peters351e3622001-09-27 03:29:51 +0000787 else:
788 thisclass = attrs[0][2]
Tim Petersfa26f7c2001-09-24 08:05:11 +0000789 attrs, inherited = _split_list(attrs, lambda t: t[2] is thisclass)
790
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000791 if thisclass is __builtin__.object:
792 attrs = inherited
793 continue
794 elif thisclass is object:
795 tag = 'defined here'
Tim Petersb47879b2001-09-24 04:47:19 +0000796 else:
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000797 tag = 'inherited from %s' % self.classlink(thisclass,
798 object.__module__)
Tim Petersb47879b2001-09-24 04:47:19 +0000799 tag += ':<br>\n'
800
801 # Sort attrs by name.
Thomas Wouters49fd7fa2006-04-21 10:40:58 +0000802 try:
803 attrs.sort(key=lambda t: t[0])
804 except TypeError:
805 attrs.sort(lambda t1, t2: cmp(t1[0], t2[0])) # 2.3 compat
Tim Petersb47879b2001-09-24 04:47:19 +0000806
807 # Pump out the attrs, segregated by kind.
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000808 attrs = spill('Methods %s' % tag, attrs,
Tim Petersb47879b2001-09-24 04:47:19 +0000809 lambda t: t[1] == 'method')
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000810 attrs = spill('Class methods %s' % tag, attrs,
Tim Petersb47879b2001-09-24 04:47:19 +0000811 lambda t: t[1] == 'class method')
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000812 attrs = spill('Static methods %s' % tag, attrs,
Tim Petersb47879b2001-09-24 04:47:19 +0000813 lambda t: t[1] == 'static method')
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +0000814 attrs = spilldescriptors('Data descriptors %s' % tag, attrs,
815 lambda t: t[1] == 'data descriptor')
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000816 attrs = spilldata('Data and other attributes %s' % tag, attrs,
Tim Petersfa26f7c2001-09-24 08:05:11 +0000817 lambda t: t[1] == 'data')
Tim Petersb47879b2001-09-24 04:47:19 +0000818 assert attrs == []
Tim Peters351e3622001-09-27 03:29:51 +0000819 attrs = inherited
Tim Petersb47879b2001-09-24 04:47:19 +0000820
821 contents = ''.join(contents)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000822
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000823 if name == realname:
824 title = '<a name="%s">class <strong>%s</strong></a>' % (
825 name, realname)
826 else:
827 title = '<strong>%s</strong> = <a name="%s">class %s</a>' % (
828 name, name, realname)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000829 if bases:
830 parents = []
831 for base in bases:
Ka-Ping Yeeb7a48302001-04-12 20:39:14 +0000832 parents.append(self.classlink(base, object.__module__))
Neal Norwitz9d72bb42007-04-17 08:48:32 +0000833 title = title + '(%s)' % ', '.join(parents)
Tim Peters2306d242001-09-25 03:18:32 +0000834 doc = self.markup(getdoc(object), self.preformat, funcs, classes, mdict)
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000835 doc = doc and '<tt>%s<br>&nbsp;</tt>' % doc
Tim Petersc86f6ca2001-09-26 21:31:51 +0000836
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000837 return self.section(title, '#000000', '#ffc8d8', contents, 3, doc)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000838
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000839 def formatvalue(self, object):
840 """Format an argument default value as text."""
Tim Peters2306d242001-09-25 03:18:32 +0000841 return self.grey('=' + self.repr(object))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000842
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +0000843 def docroutine(self, object, name=None, mod=None,
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000844 funcs={}, classes={}, methods={}, cl=None):
Ka-Ping Yee66efbc72001-03-01 13:55:20 +0000845 """Produce HTML documentation for a function or method object."""
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000846 realname = object.__name__
847 name = name or realname
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000848 anchor = (cl and cl.__name__ or '') + '-' + name
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000849 note = ''
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000850 skipdocs = 0
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000851 if inspect.ismethod(object):
Ka-Ping Yee6dcfa382001-04-12 20:27:31 +0000852 imclass = object.im_class
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000853 if cl:
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +0000854 if imclass is not cl:
Ka-Ping Yeeb7a48302001-04-12 20:39:14 +0000855 note = ' from ' + self.classlink(imclass, mod)
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000856 else:
Guido van Rossumd8faa362007-04-27 19:54:29 +0000857 if object.im_self is not None:
Ka-Ping Yeeb7a48302001-04-12 20:39:14 +0000858 note = ' method of %s instance' % self.classlink(
859 object.im_self.__class__, mod)
860 else:
861 note = ' unbound %s method' % self.classlink(imclass,mod)
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000862 object = object.im_func
863
864 if name == realname:
865 title = '<a name="%s"><strong>%s</strong></a>' % (anchor, realname)
866 else:
Raymond Hettinger54f02222002-06-01 14:18:47 +0000867 if (cl and realname in cl.__dict__ and
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000868 cl.__dict__[realname] is object):
Ka-Ping Yeee280c062001-03-23 14:05:53 +0000869 reallink = '<a href="#%s">%s</a>' % (
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000870 cl.__name__ + '-' + realname, realname)
871 skipdocs = 1
872 else:
873 reallink = realname
874 title = '<a name="%s"><strong>%s</strong></a> = %s' % (
875 anchor, name, reallink)
Tim Peters4bcfa312001-09-20 06:08:24 +0000876 if inspect.isfunction(object):
Guido van Rossum2e65f892007-02-28 22:03:49 +0000877 args, varargs, kwonlyargs, kwdefaults, varkw, defaults, ann = \
878 inspect.getfullargspec(object)
Ka-Ping Yee66efbc72001-03-01 13:55:20 +0000879 argspec = inspect.formatargspec(
Guido van Rossum2e65f892007-02-28 22:03:49 +0000880 args, varargs, kwonlyargs, kwdefaults, varkw, defaults, ann,
881 formatvalue=self.formatvalue,
882 formatannotation=inspect.formatannotationrelativeto(object))
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000883 if realname == '<lambda>':
Tim Peters59ed4482001-10-31 04:20:26 +0000884 title = '<strong>%s</strong> <em>lambda</em> ' % name
Guido van Rossum2e65f892007-02-28 22:03:49 +0000885 # XXX lambda's won't usually have func_annotations['return']
886 # since the syntax doesn't support but it is possible.
887 # So removing parentheses isn't truly safe.
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000888 argspec = argspec[1:-1] # remove parentheses
Tim Peters4bcfa312001-09-20 06:08:24 +0000889 else:
890 argspec = '(...)'
Ka-Ping Yee66efbc72001-03-01 13:55:20 +0000891
Tim Peters2306d242001-09-25 03:18:32 +0000892 decl = title + argspec + (note and self.grey(
893 '<font face="helvetica, arial">%s</font>' % note))
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000894
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000895 if skipdocs:
Tim Peters2306d242001-09-25 03:18:32 +0000896 return '<dl><dt>%s</dt></dl>\n' % decl
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000897 else:
898 doc = self.markup(
899 getdoc(object), self.preformat, funcs, classes, methods)
Tim Peters2306d242001-09-25 03:18:32 +0000900 doc = doc and '<dd><tt>%s</tt></dd>' % doc
901 return '<dl><dt>%s</dt>%s</dl>\n' % (decl, doc)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000902
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +0000903 def _docdescriptor(self, name, value, mod):
Johannes Gijsbers8de645a2004-11-07 19:16:05 +0000904 results = []
905 push = results.append
906
907 if name:
908 push('<dl><dt><strong>%s</strong></dt>\n' % name)
909 if value.__doc__ is not None:
Ka-Ping Yeebba6acc2005-02-19 22:58:26 +0000910 doc = self.markup(getdoc(value), self.preformat)
Johannes Gijsbers8de645a2004-11-07 19:16:05 +0000911 push('<dd><tt>%s</tt></dd>\n' % doc)
Johannes Gijsbers8de645a2004-11-07 19:16:05 +0000912 push('</dl>\n')
913
914 return ''.join(results)
915
916 def docproperty(self, object, name=None, mod=None, cl=None):
917 """Produce html documentation for a property."""
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +0000918 return self._docdescriptor(name, object, mod)
Johannes Gijsbers8de645a2004-11-07 19:16:05 +0000919
Tim Peters8dd7ade2001-10-18 19:56:17 +0000920 def docother(self, object, name=None, mod=None, *ignored):
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000921 """Produce HTML documentation for a data object."""
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +0000922 lhs = name and '<strong>%s</strong> = ' % name or ''
923 return lhs + self.repr(object)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000924
Thomas Wouters0e3f5912006-08-11 14:57:12 +0000925 def docdata(self, object, name=None, mod=None, cl=None):
926 """Produce html documentation for a data descriptor."""
927 return self._docdescriptor(name, object, mod)
928
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000929 def index(self, dir, shadowed=None):
930 """Generate an HTML index for a directory of modules."""
931 modpkgs = []
932 if shadowed is None: shadowed = {}
Thomas Wouters49fd7fa2006-04-21 10:40:58 +0000933 for importer, name, ispkg in pkgutil.iter_modules([dir]):
934 modpkgs.append((name, '', ispkg, name in shadowed))
935 shadowed[name] = 1
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000936
937 modpkgs.sort()
938 contents = self.multicolumn(modpkgs, self.modpkglink)
939 return self.bigsection(dir, '#ffffff', '#ee77aa', contents)
940
941# -------------------------------------------- text documentation generator
942
943class TextRepr(Repr):
944 """Class for safely making a text representation of a Python object."""
945 def __init__(self):
946 Repr.__init__(self)
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000947 self.maxlist = self.maxtuple = 20
948 self.maxdict = 10
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000949 self.maxstring = self.maxother = 100
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000950
951 def repr1(self, x, level):
Skip Montanaro0fe8fce2003-06-27 15:45:41 +0000952 if hasattr(type(x), '__name__'):
Neal Norwitz9d72bb42007-04-17 08:48:32 +0000953 methodname = 'repr_' + '_'.join(type(x).__name__.split())
Skip Montanaro0fe8fce2003-06-27 15:45:41 +0000954 if hasattr(self, methodname):
955 return getattr(self, methodname)(x, level)
956 return cram(stripid(repr(x)), self.maxother)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000957
Ka-Ping Yeea2fe1032001-03-02 01:19:14 +0000958 def repr_string(self, x, level):
959 test = cram(x, self.maxstring)
960 testrepr = repr(test)
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +0000961 if '\\' in test and '\\' not in replace(testrepr, r'\\', ''):
Ka-Ping Yeea2fe1032001-03-02 01:19:14 +0000962 # Backslashes are only literal in the string and are never
963 # needed to make any special characters, so show a raw string.
964 return 'r' + testrepr[0] + test + testrepr[0]
965 return testrepr
966
Skip Montanarodf708782002-03-07 22:58:02 +0000967 repr_str = repr_string
968
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000969 def repr_instance(self, x, level):
970 try:
Ka-Ping Yee1d384632001-03-01 00:24:32 +0000971 return cram(stripid(repr(x)), self.maxstring)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000972 except:
973 return '<%s instance>' % x.__class__.__name__
974
975class TextDoc(Doc):
976 """Formatter class for text documentation."""
977
978 # ------------------------------------------- text formatting utilities
979
980 _repr_instance = TextRepr()
981 repr = _repr_instance.repr
982
983 def bold(self, text):
984 """Format a string in bold by overstriking."""
Neal Norwitz9d72bb42007-04-17 08:48:32 +0000985 return ''.join(map(lambda ch: ch + '\b' + ch, text))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000986
987 def indent(self, text, prefix=' '):
988 """Indent text by prepending a given prefix to each line."""
989 if not text: return ''
Neal Norwitz9d72bb42007-04-17 08:48:32 +0000990 lines = text.split('\n')
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000991 lines = map(lambda line, prefix=prefix: prefix + line, lines)
Neal Norwitz9d72bb42007-04-17 08:48:32 +0000992 if lines: lines[-1] = lines[-1].rstrip()
993 return '\n'.join(lines)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000994
995 def section(self, title, contents):
996 """Format a section with a given heading."""
Neal Norwitz9d72bb42007-04-17 08:48:32 +0000997 clean_contents = self.indent(contents).rstrip()
998 return self.bold(title) + '\n' + clean_contents + '\n\n'
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000999
1000 # ---------------------------------------------- type-specific routines
1001
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001002 def formattree(self, tree, modname, parent=None, prefix=''):
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001003 """Render in text a class tree as returned by inspect.getclasstree()."""
1004 result = ''
1005 for entry in tree:
1006 if type(entry) is type(()):
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001007 c, bases = entry
1008 result = result + prefix + classname(c, modname)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001009 if bases and bases != (parent,):
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001010 parents = map(lambda c, m=modname: classname(c, m), bases)
Neal Norwitz9d72bb42007-04-17 08:48:32 +00001011 result = result + '(%s)' % ', '.join(parents)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001012 result = result + '\n'
1013 elif type(entry) is type([]):
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001014 result = result + self.formattree(
1015 entry, modname, c, prefix + ' ')
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001016 return result
1017
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001018 def docmodule(self, object, name=None, mod=None):
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001019 """Produce text documentation for a given module object."""
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001020 name = object.__name__ # ignore the passed-in name
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +00001021 synop, desc = splitdoc(getdoc(object))
1022 result = self.section('NAME', name + (synop and ' - ' + synop))
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001023
1024 try:
Skip Montanaroa5616d22004-06-11 04:46:12 +00001025 all = object.__all__
1026 except AttributeError:
1027 all = None
1028
1029 try:
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001030 file = inspect.getabsfile(object)
1031 except TypeError:
1032 file = '(built-in)'
Ka-Ping Yee239432a2001-03-02 02:45:08 +00001033 result = result + self.section('FILE', file)
Skip Montanaro4997a692003-09-10 16:47:51 +00001034
1035 docloc = self.getdocloc(object)
1036 if docloc is not None:
1037 result = result + self.section('MODULE DOCS', docloc)
1038
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +00001039 if desc:
1040 result = result + self.section('DESCRIPTION', desc)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001041
1042 classes = []
1043 for key, value in inspect.getmembers(object, inspect.isclass):
Johannes Gijsbers4c11f602004-08-30 14:13:04 +00001044 # if __all__ exists, believe it. Otherwise use old heuristic.
1045 if (all is not None
1046 or (inspect.getmodule(value) or object) is object):
Skip Montanaroa5616d22004-06-11 04:46:12 +00001047 if visiblename(key, all):
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +00001048 classes.append((key, value))
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001049 funcs = []
1050 for key, value in inspect.getmembers(object, inspect.isroutine):
Johannes Gijsbers4c11f602004-08-30 14:13:04 +00001051 # if __all__ exists, believe it. Otherwise use old heuristic.
1052 if (all is not None or
1053 inspect.isbuiltin(value) or inspect.getmodule(value) is object):
Skip Montanaroa5616d22004-06-11 04:46:12 +00001054 if visiblename(key, all):
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +00001055 funcs.append((key, value))
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001056 data = []
1057 for key, value in inspect.getmembers(object, isdata):
Skip Montanaroa5616d22004-06-11 04:46:12 +00001058 if visiblename(key, all):
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001059 data.append((key, value))
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001060
1061 if hasattr(object, '__path__'):
1062 modpkgs = []
Thomas Wouters49fd7fa2006-04-21 10:40:58 +00001063 for importer, modname, ispkg in pkgutil.iter_modules(object.__path__):
1064 if ispkg:
1065 modpkgs.append(modname + ' (package)')
1066 else:
1067 modpkgs.append(modname)
1068
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001069 modpkgs.sort()
1070 result = result + self.section(
Neal Norwitz9d72bb42007-04-17 08:48:32 +00001071 'PACKAGE CONTENTS', '\n'.join(modpkgs))
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001072
1073 if classes:
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001074 classlist = map(lambda (key, value): value, classes)
1075 contents = [self.formattree(
1076 inspect.getclasstree(classlist, 1), name)]
1077 for key, value in classes:
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001078 contents.append(self.document(value, key, name))
Neal Norwitz9d72bb42007-04-17 08:48:32 +00001079 result = result + self.section('CLASSES', '\n'.join(contents))
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001080
1081 if funcs:
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001082 contents = []
1083 for key, value in funcs:
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001084 contents.append(self.document(value, key, name))
Neal Norwitz9d72bb42007-04-17 08:48:32 +00001085 result = result + self.section('FUNCTIONS', '\n'.join(contents))
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001086
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001087 if data:
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001088 contents = []
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001089 for key, value in data:
Georg Brandl8b813db2005-10-01 16:32:31 +00001090 contents.append(self.docother(value, key, name, maxlen=70))
Neal Norwitz9d72bb42007-04-17 08:48:32 +00001091 result = result + self.section('DATA', '\n'.join(contents))
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001092
1093 if hasattr(object, '__version__'):
1094 version = str(object.__version__)
Ka-Ping Yee1d384632001-03-01 00:24:32 +00001095 if version[:11] == '$' + 'Revision: ' and version[-1:] == '$':
Neal Norwitz9d72bb42007-04-17 08:48:32 +00001096 version = version[11:-1].strip()
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001097 result = result + self.section('VERSION', version)
Ka-Ping Yee6f3f9a42001-02-27 22:42:36 +00001098 if hasattr(object, '__date__'):
1099 result = result + self.section('DATE', str(object.__date__))
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001100 if hasattr(object, '__author__'):
Ka-Ping Yee6f3f9a42001-02-27 22:42:36 +00001101 result = result + self.section('AUTHOR', str(object.__author__))
1102 if hasattr(object, '__credits__'):
1103 result = result + self.section('CREDITS', str(object.__credits__))
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001104 return result
1105
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001106 def docclass(self, object, name=None, mod=None):
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001107 """Produce text documentation for a given class object."""
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001108 realname = object.__name__
1109 name = name or realname
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001110 bases = object.__bases__
1111
Tim Petersc86f6ca2001-09-26 21:31:51 +00001112 def makename(c, m=object.__module__):
1113 return classname(c, m)
1114
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001115 if name == realname:
1116 title = 'class ' + self.bold(realname)
1117 else:
1118 title = self.bold(name) + ' = class ' + realname
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001119 if bases:
Ka-Ping Yee3bda8792001-03-23 13:17:50 +00001120 parents = map(makename, bases)
Neal Norwitz9d72bb42007-04-17 08:48:32 +00001121 title = title + '(%s)' % ', '.join(parents)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001122
1123 doc = getdoc(object)
Tim Peters28355492001-09-23 21:29:55 +00001124 contents = doc and [doc + '\n'] or []
1125 push = contents.append
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001126
Tim Petersc86f6ca2001-09-26 21:31:51 +00001127 # List the mro, if non-trivial.
Raymond Hettinger756b3f32004-01-29 06:37:52 +00001128 mro = deque(inspect.getmro(object))
Tim Petersc86f6ca2001-09-26 21:31:51 +00001129 if len(mro) > 2:
1130 push("Method resolution order:")
1131 for base in mro:
1132 push(' ' + makename(base))
1133 push('')
1134
Tim Petersf4aad8e2001-09-24 22:40:47 +00001135 # Cute little class to pump out a horizontal rule between sections.
1136 class HorizontalRule:
1137 def __init__(self):
1138 self.needone = 0
1139 def maybe(self):
1140 if self.needone:
1141 push('-' * 70)
1142 self.needone = 1
1143 hr = HorizontalRule()
1144
Tim Peters28355492001-09-23 21:29:55 +00001145 def spill(msg, attrs, predicate):
Tim Petersfa26f7c2001-09-24 08:05:11 +00001146 ok, attrs = _split_list(attrs, predicate)
Tim Peters28355492001-09-23 21:29:55 +00001147 if ok:
Tim Petersf4aad8e2001-09-24 22:40:47 +00001148 hr.maybe()
Tim Peters28355492001-09-23 21:29:55 +00001149 push(msg)
1150 for name, kind, homecls, value in ok:
1151 push(self.document(getattr(object, name),
1152 name, mod, object))
1153 return attrs
1154
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +00001155 def spilldescriptors(msg, attrs, predicate):
Tim Petersfa26f7c2001-09-24 08:05:11 +00001156 ok, attrs = _split_list(attrs, predicate)
Tim Peters28355492001-09-23 21:29:55 +00001157 if ok:
Tim Petersf4aad8e2001-09-24 22:40:47 +00001158 hr.maybe()
Tim Peters28355492001-09-23 21:29:55 +00001159 push(msg)
1160 for name, kind, homecls, value in ok:
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +00001161 push(self._docdescriptor(name, value, mod))
Tim Peters28355492001-09-23 21:29:55 +00001162 return attrs
Tim Petersb47879b2001-09-24 04:47:19 +00001163
Tim Petersfa26f7c2001-09-24 08:05:11 +00001164 def spilldata(msg, attrs, predicate):
1165 ok, attrs = _split_list(attrs, predicate)
Tim Peters28355492001-09-23 21:29:55 +00001166 if ok:
Tim Petersf4aad8e2001-09-24 22:40:47 +00001167 hr.maybe()
Tim Peters28355492001-09-23 21:29:55 +00001168 push(msg)
1169 for name, kind, homecls, value in ok:
Martin v. Löwise59e2ba2003-05-03 09:09:02 +00001170 if callable(value) or inspect.isdatadescriptor(value):
Ka-Ping Yeebba6acc2005-02-19 22:58:26 +00001171 doc = getdoc(value)
Guido van Rossum5e355b22002-05-21 20:56:15 +00001172 else:
1173 doc = None
Tim Peters28355492001-09-23 21:29:55 +00001174 push(self.docother(getattr(object, name),
Georg Brandl8b813db2005-10-01 16:32:31 +00001175 name, mod, maxlen=70, doc=doc) + '\n')
Tim Peters28355492001-09-23 21:29:55 +00001176 return attrs
1177
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +00001178 attrs = filter(lambda (name, kind, cls, value): visiblename(name),
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +00001179 classify_class_attrs(object))
Tim Petersfa26f7c2001-09-24 08:05:11 +00001180 while attrs:
Tim Peters351e3622001-09-27 03:29:51 +00001181 if mro:
Raymond Hettinger756b3f32004-01-29 06:37:52 +00001182 thisclass = mro.popleft()
Tim Peters351e3622001-09-27 03:29:51 +00001183 else:
1184 thisclass = attrs[0][2]
Tim Petersfa26f7c2001-09-24 08:05:11 +00001185 attrs, inherited = _split_list(attrs, lambda t: t[2] is thisclass)
1186
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +00001187 if thisclass is __builtin__.object:
1188 attrs = inherited
1189 continue
1190 elif thisclass is object:
Tim Peters28355492001-09-23 21:29:55 +00001191 tag = "defined here"
1192 else:
Tim Petersfa26f7c2001-09-24 08:05:11 +00001193 tag = "inherited from %s" % classname(thisclass,
1194 object.__module__)
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +00001195 filter(lambda t: not t[0].startswith('_'), attrs)
Tim Peters28355492001-09-23 21:29:55 +00001196
1197 # Sort attrs by name.
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +00001198 attrs.sort()
Tim Peters28355492001-09-23 21:29:55 +00001199
1200 # Pump out the attrs, segregated by kind.
Tim Petersf4aad8e2001-09-24 22:40:47 +00001201 attrs = spill("Methods %s:\n" % tag, attrs,
Tim Peters28355492001-09-23 21:29:55 +00001202 lambda t: t[1] == 'method')
Tim Petersf4aad8e2001-09-24 22:40:47 +00001203 attrs = spill("Class methods %s:\n" % tag, attrs,
Tim Peters28355492001-09-23 21:29:55 +00001204 lambda t: t[1] == 'class method')
Tim Petersf4aad8e2001-09-24 22:40:47 +00001205 attrs = spill("Static methods %s:\n" % tag, attrs,
Tim Peters28355492001-09-23 21:29:55 +00001206 lambda t: t[1] == 'static method')
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +00001207 attrs = spilldescriptors("Data descriptors %s:\n" % tag, attrs,
1208 lambda t: t[1] == 'data descriptor')
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +00001209 attrs = spilldata("Data and other attributes %s:\n" % tag, attrs,
1210 lambda t: t[1] == 'data')
Tim Peters28355492001-09-23 21:29:55 +00001211 assert attrs == []
Tim Peters351e3622001-09-27 03:29:51 +00001212 attrs = inherited
Tim Peters28355492001-09-23 21:29:55 +00001213
1214 contents = '\n'.join(contents)
1215 if not contents:
1216 return title + '\n'
Neal Norwitz9d72bb42007-04-17 08:48:32 +00001217 return title + '\n' + self.indent(contents.rstrip(), ' | ') + '\n'
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001218
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001219 def formatvalue(self, object):
1220 """Format an argument default value as text."""
1221 return '=' + self.repr(object)
1222
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001223 def docroutine(self, object, name=None, mod=None, cl=None):
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001224 """Produce text documentation for a function or method object."""
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001225 realname = object.__name__
1226 name = name or realname
1227 note = ''
Ka-Ping Yee3bda8792001-03-23 13:17:50 +00001228 skipdocs = 0
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001229 if inspect.ismethod(object):
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001230 imclass = object.im_class
Ka-Ping Yee3bda8792001-03-23 13:17:50 +00001231 if cl:
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001232 if imclass is not cl:
1233 note = ' from ' + classname(imclass, mod)
Ka-Ping Yee3bda8792001-03-23 13:17:50 +00001234 else:
Guido van Rossumd8faa362007-04-27 19:54:29 +00001235 if object.im_self is not None:
Ka-Ping Yeeb7a48302001-04-12 20:39:14 +00001236 note = ' method of %s instance' % classname(
1237 object.im_self.__class__, mod)
1238 else:
1239 note = ' unbound %s method' % classname(imclass,mod)
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001240 object = object.im_func
1241
1242 if name == realname:
1243 title = self.bold(realname)
1244 else:
Raymond Hettinger54f02222002-06-01 14:18:47 +00001245 if (cl and realname in cl.__dict__ and
Ka-Ping Yee3bda8792001-03-23 13:17:50 +00001246 cl.__dict__[realname] is object):
1247 skipdocs = 1
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001248 title = self.bold(name) + ' = ' + realname
Tim Peters4bcfa312001-09-20 06:08:24 +00001249 if inspect.isfunction(object):
Guido van Rossum2e65f892007-02-28 22:03:49 +00001250 args, varargs, varkw, defaults, kwonlyargs, kwdefaults, ann = \
1251 inspect.getfullargspec(object)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001252 argspec = inspect.formatargspec(
Guido van Rossum2e65f892007-02-28 22:03:49 +00001253 args, varargs, varkw, defaults, kwonlyargs, kwdefaults, ann,
1254 formatvalue=self.formatvalue,
1255 formatannotation=inspect.formatannotationrelativeto(object))
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001256 if realname == '<lambda>':
Georg Brandl501dd0d2006-02-17 09:45:40 +00001257 title = self.bold(name) + ' lambda '
Guido van Rossum2e65f892007-02-28 22:03:49 +00001258 # XXX lambda's won't usually have func_annotations['return']
1259 # since the syntax doesn't support but it is possible.
1260 # So removing parentheses isn't truly safe.
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001261 argspec = argspec[1:-1] # remove parentheses
Tim Peters4bcfa312001-09-20 06:08:24 +00001262 else:
1263 argspec = '(...)'
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001264 decl = title + argspec + note
1265
Ka-Ping Yee3bda8792001-03-23 13:17:50 +00001266 if skipdocs:
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001267 return decl + '\n'
Ka-Ping Yee3bda8792001-03-23 13:17:50 +00001268 else:
1269 doc = getdoc(object) or ''
Neal Norwitz9d72bb42007-04-17 08:48:32 +00001270 return decl + '\n' + (doc and self.indent(doc).rstrip() + '\n')
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001271
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +00001272 def _docdescriptor(self, name, value, mod):
Johannes Gijsbers8de645a2004-11-07 19:16:05 +00001273 results = []
1274 push = results.append
1275
1276 if name:
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +00001277 push(self.bold(name))
1278 push('\n')
Johannes Gijsbers8de645a2004-11-07 19:16:05 +00001279 doc = getdoc(value) or ''
1280 if doc:
1281 push(self.indent(doc))
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +00001282 push('\n')
1283 return ''.join(results)
Johannes Gijsbers8de645a2004-11-07 19:16:05 +00001284
1285 def docproperty(self, object, name=None, mod=None, cl=None):
1286 """Produce text documentation for a property."""
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +00001287 return self._docdescriptor(name, object, mod)
Johannes Gijsbers8de645a2004-11-07 19:16:05 +00001288
Thomas Wouters0e3f5912006-08-11 14:57:12 +00001289 def docdata(self, object, name=None, mod=None, cl=None):
1290 """Produce text documentation for a data descriptor."""
1291 return self._docdescriptor(name, object, mod)
1292
Georg Brandl8b813db2005-10-01 16:32:31 +00001293 def docother(self, object, name=None, mod=None, parent=None, maxlen=None, doc=None):
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001294 """Produce text documentation for a data object."""
1295 repr = self.repr(object)
1296 if maxlen:
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001297 line = (name and name + ' = ' or '') + repr
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001298 chop = maxlen - len(line)
1299 if chop < 0: repr = repr[:chop] + '...'
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001300 line = (name and self.bold(name) + ' = ' or '') + repr
Tim Peters28355492001-09-23 21:29:55 +00001301 if doc is not None:
1302 line += '\n' + self.indent(str(doc))
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001303 return line
1304
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001305# --------------------------------------------------------- user interfaces
1306
1307def pager(text):
1308 """The first time this is called, determine what kind of pager to use."""
1309 global pager
1310 pager = getpager()
1311 pager(text)
1312
1313def getpager():
1314 """Decide what method to use for paging through text."""
1315 if type(sys.stdout) is not types.FileType:
1316 return plainpager
1317 if not sys.stdin.isatty() or not sys.stdout.isatty():
1318 return plainpager
Raymond Hettinger54f02222002-06-01 14:18:47 +00001319 if 'PAGER' in os.environ:
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +00001320 if sys.platform == 'win32': # pipes completely broken in Windows
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001321 return lambda text: tempfilepager(plain(text), os.environ['PAGER'])
Raymond Hettingerdbecd932005-02-06 06:57:08 +00001322 elif os.environ.get('TERM') in ('dumb', 'emacs'):
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001323 return lambda text: pipepager(plain(text), os.environ['PAGER'])
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +00001324 else:
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001325 return lambda text: pipepager(text, os.environ['PAGER'])
Ka-Ping Yeea487e4e2005-11-05 04:49:18 +00001326 if os.environ.get('TERM') in ('dumb', 'emacs'):
1327 return plainpager
Andrew MacIntyre54e0eab2002-03-03 03:12:30 +00001328 if sys.platform == 'win32' or sys.platform.startswith('os2'):
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001329 return lambda text: tempfilepager(plain(text), 'more <')
Skip Montanarod404bee2002-09-26 21:44:57 +00001330 if hasattr(os, 'system') and os.system('(less) 2>/dev/null') == 0:
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001331 return lambda text: pipepager(text, 'less')
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001332
1333 import tempfile
Guido van Rossum3b0a3292002-08-09 16:38:32 +00001334 (fd, filename) = tempfile.mkstemp()
1335 os.close(fd)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001336 try:
1337 if hasattr(os, 'system') and os.system('more %s' % filename) == 0:
1338 return lambda text: pipepager(text, 'more')
1339 else:
1340 return ttypager
1341 finally:
1342 os.unlink(filename)
1343
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001344def plain(text):
1345 """Remove boldface formatting from text."""
1346 return re.sub('.\b', '', text)
1347
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001348def pipepager(text, cmd):
1349 """Page through text by feeding it to another program."""
1350 pipe = os.popen(cmd, 'w')
1351 try:
1352 pipe.write(text)
1353 pipe.close()
1354 except IOError:
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001355 pass # Ignore broken pipes caused by quitting the pager program.
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001356
1357def tempfilepager(text, cmd):
1358 """Page through text by invoking a program on a temporary file."""
1359 import tempfile
Tim Peters550e4e52003-02-07 01:53:46 +00001360 filename = tempfile.mktemp()
1361 file = open(filename, 'w')
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001362 file.write(text)
1363 file.close()
1364 try:
Ka-Ping Yeec92cdf72001-03-02 05:54:35 +00001365 os.system(cmd + ' ' + filename)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001366 finally:
1367 os.unlink(filename)
1368
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001369def ttypager(text):
1370 """Page through text on a text terminal."""
Neal Norwitz9d72bb42007-04-17 08:48:32 +00001371 lines = plain(text).split('\n')
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001372 try:
1373 import tty
1374 fd = sys.stdin.fileno()
1375 old = tty.tcgetattr(fd)
1376 tty.setcbreak(fd)
1377 getchar = lambda: sys.stdin.read(1)
Ka-Ping Yee457aab22001-02-27 23:36:29 +00001378 except (ImportError, AttributeError):
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001379 tty = None
1380 getchar = lambda: sys.stdin.readline()[:-1][:1]
1381
1382 try:
1383 r = inc = os.environ.get('LINES', 25) - 1
Neal Norwitz9d72bb42007-04-17 08:48:32 +00001384 sys.stdout.write('\n'.join(lines[:inc]) + '\n')
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001385 while lines[r:]:
1386 sys.stdout.write('-- more --')
1387 sys.stdout.flush()
1388 c = getchar()
1389
Raymond Hettingerdbecd932005-02-06 06:57:08 +00001390 if c in ('q', 'Q'):
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001391 sys.stdout.write('\r \r')
1392 break
Raymond Hettingerdbecd932005-02-06 06:57:08 +00001393 elif c in ('\r', '\n'):
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001394 sys.stdout.write('\r \r' + lines[r] + '\n')
1395 r = r + 1
1396 continue
Raymond Hettingerdbecd932005-02-06 06:57:08 +00001397 if c in ('b', 'B', '\x1b'):
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001398 r = r - inc - inc
1399 if r < 0: r = 0
Neal Norwitz9d72bb42007-04-17 08:48:32 +00001400 sys.stdout.write('\n' + '\n'.join(lines[r:r+inc]) + '\n')
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001401 r = r + inc
1402
1403 finally:
1404 if tty:
1405 tty.tcsetattr(fd, tty.TCSAFLUSH, old)
1406
1407def plainpager(text):
1408 """Simply print unformatted text. This is the ultimate fallback."""
1409 sys.stdout.write(plain(text))
1410
1411def describe(thing):
Ka-Ping Yee3bda8792001-03-23 13:17:50 +00001412 """Produce a short description of the given thing."""
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001413 if inspect.ismodule(thing):
1414 if thing.__name__ in sys.builtin_module_names:
1415 return 'built-in module ' + thing.__name__
1416 if hasattr(thing, '__path__'):
1417 return 'package ' + thing.__name__
1418 else:
1419 return 'module ' + thing.__name__
1420 if inspect.isbuiltin(thing):
1421 return 'built-in function ' + thing.__name__
Thomas Wouters0e3f5912006-08-11 14:57:12 +00001422 if inspect.isgetsetdescriptor(thing):
1423 return 'getset descriptor %s.%s.%s' % (
1424 thing.__objclass__.__module__, thing.__objclass__.__name__,
1425 thing.__name__)
1426 if inspect.ismemberdescriptor(thing):
1427 return 'member descriptor %s.%s.%s' % (
1428 thing.__objclass__.__module__, thing.__objclass__.__name__,
1429 thing.__name__)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001430 if inspect.isclass(thing):
1431 return 'class ' + thing.__name__
1432 if inspect.isfunction(thing):
1433 return 'function ' + thing.__name__
1434 if inspect.ismethod(thing):
1435 return 'method ' + thing.__name__
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001436 return type(thing).__name__
1437
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001438def locate(path, forceload=0):
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001439 """Locate an object by name or dotted path, importing as necessary."""
Neal Norwitz9d72bb42007-04-17 08:48:32 +00001440 parts = [part for part in path.split('.') if part]
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001441 module, n = None, 0
1442 while n < len(parts):
Neal Norwitz9d72bb42007-04-17 08:48:32 +00001443 nextmodule = safeimport('.'.join(parts[:n+1]), forceload)
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001444 if nextmodule: module, n = nextmodule, n + 1
1445 else: break
1446 if module:
1447 object = module
1448 for part in parts[n:]:
1449 try: object = getattr(object, part)
1450 except AttributeError: return None
1451 return object
1452 else:
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001453 if hasattr(__builtin__, path):
1454 return getattr(__builtin__, path)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001455
1456# --------------------------------------- interactive interpreter interface
1457
1458text = TextDoc()
1459html = HTMLDoc()
1460
Ka-Ping Yee45daeb02002-08-11 15:11:33 +00001461def resolve(thing, forceload=0):
1462 """Given an object or a path to an object, get the object and its name."""
1463 if isinstance(thing, str):
1464 object = locate(thing, forceload)
1465 if not object:
1466 raise ImportError, 'no Python documentation found for %r' % thing
1467 return object, thing
1468 else:
1469 return thing, getattr(thing, '__name__', None)
1470
Guido van Rossumd8faa362007-04-27 19:54:29 +00001471def render_doc(thing, title='Python Library Documentation: %s', forceload=0):
1472 """Render text documentation, given an object or a path to an object."""
1473 object, name = resolve(thing, forceload)
1474 desc = describe(object)
1475 module = inspect.getmodule(object)
1476 if name and '.' in name:
1477 desc += ' in ' + name[:name.rfind('.')]
1478 elif module and module is not object:
1479 desc += ' in module ' + module.__name__
1480 elif not (inspect.ismodule(object) or
1481 inspect.isclass(object) or
1482 inspect.isroutine(object) or
1483 inspect.isgetsetdescriptor(object) or
1484 inspect.ismemberdescriptor(object) or
1485 isinstance(object, property)):
1486 # If the passed object is a piece of data or an instance,
1487 # document its available methods instead of its value.
1488 object = type(object)
1489 desc += ' object'
1490 return title % desc + '\n\n' + text.document(object, name)
1491
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001492def doc(thing, title='Python Library Documentation: %s', forceload=0):
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001493 """Display text documentation, given an object or a path to an object."""
Ka-Ping Yee45daeb02002-08-11 15:11:33 +00001494 try:
1495 object, name = resolve(thing, forceload)
1496 desc = describe(object)
1497 module = inspect.getmodule(object)
1498 if name and '.' in name:
1499 desc += ' in ' + name[:name.rfind('.')]
1500 elif module and module is not object:
1501 desc += ' in module ' + module.__name__
Thomas Woutersfc7bb8c2007-01-15 15:49:28 +00001502 elif not (inspect.ismodule(object) or
1503 inspect.isclass(object) or
1504 inspect.isroutine(object) or
1505 inspect.isgetsetdescriptor(object) or
1506 inspect.ismemberdescriptor(object) or
1507 isinstance(object, property)):
Ka-Ping Yeebba6acc2005-02-19 22:58:26 +00001508 # If the passed object is a piece of data or an instance,
1509 # document its available methods instead of its value.
1510 object = type(object)
1511 desc += ' object'
Guido van Rossumd8faa362007-04-27 19:54:29 +00001512 pager(render_doc(thing, title, forceload))
Guido van Rossumb940e112007-01-10 16:19:56 +00001513 except (ImportError, ErrorDuringImport) as value:
Guido van Rossumbe19ed72007-02-09 05:37:30 +00001514 print(value)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001515
Ka-Ping Yee45daeb02002-08-11 15:11:33 +00001516def writedoc(thing, forceload=0):
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001517 """Write HTML documentation to a file in the current directory."""
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001518 try:
Ka-Ping Yee45daeb02002-08-11 15:11:33 +00001519 object, name = resolve(thing, forceload)
1520 page = html.page(describe(object), html.document(object, name))
1521 file = open(name + '.html', 'w')
1522 file.write(page)
1523 file.close()
Guido van Rossumbe19ed72007-02-09 05:37:30 +00001524 print('wrote', name + '.html')
Guido van Rossumb940e112007-01-10 16:19:56 +00001525 except (ImportError, ErrorDuringImport) as value:
Guido van Rossumbe19ed72007-02-09 05:37:30 +00001526 print(value)
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001527
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001528def writedocs(dir, pkgpath='', done=None):
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001529 """Write out HTML documentation for all modules in a directory tree."""
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001530 if done is None: done = {}
Thomas Wouters49fd7fa2006-04-21 10:40:58 +00001531 for importer, modname, ispkg in pkgutil.walk_packages([dir], pkgpath):
1532 writedoc(modname)
1533 return
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001534
Neal Norwitzce96f692006-03-17 06:49:51 +00001535def raw_input(prompt):
1536 sys.stdout.write(prompt)
1537 sys.stdout.flush()
1538 return sys.stdin.readline()
1539
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001540class Helper:
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001541 keywords = {
1542 'and': 'BOOLEAN',
Guido van Rossumd8faa362007-04-27 19:54:29 +00001543 'as': 'with',
Ka-Ping Yee79c009d2001-04-13 10:53:25 +00001544 'assert': ('ref/assert', ''),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001545 'break': ('ref/break', 'while for'),
1546 'class': ('ref/class', 'CLASSES SPECIALMETHODS'),
1547 'continue': ('ref/continue', 'while for'),
1548 'def': ('ref/function', ''),
1549 'del': ('ref/del', 'BASICMETHODS'),
1550 'elif': 'if',
1551 'else': ('ref/if', 'while for'),
1552 'except': 'try',
1553 'exec': ('ref/exec', ''),
1554 'finally': 'try',
1555 'for': ('ref/for', 'break continue while'),
1556 'from': 'import',
1557 'global': ('ref/global', 'NAMESPACES'),
1558 'if': ('ref/if', 'TRUTHVALUE'),
1559 'import': ('ref/import', 'MODULES'),
1560 'in': ('ref/comparisons', 'SEQUENCEMETHODS2'),
1561 'is': 'COMPARISON',
Neal Norwitz742dde42003-03-30 20:31:34 +00001562 'lambda': ('ref/lambdas', 'FUNCTIONS'),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001563 'not': 'BOOLEAN',
1564 'or': 'BOOLEAN',
Neal Norwitz742dde42003-03-30 20:31:34 +00001565 'pass': ('ref/pass', ''),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001566 'print': ('ref/print', ''),
1567 'raise': ('ref/raise', 'EXCEPTIONS'),
Ka-Ping Yee79c009d2001-04-13 10:53:25 +00001568 'return': ('ref/return', 'FUNCTIONS'),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001569 'try': ('ref/try', 'EXCEPTIONS'),
1570 'while': ('ref/while', 'break continue if TRUTHVALUE'),
Guido van Rossumd8faa362007-04-27 19:54:29 +00001571 'with': ('ref/with', 'CONTEXTMANAGERS EXCEPTIONS yield'),
Tim Petersfb05c4e2002-10-30 05:21:00 +00001572 'yield': ('ref/yield', ''),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001573 }
1574
1575 topics = {
1576 'TYPES': ('ref/types', 'STRINGS UNICODE NUMBERS SEQUENCES MAPPINGS FUNCTIONS CLASSES MODULES FILES inspect'),
Ka-Ping Yee79c009d2001-04-13 10:53:25 +00001577 'STRINGS': ('ref/strings', 'str UNICODE SEQUENCES STRINGMETHODS FORMATTING TYPES'),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001578 'STRINGMETHODS': ('lib/string-methods', 'STRINGS FORMATTING'),
1579 'FORMATTING': ('lib/typesseq-strings', 'OPERATORS'),
Neal Norwitz54f871e2003-05-26 13:49:54 +00001580 'UNICODE': ('ref/strings', 'encodings unicode SEQUENCES STRINGMETHODS FORMATTING TYPES'),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001581 'NUMBERS': ('ref/numbers', 'INTEGER FLOAT COMPLEX TYPES'),
1582 'INTEGER': ('ref/integers', 'int range'),
1583 'FLOAT': ('ref/floating', 'float math'),
1584 'COMPLEX': ('ref/imaginary', 'complex cmath'),
Guido van Rossum805365e2007-05-07 22:24:25 +00001585 'SEQUENCES': ('lib/typesseq', 'STRINGMETHODS FORMATTING range LISTS'),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001586 'MAPPINGS': 'DICTIONARIES',
1587 'FUNCTIONS': ('lib/typesfunctions', 'def TYPES'),
1588 'METHODS': ('lib/typesmethods', 'class def CLASSES TYPES'),
1589 'CODEOBJECTS': ('lib/bltin-code-objects', 'compile FUNCTIONS TYPES'),
Ka-Ping Yee79c009d2001-04-13 10:53:25 +00001590 'TYPEOBJECTS': ('lib/bltin-type-objects', 'types TYPES'),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001591 'FRAMEOBJECTS': 'TYPES',
1592 'TRACEBACKS': 'TYPES',
1593 'NONE': ('lib/bltin-null-object', ''),
1594 'ELLIPSIS': ('lib/bltin-ellipsis-object', 'SLICINGS'),
1595 'FILES': ('lib/bltin-file-objects', ''),
1596 'SPECIALATTRIBUTES': ('lib/specialattrs', ''),
1597 'CLASSES': ('ref/types', 'class SPECIALMETHODS PRIVATENAMES'),
1598 'MODULES': ('lib/typesmodules', 'import'),
1599 'PACKAGES': 'import',
Neal Norwitz2eca4402006-08-29 04:40:24 +00001600 'EXPRESSIONS': ('ref/summary', 'lambda or and not in is BOOLEAN COMPARISON BITWISE SHIFTING BINARY FORMATTING POWER UNARY ATTRIBUTES SUBSCRIPTS SLICINGS CALLS TUPLES LISTS DICTIONARIES'),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001601 'OPERATORS': 'EXPRESSIONS',
1602 'PRECEDENCE': 'EXPRESSIONS',
1603 'OBJECTS': ('ref/objects', 'TYPES'),
1604 'SPECIALMETHODS': ('ref/specialnames', 'BASICMETHODS ATTRIBUTEMETHODS CALLABLEMETHODS SEQUENCEMETHODS1 MAPPINGMETHODS SEQUENCEMETHODS2 NUMBERMETHODS CLASSES'),
Ka-Ping Yee79c009d2001-04-13 10:53:25 +00001605 'BASICMETHODS': ('ref/customization', 'cmp hash repr str SPECIALMETHODS'),
1606 'ATTRIBUTEMETHODS': ('ref/attribute-access', 'ATTRIBUTES SPECIALMETHODS'),
1607 'CALLABLEMETHODS': ('ref/callable-types', 'CALLS SPECIALMETHODS'),
1608 'SEQUENCEMETHODS1': ('ref/sequence-types', 'SEQUENCES SEQUENCEMETHODS2 SPECIALMETHODS'),
1609 'SEQUENCEMETHODS2': ('ref/sequence-methods', 'SEQUENCES SEQUENCEMETHODS1 SPECIALMETHODS'),
1610 'MAPPINGMETHODS': ('ref/sequence-types', 'MAPPINGS SPECIALMETHODS'),
1611 'NUMBERMETHODS': ('ref/numeric-types', 'NUMBERS AUGMENTEDASSIGNMENT SPECIALMETHODS'),
Neal Norwitz54f871e2003-05-26 13:49:54 +00001612 'EXECUTION': ('ref/execmodel', 'NAMESPACES DYNAMICFEATURES EXCEPTIONS'),
1613 'NAMESPACES': ('ref/naming', 'global ASSIGNMENT DELETION DYNAMICFEATURES'),
1614 'DYNAMICFEATURES': ('ref/dynamic-features', ''),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001615 'SCOPING': 'NAMESPACES',
1616 'FRAMES': 'NAMESPACES',
1617 'EXCEPTIONS': ('ref/exceptions', 'try except finally raise'),
Neal Norwitz54f871e2003-05-26 13:49:54 +00001618 'COERCIONS': ('ref/coercion-rules','CONVERSIONS'),
1619 'CONVERSIONS': ('ref/conversions', 'COERCIONS'),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001620 'IDENTIFIERS': ('ref/identifiers', 'keywords SPECIALIDENTIFIERS'),
1621 'SPECIALIDENTIFIERS': ('ref/id-classes', ''),
Ka-Ping Yee79c009d2001-04-13 10:53:25 +00001622 'PRIVATENAMES': ('ref/atom-identifiers', ''),
Neal Norwitz2eca4402006-08-29 04:40:24 +00001623 'LITERALS': ('ref/atom-literals', 'STRINGS NUMBERS TUPLELITERALS LISTLITERALS DICTIONARYLITERALS'),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001624 'TUPLES': 'SEQUENCES',
Ka-Ping Yee79c009d2001-04-13 10:53:25 +00001625 'TUPLELITERALS': ('ref/exprlists', 'TUPLES LITERALS'),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001626 'LISTS': ('lib/typesseq-mutable', 'LISTLITERALS'),
Ka-Ping Yee79c009d2001-04-13 10:53:25 +00001627 'LISTLITERALS': ('ref/lists', 'LISTS LITERALS'),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001628 'DICTIONARIES': ('lib/typesmapping', 'DICTIONARYLITERALS'),
Ka-Ping Yee79c009d2001-04-13 10:53:25 +00001629 'DICTIONARYLITERALS': ('ref/dict', 'DICTIONARIES LITERALS'),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001630 'ATTRIBUTES': ('ref/attribute-references', 'getattr hasattr setattr ATTRIBUTEMETHODS'),
1631 'SUBSCRIPTS': ('ref/subscriptions', 'SEQUENCEMETHODS1'),
1632 'SLICINGS': ('ref/slicings', 'SEQUENCEMETHODS2'),
1633 'CALLS': ('ref/calls', 'EXPRESSIONS'),
1634 'POWER': ('ref/power', 'EXPRESSIONS'),
1635 'UNARY': ('ref/unary', 'EXPRESSIONS'),
1636 'BINARY': ('ref/binary', 'EXPRESSIONS'),
1637 'SHIFTING': ('ref/shifting', 'EXPRESSIONS'),
1638 'BITWISE': ('ref/bitwise', 'EXPRESSIONS'),
1639 'COMPARISON': ('ref/comparisons', 'EXPRESSIONS BASICMETHODS'),
Neal Norwitzf98159c2003-02-07 20:49:40 +00001640 'BOOLEAN': ('ref/Booleans', 'EXPRESSIONS TRUTHVALUE'),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001641 'ASSERTION': 'assert',
1642 'ASSIGNMENT': ('ref/assignment', 'AUGMENTEDASSIGNMENT'),
Ka-Ping Yee79c009d2001-04-13 10:53:25 +00001643 'AUGMENTEDASSIGNMENT': ('ref/augassign', 'NUMBERMETHODS'),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001644 'DELETION': 'del',
1645 'PRINTING': 'print',
1646 'RETURNING': 'return',
1647 'IMPORTING': 'import',
1648 'CONDITIONAL': 'if',
1649 'LOOPING': ('ref/compound', 'for while break continue'),
1650 'TRUTHVALUE': ('lib/truth', 'if while and or not BASICMETHODS'),
Ka-Ping Yee79c009d2001-04-13 10:53:25 +00001651 'DEBUGGING': ('lib/module-pdb', 'pdb'),
Guido van Rossumd8faa362007-04-27 19:54:29 +00001652 'CONTEXTMANAGERS': ('ref/context-managers', 'with'),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001653 }
1654
1655 def __init__(self, input, output):
1656 self.input = input
1657 self.output = output
1658 self.docdir = None
1659 execdir = os.path.dirname(sys.executable)
1660 homedir = os.environ.get('PYTHONHOME')
Guido van Rossumd8faa362007-04-27 19:54:29 +00001661 join = os.path.join
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001662 for dir in [os.environ.get('PYTHONDOCS'),
1663 homedir and os.path.join(homedir, 'doc'),
Guido van Rossumd8faa362007-04-27 19:54:29 +00001664 join(execdir, 'doc'), # for Windows
1665 join(sys.prefix, 'doc/python-docs-' + sys.version.split()[0]),
1666 join(sys.prefix, 'doc/python-' + sys.version.split()[0]),
1667 join(sys.prefix, 'doc/python-docs-' + sys.version[:3]),
1668 join(sys.prefix, 'doc/python-' + sys.version[:3]),
1669 join(sys.prefix, 'Resources/English.lproj/Documentation')]:
1670 if dir and os.path.isdir(join(dir, 'lib')):
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001671 self.docdir = dir
Guido van Rossumd8faa362007-04-27 19:54:29 +00001672 break
1673 if dir and os.path.isdir(join(dir, 'html', 'lib')):
1674 self.docdir = join(dir, 'html')
1675 break
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001676
Ka-Ping Yee79c009d2001-04-13 10:53:25 +00001677 def __repr__(self):
Ka-Ping Yee9bc576b2001-04-13 13:57:31 +00001678 if inspect.stack()[1][3] == '?':
Ka-Ping Yee79c009d2001-04-13 10:53:25 +00001679 self()
1680 return ''
Ka-Ping Yee9bc576b2001-04-13 13:57:31 +00001681 return '<pydoc.Helper instance>'
Ka-Ping Yee79c009d2001-04-13 10:53:25 +00001682
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001683 def __call__(self, request=None):
1684 if request is not None:
1685 self.help(request)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001686 else:
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001687 self.intro()
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001688 self.interact()
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001689 self.output.write('''
Fred Drakee61967f2001-05-10 18:41:02 +00001690You are now leaving help and returning to the Python interpreter.
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001691If you want to ask for help on a particular object directly from the
1692interpreter, you can type "help(object)". Executing "help('string')"
1693has the same effect as typing a particular string at the help> prompt.
1694''')
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001695
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001696 def interact(self):
1697 self.output.write('\n')
Guido van Rossum8ca162f2002-04-07 06:36:23 +00001698 while True:
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001699 try:
Johannes Gijsberse7691d32004-08-17 13:21:53 +00001700 request = self.getline('help> ')
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001701 if not request: break
Johannes Gijsberse7691d32004-08-17 13:21:53 +00001702 except (KeyboardInterrupt, EOFError):
1703 break
Neal Norwitz9d72bb42007-04-17 08:48:32 +00001704 request = replace(request, '"', '', "'", '').strip()
1705 if request.lower() in ('q', 'quit'): break
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001706 self.help(request)
1707
Johannes Gijsberse7691d32004-08-17 13:21:53 +00001708 def getline(self, prompt):
1709 """Read one line, using raw_input when available."""
1710 if self.input is sys.stdin:
1711 return raw_input(prompt)
1712 else:
1713 self.output.write(prompt)
1714 self.output.flush()
1715 return self.input.readline()
1716
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001717 def help(self, request):
1718 if type(request) is type(''):
1719 if request == 'help': self.intro()
1720 elif request == 'keywords': self.listkeywords()
1721 elif request == 'topics': self.listtopics()
1722 elif request == 'modules': self.listmodules()
1723 elif request[:8] == 'modules ':
Neal Norwitz9d72bb42007-04-17 08:48:32 +00001724 self.listmodules(request.split()[1])
Raymond Hettinger54f02222002-06-01 14:18:47 +00001725 elif request in self.keywords: self.showtopic(request)
1726 elif request in self.topics: self.showtopic(request)
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001727 elif request: doc(request, 'Help on %s:')
1728 elif isinstance(request, Helper): self()
1729 else: doc(request, 'Help on %s:')
1730 self.output.write('\n')
1731
1732 def intro(self):
1733 self.output.write('''
1734Welcome to Python %s! This is the online help utility.
1735
1736If this is your first time using Python, you should definitely check out
1737the tutorial on the Internet at http://www.python.org/doc/tut/.
1738
1739Enter the name of any module, keyword, or topic to get help on writing
1740Python programs and using Python modules. To quit this help utility and
1741return to the interpreter, just type "quit".
1742
1743To get a list of available modules, keywords, or topics, type "modules",
1744"keywords", or "topics". Each module also comes with a one-line summary
1745of what it does; to list the modules whose summaries contain a given word
1746such as "spam", type "modules spam".
1747''' % sys.version[:3])
1748
1749 def list(self, items, columns=4, width=80):
1750 items = items[:]
1751 items.sort()
1752 colw = width / columns
1753 rows = (len(items) + columns - 1) / columns
1754 for row in range(rows):
1755 for col in range(columns):
1756 i = col * rows + row
1757 if i < len(items):
1758 self.output.write(items[i])
1759 if col < columns - 1:
1760 self.output.write(' ' + ' ' * (colw-1 - len(items[i])))
1761 self.output.write('\n')
1762
1763 def listkeywords(self):
1764 self.output.write('''
1765Here is a list of the Python keywords. Enter any keyword to get more help.
1766
1767''')
1768 self.list(self.keywords.keys())
1769
1770 def listtopics(self):
1771 self.output.write('''
1772Here is a list of available topics. Enter any topic name to get more help.
1773
1774''')
1775 self.list(self.topics.keys())
1776
1777 def showtopic(self, topic):
1778 if not self.docdir:
1779 self.output.write('''
1780Sorry, topic and keyword documentation is not available because the Python
1781HTML documentation files could not be found. If you have installed them,
1782please set the environment variable PYTHONDOCS to indicate their location.
Thomas Wouters902d6eb2007-01-09 23:18:33 +00001783
1784On the Microsoft Windows operating system, the files can be built by
1785running "hh -decompile . PythonNN.chm" in the C:\PythonNN\Doc> directory.
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001786''')
1787 return
1788 target = self.topics.get(topic, self.keywords.get(topic))
1789 if not target:
1790 self.output.write('no documentation found for %s\n' % repr(topic))
1791 return
1792 if type(target) is type(''):
1793 return self.showtopic(target)
1794
1795 filename, xrefs = target
1796 filename = self.docdir + '/' + filename + '.html'
1797 try:
1798 file = open(filename)
1799 except:
1800 self.output.write('could not read docs from %s\n' % filename)
1801 return
1802
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001803 divpat = re.compile('<div[^>]*navigat.*?</div.*?>', re.I | re.S)
1804 addrpat = re.compile('<address.*?>.*?</address.*?>', re.I | re.S)
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001805 document = re.sub(addrpat, '', re.sub(divpat, '', file.read()))
1806 file.close()
1807
1808 import htmllib, formatter, StringIO
1809 buffer = StringIO.StringIO()
1810 parser = htmllib.HTMLParser(
1811 formatter.AbstractFormatter(formatter.DumbWriter(buffer)))
1812 parser.start_table = parser.do_p
1813 parser.end_table = lambda parser=parser: parser.do_p({})
1814 parser.start_tr = parser.do_br
1815 parser.start_td = parser.start_th = lambda a, b=buffer: b.write('\t')
1816 parser.feed(document)
1817 buffer = replace(buffer.getvalue(), '\xa0', ' ', '\n', '\n ')
Neal Norwitz9d72bb42007-04-17 08:48:32 +00001818 pager(' ' + buffer.strip() + '\n')
Ka-Ping Yeeda793892001-04-13 11:02:51 +00001819 if xrefs:
1820 buffer = StringIO.StringIO()
1821 formatter.DumbWriter(buffer).send_flowing_data(
Neal Norwitz9d72bb42007-04-17 08:48:32 +00001822 'Related help topics: ' + ', '.join(xrefs.split()) + '\n')
Ka-Ping Yeeda793892001-04-13 11:02:51 +00001823 self.output.write('\n%s\n' % buffer.getvalue())
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001824
1825 def listmodules(self, key=''):
1826 if key:
1827 self.output.write('''
1828Here is a list of matching modules. Enter any module name to get more help.
1829
1830''')
1831 apropos(key)
1832 else:
1833 self.output.write('''
1834Please wait a moment while I gather a list of all available modules...
1835
1836''')
1837 modules = {}
1838 def callback(path, modname, desc, modules=modules):
1839 if modname and modname[-9:] == '.__init__':
1840 modname = modname[:-9] + ' (package)'
Neal Norwitz9d72bb42007-04-17 08:48:32 +00001841 if modname.find('.') < 0:
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001842 modules[modname] = 1
1843 ModuleScanner().run(callback)
1844 self.list(modules.keys())
1845 self.output.write('''
1846Enter any module name to get more help. Or, type "modules spam" to search
1847for modules whose descriptions contain the word "spam".
1848''')
1849
1850help = Helper(sys.stdin, sys.stdout)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001851
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001852class Scanner:
1853 """A generic tree iterator."""
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001854 def __init__(self, roots, children, descendp):
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001855 self.roots = roots[:]
1856 self.state = []
1857 self.children = children
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001858 self.descendp = descendp
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001859
1860 def next(self):
1861 if not self.state:
1862 if not self.roots:
1863 return None
1864 root = self.roots.pop(0)
1865 self.state = [(root, self.children(root))]
1866 node, children = self.state[-1]
1867 if not children:
1868 self.state.pop()
1869 return self.next()
1870 child = children.pop(0)
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001871 if self.descendp(child):
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001872 self.state.append((child, self.children(child)))
1873 return child
1874
Thomas Wouters49fd7fa2006-04-21 10:40:58 +00001875
1876class ModuleScanner:
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001877 """An interruptible scanner that searches module synopses."""
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001878
Ka-Ping Yee66246962001-04-12 11:59:50 +00001879 def run(self, callback, key=None, completer=None):
Neal Norwitz9d72bb42007-04-17 08:48:32 +00001880 if key: key = key.lower()
Guido van Rossum8ca162f2002-04-07 06:36:23 +00001881 self.quit = False
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001882 seen = {}
1883
1884 for modname in sys.builtin_module_names:
Ka-Ping Yee239432a2001-03-02 02:45:08 +00001885 if modname != '__main__':
1886 seen[modname] = 1
Ka-Ping Yee66246962001-04-12 11:59:50 +00001887 if key is None:
1888 callback(None, modname, '')
1889 else:
Neal Norwitz9d72bb42007-04-17 08:48:32 +00001890 name = __import__(modname).__doc__ or ''
1891 desc = name.split('\n')[0]
1892 name = modname + ' - ' + desc
1893 if name.lower().find(key) >= 0:
Ka-Ping Yee66246962001-04-12 11:59:50 +00001894 callback(None, modname, desc)
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001895
Thomas Wouters49fd7fa2006-04-21 10:40:58 +00001896 for importer, modname, ispkg in pkgutil.walk_packages():
1897 if self.quit:
1898 break
1899 if key is None:
1900 callback(None, modname, '')
1901 else:
1902 loader = importer.find_module(modname)
1903 if hasattr(loader,'get_source'):
1904 import StringIO
1905 desc = source_synopsis(
1906 StringIO.StringIO(loader.get_source(modname))
1907 ) or ''
1908 if hasattr(loader,'get_filename'):
1909 path = loader.get_filename(modname)
Ka-Ping Yee66246962001-04-12 11:59:50 +00001910 else:
Thomas Wouters49fd7fa2006-04-21 10:40:58 +00001911 path = None
1912 else:
1913 module = loader.load_module(modname)
1914 desc = (module.__doc__ or '').splitlines()[0]
1915 path = getattr(module,'__file__',None)
Neal Norwitz9d72bb42007-04-17 08:48:32 +00001916 name = modname + ' - ' + desc
1917 if name.lower().find(key) >= 0:
Thomas Wouters49fd7fa2006-04-21 10:40:58 +00001918 callback(path, modname, desc)
1919
1920 if completer:
1921 completer()
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001922
1923def apropos(key):
1924 """Print all the one-line module summaries that contain a substring."""
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001925 def callback(path, modname, desc):
1926 if modname[-9:] == '.__init__':
1927 modname = modname[:-9] + ' (package)'
Guido van Rossumbe19ed72007-02-09 05:37:30 +00001928 print(modname, desc and '- ' + desc)
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +00001929 try: import warnings
1930 except ImportError: pass
1931 else: warnings.filterwarnings('ignore') # ignore problems during import
Ka-Ping Yee66246962001-04-12 11:59:50 +00001932 ModuleScanner().run(callback, key)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001933
1934# --------------------------------------------------- web browser interface
1935
Ka-Ping Yee66246962001-04-12 11:59:50 +00001936def serve(port, callback=None, completer=None):
Ka-Ping Yeefd540692001-04-12 12:54:36 +00001937 import BaseHTTPServer, mimetools, select
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001938
1939 # Patch up mimetools.Message so it doesn't break if rfc822 is reloaded.
1940 class Message(mimetools.Message):
1941 def __init__(self, fp, seekable=1):
1942 Message = self.__class__
1943 Message.__bases__[0].__bases__[0].__init__(self, fp, seekable)
1944 self.encodingheader = self.getheader('content-transfer-encoding')
1945 self.typeheader = self.getheader('content-type')
1946 self.parsetype()
1947 self.parseplist()
1948
1949 class DocHandler(BaseHTTPServer.BaseHTTPRequestHandler):
1950 def send_document(self, title, contents):
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001951 try:
1952 self.send_response(200)
1953 self.send_header('Content-Type', 'text/html')
1954 self.end_headers()
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001955 self.wfile.write(html.page(title, contents))
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001956 except IOError: pass
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001957
1958 def do_GET(self):
1959 path = self.path
1960 if path[-5:] == '.html': path = path[:-5]
1961 if path[:1] == '/': path = path[1:]
1962 if path and path != '.':
1963 try:
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001964 obj = locate(path, forceload=1)
Guido van Rossumb940e112007-01-10 16:19:56 +00001965 except ErrorDuringImport as value:
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001966 self.send_document(path, html.escape(str(value)))
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001967 return
Ka-Ping Yee3bda8792001-03-23 13:17:50 +00001968 if obj:
1969 self.send_document(describe(obj), html.document(obj, path))
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001970 else:
1971 self.send_document(path,
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001972'no Python documentation found for %s' % repr(path))
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001973 else:
1974 heading = html.heading(
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001975'<big><big><strong>Python: Index of Modules</strong></big></big>',
1976'#ffffff', '#7799ee')
Ka-Ping Yee239432a2001-03-02 02:45:08 +00001977 def bltinlink(name):
1978 return '<a href="%s.html">%s</a>' % (name, name)
Ka-Ping Yee3bda8792001-03-23 13:17:50 +00001979 names = filter(lambda x: x != '__main__',
1980 sys.builtin_module_names)
Ka-Ping Yee239432a2001-03-02 02:45:08 +00001981 contents = html.multicolumn(names, bltinlink)
1982 indices = ['<p>' + html.bigsection(
1983 'Built-in Modules', '#ffffff', '#ee77aa', contents)]
1984
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001985 seen = {}
Thomas Wouters49fd7fa2006-04-21 10:40:58 +00001986 for dir in sys.path:
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001987 indices.append(html.index(dir, seen))
Neal Norwitz9d72bb42007-04-17 08:48:32 +00001988 contents = heading + ' '.join(indices) + '''<p align=right>
Tim Peters2306d242001-09-25 03:18:32 +00001989<font color="#909090" face="helvetica, arial"><strong>
1990pydoc</strong> by Ka-Ping Yee &lt;ping@lfw.org&gt;</font>'''
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001991 self.send_document('Index of Modules', contents)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001992
1993 def log_message(self, *args): pass
1994
Ka-Ping Yeefd540692001-04-12 12:54:36 +00001995 class DocServer(BaseHTTPServer.HTTPServer):
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001996 def __init__(self, port, callback):
Ka-Ping Yeedb8ed152001-03-02 05:58:17 +00001997 host = (sys.platform == 'mac') and '127.0.0.1' or 'localhost'
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001998 self.address = ('', port)
Ka-Ping Yeedb8ed152001-03-02 05:58:17 +00001999 self.url = 'http://%s:%d/' % (host, port)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002000 self.callback = callback
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002001 self.base.__init__(self, self.address, self.handler)
2002
2003 def serve_until_quit(self):
2004 import select
Guido van Rossum8ca162f2002-04-07 06:36:23 +00002005 self.quit = False
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002006 while not self.quit:
2007 rd, wr, ex = select.select([self.socket.fileno()], [], [], 1)
2008 if rd: self.handle_request()
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002009
2010 def server_activate(self):
2011 self.base.server_activate(self)
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002012 if self.callback: self.callback(self)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002013
2014 DocServer.base = BaseHTTPServer.HTTPServer
2015 DocServer.handler = DocHandler
2016 DocHandler.MessageClass = Message
2017 try:
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00002018 try:
2019 DocServer(port, callback).serve_until_quit()
2020 except (KeyboardInterrupt, select.error):
2021 pass
2022 finally:
Ka-Ping Yee66246962001-04-12 11:59:50 +00002023 if completer: completer()
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002024
2025# ----------------------------------------------------- graphical interface
2026
2027def gui():
2028 """Graphical interface (starts web server and pops up a control window)."""
2029 class GUI:
2030 def __init__(self, window, port=7464):
2031 self.window = window
2032 self.server = None
2033 self.scanner = None
2034
2035 import Tkinter
2036 self.server_frm = Tkinter.Frame(window)
2037 self.title_lbl = Tkinter.Label(self.server_frm,
2038 text='Starting server...\n ')
2039 self.open_btn = Tkinter.Button(self.server_frm,
2040 text='open browser', command=self.open, state='disabled')
2041 self.quit_btn = Tkinter.Button(self.server_frm,
2042 text='quit serving', command=self.quit, state='disabled')
2043
2044 self.search_frm = Tkinter.Frame(window)
2045 self.search_lbl = Tkinter.Label(self.search_frm, text='Search for')
2046 self.search_ent = Tkinter.Entry(self.search_frm)
2047 self.search_ent.bind('<Return>', self.search)
2048 self.stop_btn = Tkinter.Button(self.search_frm,
2049 text='stop', pady=0, command=self.stop, state='disabled')
2050 if sys.platform == 'win32':
Ka-Ping Yeec92cdf72001-03-02 05:54:35 +00002051 # Trying to hide and show this button crashes under Windows.
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002052 self.stop_btn.pack(side='right')
2053
2054 self.window.title('pydoc')
2055 self.window.protocol('WM_DELETE_WINDOW', self.quit)
2056 self.title_lbl.pack(side='top', fill='x')
2057 self.open_btn.pack(side='left', fill='x', expand=1)
2058 self.quit_btn.pack(side='right', fill='x', expand=1)
2059 self.server_frm.pack(side='top', fill='x')
2060
2061 self.search_lbl.pack(side='left')
2062 self.search_ent.pack(side='right', fill='x', expand=1)
2063 self.search_frm.pack(side='top', fill='x')
2064 self.search_ent.focus_set()
2065
Ka-Ping Yeec92cdf72001-03-02 05:54:35 +00002066 font = ('helvetica', sys.platform == 'win32' and 8 or 10)
Ka-Ping Yee239432a2001-03-02 02:45:08 +00002067 self.result_lst = Tkinter.Listbox(window, font=font, height=6)
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002068 self.result_lst.bind('<Button-1>', self.select)
2069 self.result_lst.bind('<Double-Button-1>', self.goto)
2070 self.result_scr = Tkinter.Scrollbar(window,
2071 orient='vertical', command=self.result_lst.yview)
2072 self.result_lst.config(yscrollcommand=self.result_scr.set)
2073
2074 self.result_frm = Tkinter.Frame(window)
2075 self.goto_btn = Tkinter.Button(self.result_frm,
2076 text='go to selected', command=self.goto)
2077 self.hide_btn = Tkinter.Button(self.result_frm,
2078 text='hide results', command=self.hide)
2079 self.goto_btn.pack(side='left', fill='x', expand=1)
2080 self.hide_btn.pack(side='right', fill='x', expand=1)
2081
2082 self.window.update()
2083 self.minwidth = self.window.winfo_width()
2084 self.minheight = self.window.winfo_height()
2085 self.bigminheight = (self.server_frm.winfo_reqheight() +
2086 self.search_frm.winfo_reqheight() +
2087 self.result_lst.winfo_reqheight() +
2088 self.result_frm.winfo_reqheight())
2089 self.bigwidth, self.bigheight = self.minwidth, self.bigminheight
2090 self.expanded = 0
2091 self.window.wm_geometry('%dx%d' % (self.minwidth, self.minheight))
2092 self.window.wm_minsize(self.minwidth, self.minheight)
Martin v. Löwis5b26abb2002-12-28 09:23:09 +00002093 self.window.tk.willdispatch()
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002094
2095 import threading
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00002096 threading.Thread(
2097 target=serve, args=(port, self.ready, self.quit)).start()
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002098
2099 def ready(self, server):
2100 self.server = server
2101 self.title_lbl.config(
2102 text='Python documentation server at\n' + server.url)
2103 self.open_btn.config(state='normal')
2104 self.quit_btn.config(state='normal')
2105
Ka-Ping Yee239432a2001-03-02 02:45:08 +00002106 def open(self, event=None, url=None):
2107 url = url or self.server.url
2108 try:
2109 import webbrowser
2110 webbrowser.open(url)
2111 except ImportError: # pre-webbrowser.py compatibility
Ka-Ping Yeec92cdf72001-03-02 05:54:35 +00002112 if sys.platform == 'win32':
Ka-Ping Yee239432a2001-03-02 02:45:08 +00002113 os.system('start "%s"' % url)
2114 elif sys.platform == 'mac':
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +00002115 try: import ic
Ka-Ping Yee239432a2001-03-02 02:45:08 +00002116 except ImportError: pass
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +00002117 else: ic.launchurl(url)
Ka-Ping Yee239432a2001-03-02 02:45:08 +00002118 else:
2119 rc = os.system('netscape -remote "openURL(%s)" &' % url)
2120 if rc: os.system('netscape "%s" &' % url)
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002121
2122 def quit(self, event=None):
2123 if self.server:
2124 self.server.quit = 1
2125 self.window.quit()
2126
2127 def search(self, event=None):
2128 key = self.search_ent.get()
2129 self.stop_btn.pack(side='right')
2130 self.stop_btn.config(state='normal')
2131 self.search_lbl.config(text='Searching for "%s"...' % key)
2132 self.search_ent.forget()
2133 self.search_lbl.pack(side='left')
2134 self.result_lst.delete(0, 'end')
2135 self.goto_btn.config(state='disabled')
2136 self.expand()
2137
2138 import threading
2139 if self.scanner:
2140 self.scanner.quit = 1
2141 self.scanner = ModuleScanner()
2142 threading.Thread(target=self.scanner.run,
Ka-Ping Yee6dcfa382001-04-12 20:27:31 +00002143 args=(self.update, key, self.done)).start()
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002144
2145 def update(self, path, modname, desc):
2146 if modname[-9:] == '.__init__':
2147 modname = modname[:-9] + ' (package)'
2148 self.result_lst.insert('end',
2149 modname + ' - ' + (desc or '(no description)'))
2150
2151 def stop(self, event=None):
2152 if self.scanner:
2153 self.scanner.quit = 1
2154 self.scanner = None
2155
2156 def done(self):
2157 self.scanner = None
2158 self.search_lbl.config(text='Search for')
2159 self.search_lbl.pack(side='left')
2160 self.search_ent.pack(side='right', fill='x', expand=1)
2161 if sys.platform != 'win32': self.stop_btn.forget()
2162 self.stop_btn.config(state='disabled')
2163
2164 def select(self, event=None):
2165 self.goto_btn.config(state='normal')
2166
2167 def goto(self, event=None):
2168 selection = self.result_lst.curselection()
2169 if selection:
Neal Norwitz9d72bb42007-04-17 08:48:32 +00002170 modname = self.result_lst.get(selection[0]).split()[0]
Ka-Ping Yee239432a2001-03-02 02:45:08 +00002171 self.open(url=self.server.url + modname + '.html')
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002172
2173 def collapse(self):
2174 if not self.expanded: return
2175 self.result_frm.forget()
2176 self.result_scr.forget()
2177 self.result_lst.forget()
2178 self.bigwidth = self.window.winfo_width()
2179 self.bigheight = self.window.winfo_height()
2180 self.window.wm_geometry('%dx%d' % (self.minwidth, self.minheight))
2181 self.window.wm_minsize(self.minwidth, self.minheight)
2182 self.expanded = 0
2183
2184 def expand(self):
2185 if self.expanded: return
2186 self.result_frm.pack(side='bottom', fill='x')
2187 self.result_scr.pack(side='right', fill='y')
2188 self.result_lst.pack(side='top', fill='both', expand=1)
2189 self.window.wm_geometry('%dx%d' % (self.bigwidth, self.bigheight))
2190 self.window.wm_minsize(self.minwidth, self.bigminheight)
2191 self.expanded = 1
2192
2193 def hide(self, event=None):
2194 self.stop()
2195 self.collapse()
2196
2197 import Tkinter
2198 try:
Martin v. Löwise09bd932004-08-22 16:13:26 +00002199 root = Tkinter.Tk()
2200 # Tk will crash if pythonw.exe has an XP .manifest
2201 # file and the root has is not destroyed explicitly.
2202 # If the problem is ever fixed in Tk, the explicit
2203 # destroy can go.
2204 try:
2205 gui = GUI(root)
2206 root.mainloop()
2207 finally:
2208 root.destroy()
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002209 except KeyboardInterrupt:
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002210 pass
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002211
2212# -------------------------------------------------- command-line interface
2213
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00002214def ispath(x):
Neal Norwitz9d72bb42007-04-17 08:48:32 +00002215 return isinstance(x, str) and x.find(os.sep) >= 0
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00002216
Ka-Ping Yee1d384632001-03-01 00:24:32 +00002217def cli():
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002218 """Command-line interface (looks at sys.argv to decide what to do)."""
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002219 import getopt
Guido van Rossum756aa932007-04-07 03:04:01 +00002220 class BadUsage(Exception): pass
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002221
Ka-Ping Yee3bda8792001-03-23 13:17:50 +00002222 # Scripts don't get the current directory in their path by default.
Ka-Ping Yeef78a81b2001-03-27 08:13:42 +00002223 scriptdir = os.path.dirname(sys.argv[0])
2224 if scriptdir in sys.path:
2225 sys.path.remove(scriptdir)
Ka-Ping Yee3bda8792001-03-23 13:17:50 +00002226 sys.path.insert(0, '.')
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002227
Ka-Ping Yee3bda8792001-03-23 13:17:50 +00002228 try:
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002229 opts, args = getopt.getopt(sys.argv[1:], 'gk:p:w')
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002230 writing = 0
2231
2232 for opt, val in opts:
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002233 if opt == '-g':
2234 gui()
2235 return
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002236 if opt == '-k':
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002237 apropos(val)
2238 return
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002239 if opt == '-p':
2240 try:
2241 port = int(val)
2242 except ValueError:
2243 raise BadUsage
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002244 def ready(server):
Guido van Rossumbe19ed72007-02-09 05:37:30 +00002245 print('pydoc server ready at %s' % server.url)
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00002246 def stopped():
Guido van Rossumbe19ed72007-02-09 05:37:30 +00002247 print('pydoc server stopped')
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00002248 serve(port, ready, stopped)
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002249 return
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002250 if opt == '-w':
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002251 writing = 1
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002252
2253 if not args: raise BadUsage
2254 for arg in args:
Ka-Ping Yee45daeb02002-08-11 15:11:33 +00002255 if ispath(arg) and not os.path.exists(arg):
Guido van Rossumbe19ed72007-02-09 05:37:30 +00002256 print('file %r does not exist' % arg)
Ka-Ping Yee45daeb02002-08-11 15:11:33 +00002257 break
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002258 try:
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00002259 if ispath(arg) and os.path.isfile(arg):
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002260 arg = importfile(arg)
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00002261 if writing:
2262 if ispath(arg) and os.path.isdir(arg):
2263 writedocs(arg)
2264 else:
2265 writedoc(arg)
2266 else:
Martin v. Löwisb8c084e2003-06-14 09:03:46 +00002267 help.help(arg)
Guido van Rossumb940e112007-01-10 16:19:56 +00002268 except ErrorDuringImport as value:
Guido van Rossumbe19ed72007-02-09 05:37:30 +00002269 print(value)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002270
2271 except (getopt.error, BadUsage):
Martin v. Löwisf9b08b82003-10-31 13:05:21 +00002272 cmd = os.path.basename(sys.argv[0])
Guido van Rossumbe19ed72007-02-09 05:37:30 +00002273 print("""pydoc - the Python documentation tool
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002274
2275%s <name> ...
2276 Show text documentation on something. <name> may be the name of a
Martin v. Löwisb8c084e2003-06-14 09:03:46 +00002277 Python keyword, topic, function, module, or package, or a dotted
2278 reference to a class or function within a module or module in a
2279 package. If <name> contains a '%s', it is used as the path to a
2280 Python source file to document. If name is 'keywords', 'topics',
2281 or 'modules', a listing of these things is displayed.
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002282
2283%s -k <keyword>
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002284 Search for a keyword in the synopsis lines of all available modules.
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002285
2286%s -p <port>
2287 Start an HTTP server on the given port on the local machine.
2288
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002289%s -g
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +00002290 Pop up a graphical interface for finding and serving documentation.
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002291
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002292%s -w <name> ...
2293 Write out the HTML documentation for a module to a file in the current
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +00002294 directory. If <name> contains a '%s', it is treated as a filename; if
2295 it names a directory, documentation is written for all the contents.
Guido van Rossumbe19ed72007-02-09 05:37:30 +00002296""" % (cmd, os.sep, cmd, cmd, cmd, cmd, os.sep))
Ka-Ping Yee1d384632001-03-01 00:24:32 +00002297
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002298if __name__ == '__main__': cli()