blob: 7681a3206a62e342487ed2420e44d7a0d8aa36e6 [file] [log] [blame]
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001#!/usr/bin/env python
Georg Brandl681001e2008-06-01 20:33:55 +00002# -*- coding: latin-1 -*-
Ka-Ping Yee1d384632001-03-01 00:24:32 +00003"""Generate Python documentation in HTML or text for interactive use.
Ka-Ping Yeedd175342001-02-27 14:43:46 +00004
Ka-Ping Yeedd175342001-02-27 14:43:46 +00005In the Python interpreter, do "from pydoc import help" to provide online
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00006help. Calling help(thing) on a Python object documents the object.
7
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00008Or, at the shell command line outside of Python:
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00009
Ka-Ping Yee37f7b382001-03-23 00:12:53 +000010Run "pydoc <name>" to show documentation on something. <name> may be
11the name of a function, module, package, or a dotted reference to a
12class or function within a module or module in a package. If the
13argument contains a path segment delimiter (e.g. slash on Unix,
14backslash on Windows) it is treated as the path to a Python source file.
Ka-Ping Yee66efbc72001-03-01 13:55:20 +000015
Ka-Ping Yee37f7b382001-03-23 00:12:53 +000016Run "pydoc -k <keyword>" to search for a keyword in the synopsis lines
17of all available modules.
Ka-Ping Yee66efbc72001-03-01 13:55:20 +000018
Ka-Ping Yee37f7b382001-03-23 00:12:53 +000019Run "pydoc -p <port>" to start an HTTP server on a given port on the
20local machine to generate documentation web pages.
Ka-Ping Yee66efbc72001-03-01 13:55:20 +000021
Ka-Ping Yee37f7b382001-03-23 00:12:53 +000022For platforms without a command line, "pydoc -g" starts the HTTP server
23and also pops up a little window for controlling it.
24
25Run "pydoc -w <name>" to write out the HTML documentation for a module
26to a file named "<name>.html".
Skip Montanaro4997a692003-09-10 16:47:51 +000027
28Module docs for core modules are assumed to be in
29
Georg Brandl0751d1a2008-01-21 17:13:03 +000030 http://docs.python.org/library/
Skip Montanaro4997a692003-09-10 16:47:51 +000031
32This can be overridden by setting the PYTHONDOCS environment variable
33to a different URL or to a local directory containing the Library
34Reference Manual pages.
Ka-Ping Yee66efbc72001-03-01 13:55:20 +000035"""
Ka-Ping Yeedd175342001-02-27 14:43:46 +000036
37__author__ = "Ka-Ping Yee <ping@lfw.org>"
Ka-Ping Yee6f3f9a42001-02-27 22:42:36 +000038__date__ = "26 February 2001"
Jeremy Hylton3e0055f2005-10-20 19:59:25 +000039
Ka-Ping Yee09d7d9a2001-02-27 22:43:48 +000040__version__ = "$Revision$"
Martin v. Löwis6fe8f192004-11-14 10:21:04 +000041__credits__ = """Guido van Rossum, for an excellent programming language.
Ka-Ping Yee5e2b1732001-02-27 23:35:09 +000042Tommy Burnette, the original creator of manpy.
Ka-Ping Yee6f3f9a42001-02-27 22:42:36 +000043Paul Prescod, for all his work on onlinehelp.
44Richard Chamberlain, for the first implementation of textdoc.
Raymond Hettingered2dbe32005-01-01 07:51:01 +000045"""
Ka-Ping Yeedd175342001-02-27 14:43:46 +000046
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +000047# Known bugs that can't be fixed here:
48# - imp.load_module() cannot be prevented from clobbering existing
49# loaded modules, so calling synopsis() on a binary module file
50# changes the contents of any existing module with the same name.
51# - If the __file__ attribute on a module is a relative path and
52# the current directory is changed with os.chdir(), an incorrect
53# path will be displayed.
Ka-Ping Yee66efbc72001-03-01 13:55:20 +000054
Phillip J. Ebyceb30872006-04-18 00:59:55 +000055import sys, imp, os, re, types, inspect, __builtin__, pkgutil
Brett Cannon2ee0e8e2008-05-23 05:03:59 +000056from repr import Repr
Ka-Ping Yee3bda8792001-03-23 13:17:50 +000057from string import expandtabs, find, join, lower, split, strip, rfind, rstrip
Phillip J. Ebyceb30872006-04-18 00:59:55 +000058try:
59 from collections import deque
60except ImportError:
61 # Python 2.3 compatibility
62 class deque(list):
63 def popleft(self):
64 return self.pop(0)
Ka-Ping Yeedd175342001-02-27 14:43:46 +000065
66# --------------------------------------------------------- common routines
67
Ka-Ping Yeedd175342001-02-27 14:43:46 +000068def pathdirs():
69 """Convert sys.path into a list of absolute, existing, unique paths."""
70 dirs = []
Ka-Ping Yee1d384632001-03-01 00:24:32 +000071 normdirs = []
Ka-Ping Yeedd175342001-02-27 14:43:46 +000072 for dir in sys.path:
73 dir = os.path.abspath(dir or '.')
Ka-Ping Yee1d384632001-03-01 00:24:32 +000074 normdir = os.path.normcase(dir)
75 if normdir not in normdirs and os.path.isdir(dir):
Ka-Ping Yeedd175342001-02-27 14:43:46 +000076 dirs.append(dir)
Ka-Ping Yee1d384632001-03-01 00:24:32 +000077 normdirs.append(normdir)
Ka-Ping Yeedd175342001-02-27 14:43:46 +000078 return dirs
79
80def getdoc(object):
81 """Get the doc string or comments for an object."""
Ka-Ping Yee3bda8792001-03-23 13:17:50 +000082 result = inspect.getdoc(object) or inspect.getcomments(object)
Ka-Ping Yee239432a2001-03-02 02:45:08 +000083 return result and re.sub('^ *\n', '', rstrip(result)) or ''
Ka-Ping Yeedd175342001-02-27 14:43:46 +000084
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +000085def splitdoc(doc):
86 """Split a doc string into a synopsis line (if any) and the rest."""
87 lines = split(strip(doc), '\n')
88 if len(lines) == 1:
89 return lines[0], ''
90 elif len(lines) >= 2 and not rstrip(lines[1]):
91 return lines[0], join(lines[2:], '\n')
92 return '', join(lines, '\n')
93
Ka-Ping Yeedd175342001-02-27 14:43:46 +000094def classname(object, modname):
95 """Get a class name and qualify it with a module name if necessary."""
96 name = object.__name__
97 if object.__module__ != modname:
98 name = object.__module__ + '.' + name
99 return name
100
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000101def isdata(object):
Georg Brandl94432422005-07-22 21:52:25 +0000102 """Check if an object is of a type that probably means it's data."""
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000103 return not (inspect.ismodule(object) or inspect.isclass(object) or
104 inspect.isroutine(object) or inspect.isframe(object) or
105 inspect.istraceback(object) or inspect.iscode(object))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000106
107def replace(text, *pairs):
108 """Do a series of global replacements on a string."""
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +0000109 while pairs:
110 text = join(split(text, pairs[0]), pairs[1])
111 pairs = pairs[2:]
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000112 return text
113
114def cram(text, maxlen):
115 """Omit part of a string if needed to make it fit in a maximum length."""
116 if len(text) > maxlen:
Raymond Hettingerfca3bb62002-10-21 04:44:11 +0000117 pre = max(0, (maxlen-3)//2)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000118 post = max(0, maxlen-3-pre)
119 return text[:pre] + '...' + text[len(text)-post:]
120 return text
121
Brett Cannon84601f12004-06-19 01:22:48 +0000122_re_stripid = re.compile(r' at 0x[0-9a-f]{6,16}(>+)$', re.IGNORECASE)
Ka-Ping Yee1d384632001-03-01 00:24:32 +0000123def stripid(text):
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000124 """Remove the hexadecimal id from a Python object representation."""
Brett Cannonc6c1f472004-06-19 01:02:51 +0000125 # The behaviour of %p is implementation-dependent in terms of case.
126 if _re_stripid.search(repr(Exception)):
127 return _re_stripid.sub(r'\1', text)
Ka-Ping Yee1d384632001-03-01 00:24:32 +0000128 return text
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000129
Brett Cannonc6c1f472004-06-19 01:02:51 +0000130def _is_some_method(obj):
131 return inspect.ismethod(obj) or inspect.ismethoddescriptor(obj)
Tim Peters536d2262001-09-20 05:13:38 +0000132
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000133def allmethods(cl):
134 methods = {}
Tim Peters536d2262001-09-20 05:13:38 +0000135 for key, value in inspect.getmembers(cl, _is_some_method):
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000136 methods[key] = 1
137 for base in cl.__bases__:
138 methods.update(allmethods(base)) # all your base are belong to us
139 for key in methods.keys():
140 methods[key] = getattr(cl, key)
141 return methods
142
Tim Petersfa26f7c2001-09-24 08:05:11 +0000143def _split_list(s, predicate):
144 """Split sequence s via predicate, and return pair ([true], [false]).
145
146 The return value is a 2-tuple of lists,
147 ([x for x in s if predicate(x)],
148 [x for x in s if not predicate(x)])
149 """
150
Tim Peters28355492001-09-23 21:29:55 +0000151 yes = []
152 no = []
Tim Petersfa26f7c2001-09-24 08:05:11 +0000153 for x in s:
154 if predicate(x):
155 yes.append(x)
Tim Peters28355492001-09-23 21:29:55 +0000156 else:
Tim Petersfa26f7c2001-09-24 08:05:11 +0000157 no.append(x)
Tim Peters28355492001-09-23 21:29:55 +0000158 return yes, no
159
Skip Montanaroa5616d22004-06-11 04:46:12 +0000160def visiblename(name, all=None):
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000161 """Decide whether to show documentation on a variable."""
162 # Certain special names are redundant.
Nick Coghlanc060b0e2008-07-02 13:09:19 +0000163 _hidden_names = ('__builtins__', '__doc__', '__file__', '__path__',
164 '__module__', '__name__', '__slots__', '__package__')
165 if name in _hidden_names: return 0
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000166 # Private names are hidden, but special names are displayed.
167 if name.startswith('__') and name.endswith('__'): return 1
Skip Montanaroa5616d22004-06-11 04:46:12 +0000168 if all is not None:
169 # only document that which the programmer exported in __all__
170 return name in all
171 else:
172 return not name.startswith('_')
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000173
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +0000174def classify_class_attrs(object):
175 """Wrap inspect.classify_class_attrs, with fixup for data descriptors."""
176 def fixup((name, kind, cls, value)):
177 if inspect.isdatadescriptor(value):
178 kind = 'data descriptor'
179 return name, kind, cls, value
180 return map(fixup, inspect.classify_class_attrs(object))
181
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000182# ----------------------------------------------------- module manipulation
183
184def ispackage(path):
185 """Guess whether a path refers to a package directory."""
186 if os.path.isdir(path):
Raymond Hettingerdbecd932005-02-06 06:57:08 +0000187 for ext in ('.py', '.pyc', '.pyo'):
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000188 if os.path.isfile(os.path.join(path, '__init__' + ext)):
Tim Petersbc0e9102002-04-04 22:55:58 +0000189 return True
190 return False
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000191
Phillip J. Ebyceb30872006-04-18 00:59:55 +0000192def source_synopsis(file):
193 line = file.readline()
194 while line[:1] == '#' or not strip(line):
195 line = file.readline()
196 if not line: break
197 line = strip(line)
198 if line[:4] == 'r"""': line = line[1:]
199 if line[:3] == '"""':
200 line = line[3:]
201 if line[-1:] == '\\': line = line[:-1]
202 while not strip(line):
203 line = file.readline()
204 if not line: break
205 result = strip(split(line, '"""')[0])
206 else: result = None
207 return result
208
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000209def synopsis(filename, cache={}):
210 """Get the one-line summary out of a module file."""
Raymond Hettinger32200ae2002-06-01 19:51:15 +0000211 mtime = os.stat(filename).st_mtime
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000212 lastupdate, result = cache.get(filename, (0, None))
213 if lastupdate < mtime:
214 info = inspect.getmoduleinfo(filename)
Georg Brandl26fd2e12006-03-08 09:34:53 +0000215 try:
216 file = open(filename)
217 except IOError:
218 # module can't be opened, so skip it
219 return None
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000220 if info and 'b' in info[2]: # binary modules have to be imported
221 try: module = imp.load_module('__temp__', file, filename, info[1:])
222 except: return None
Phillip J. Ebyceb30872006-04-18 00:59:55 +0000223 result = (module.__doc__ or '').splitlines()[0]
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000224 del sys.modules['__temp__']
225 else: # text modules can be directly examined
Phillip J. Ebyceb30872006-04-18 00:59:55 +0000226 result = source_synopsis(file)
227 file.close()
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000228 cache[filename] = (mtime, result)
229 return result
230
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000231class ErrorDuringImport(Exception):
232 """Errors that occurred while trying to import something to document it."""
233 def __init__(self, filename, (exc, value, tb)):
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000234 self.filename = filename
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000235 self.exc = exc
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000236 self.value = value
237 self.tb = tb
238
239 def __str__(self):
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000240 exc = self.exc
241 if type(exc) is types.ClassType:
242 exc = exc.__name__
243 return 'problem in %s - %s: %s' % (self.filename, exc, self.value)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000244
245def importfile(path):
246 """Import a Python source file or compiled file given its path."""
247 magic = imp.get_magic()
248 file = open(path, 'r')
249 if file.read(len(magic)) == magic:
250 kind = imp.PY_COMPILED
251 else:
252 kind = imp.PY_SOURCE
253 file.close()
254 filename = os.path.basename(path)
255 name, ext = os.path.splitext(filename)
256 file = open(path, 'r')
257 try:
258 module = imp.load_module(name, file, path, (ext, 'r', kind))
259 except:
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000260 raise ErrorDuringImport(path, sys.exc_info())
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000261 file.close()
262 return module
263
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000264def safeimport(path, forceload=0, cache={}):
265 """Import a module; handle errors; return None if the module isn't found.
266
267 If the module *is* found but an exception occurs, it's wrapped in an
268 ErrorDuringImport exception and reraised. Unlike __import__, if a
269 package path is specified, the module at the end of the path is returned,
270 not the package at the beginning. If the optional 'forceload' argument
271 is 1, we reload the module from disk (unless it's a dynamic extension)."""
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000272 try:
Ka-Ping Yee9a2dcf82005-11-05 05:04:41 +0000273 # If forceload is 1 and the module has been previously loaded from
274 # disk, we always have to reload the module. Checking the file's
275 # mtime isn't good enough (e.g. the module could contain a class
276 # that inherits from another module that has changed).
277 if forceload and path in sys.modules:
278 if path not in sys.builtin_module_names:
279 # Avoid simply calling reload() because it leaves names in
280 # the currently loaded module lying around if they're not
281 # defined in the new source file. Instead, remove the
282 # module from sys.modules and re-import. Also remove any
283 # submodules because they won't appear in the newly loaded
284 # module's namespace if they're already in sys.modules.
285 subs = [m for m in sys.modules if m.startswith(path + '.')]
286 for key in [path] + subs:
287 # Prevent garbage collection.
288 cache[key] = sys.modules[key]
289 del sys.modules[key]
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000290 module = __import__(path)
291 except:
292 # Did the error occur before or after the module was found?
293 (exc, value, tb) = info = sys.exc_info()
Raymond Hettinger54f02222002-06-01 14:18:47 +0000294 if path in sys.modules:
Fred Drakedb390c12005-10-28 14:39:47 +0000295 # An error occurred while executing the imported module.
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000296 raise ErrorDuringImport(sys.modules[path].__file__, info)
297 elif exc is SyntaxError:
298 # A SyntaxError occurred before we could execute the module.
299 raise ErrorDuringImport(value.filename, info)
300 elif exc is ImportError and \
301 split(lower(str(value)))[:2] == ['no', 'module']:
302 # The module was not found.
303 return None
304 else:
305 # Some other error occurred during the importing process.
306 raise ErrorDuringImport(path, sys.exc_info())
307 for part in split(path, '.')[1:]:
308 try: module = getattr(module, part)
309 except AttributeError: return None
310 return module
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000311
312# ---------------------------------------------------- formatter base class
313
314class Doc:
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000315 def document(self, object, name=None, *args):
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000316 """Generate documentation for an object."""
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000317 args = (object, name) + args
Brett Cannon28a4f0f2003-06-11 23:38:55 +0000318 # 'try' clause is to attempt to handle the possibility that inspect
319 # identifies something in a way that pydoc itself has issues handling;
320 # think 'super' and how it is a descriptor (which raises the exception
321 # by lacking a __name__ attribute) and an instance.
Barry Warsaw00decd72006-07-27 23:43:15 +0000322 if inspect.isgetsetdescriptor(object): return self.docdata(*args)
323 if inspect.ismemberdescriptor(object): return self.docdata(*args)
Brett Cannon28a4f0f2003-06-11 23:38:55 +0000324 try:
325 if inspect.ismodule(object): return self.docmodule(*args)
326 if inspect.isclass(object): return self.docclass(*args)
327 if inspect.isroutine(object): return self.docroutine(*args)
328 except AttributeError:
329 pass
Johannes Gijsbers8de645a2004-11-07 19:16:05 +0000330 if isinstance(object, property): return self.docproperty(*args)
Guido van Rossum68468eb2003-02-27 20:14:51 +0000331 return self.docother(*args)
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000332
333 def fail(self, object, name=None, *args):
334 """Raise an exception for unimplemented types."""
335 message = "don't know how to document object%s of type %s" % (
336 name and ' ' + repr(name), type(object).__name__)
337 raise TypeError, message
338
Barry Warsaw00decd72006-07-27 23:43:15 +0000339 docmodule = docclass = docroutine = docother = docproperty = docdata = fail
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000340
Skip Montanaro4997a692003-09-10 16:47:51 +0000341 def getdocloc(self, object):
342 """Return the location of module docs or None"""
343
344 try:
345 file = inspect.getabsfile(object)
346 except TypeError:
347 file = '(built-in)'
348
349 docloc = os.environ.get("PYTHONDOCS",
Georg Brandl0751d1a2008-01-21 17:13:03 +0000350 "http://docs.python.org/library")
Skip Montanaro4997a692003-09-10 16:47:51 +0000351 basedir = os.path.join(sys.exec_prefix, "lib",
352 "python"+sys.version[0:3])
353 if (isinstance(object, type(os)) and
354 (object.__name__ in ('errno', 'exceptions', 'gc', 'imp',
355 'marshal', 'posix', 'signal', 'sys',
356 'thread', 'zipimport') or
357 (file.startswith(basedir) and
358 not file.startswith(os.path.join(basedir, 'site-packages'))))):
359 if docloc.startswith("http://"):
Georg Brandl0751d1a2008-01-21 17:13:03 +0000360 docloc = "%s/%s" % (docloc.rstrip("/"), object.__name__)
Skip Montanaro4997a692003-09-10 16:47:51 +0000361 else:
Georg Brandl0751d1a2008-01-21 17:13:03 +0000362 docloc = os.path.join(docloc, object.__name__ + ".html")
Skip Montanaro4997a692003-09-10 16:47:51 +0000363 else:
364 docloc = None
365 return docloc
366
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000367# -------------------------------------------- HTML documentation generator
368
369class HTMLRepr(Repr):
370 """Class for safely making an HTML representation of a Python object."""
371 def __init__(self):
372 Repr.__init__(self)
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000373 self.maxlist = self.maxtuple = 20
374 self.maxdict = 10
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000375 self.maxstring = self.maxother = 100
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000376
377 def escape(self, text):
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +0000378 return replace(text, '&', '&amp;', '<', '&lt;', '>', '&gt;')
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000379
380 def repr(self, object):
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000381 return Repr.repr(self, object)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000382
383 def repr1(self, x, level):
Skip Montanaro0fe8fce2003-06-27 15:45:41 +0000384 if hasattr(type(x), '__name__'):
385 methodname = 'repr_' + join(split(type(x).__name__), '_')
386 if hasattr(self, methodname):
387 return getattr(self, methodname)(x, level)
388 return self.escape(cram(stripid(repr(x)), self.maxother))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000389
390 def repr_string(self, x, level):
Ka-Ping Yeea2fe1032001-03-02 01:19:14 +0000391 test = cram(x, self.maxstring)
392 testrepr = repr(test)
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +0000393 if '\\' in test and '\\' not in replace(testrepr, r'\\', ''):
Ka-Ping Yeea2fe1032001-03-02 01:19:14 +0000394 # Backslashes are only literal in the string and are never
395 # needed to make any special characters, so show a raw string.
396 return 'r' + testrepr[0] + self.escape(test) + testrepr[0]
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +0000397 return re.sub(r'((\\[\\abfnrtv\'"]|\\[0-9]..|\\x..|\\u....)+)',
Ka-Ping Yeea2fe1032001-03-02 01:19:14 +0000398 r'<font color="#c040c0">\1</font>',
399 self.escape(testrepr))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000400
Skip Montanarodf708782002-03-07 22:58:02 +0000401 repr_str = repr_string
402
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000403 def repr_instance(self, x, level):
404 try:
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000405 return self.escape(cram(stripid(repr(x)), self.maxstring))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000406 except:
407 return self.escape('<%s instance>' % x.__class__.__name__)
408
409 repr_unicode = repr_string
410
411class HTMLDoc(Doc):
412 """Formatter class for HTML documentation."""
413
414 # ------------------------------------------- HTML formatting utilities
415
416 _repr_instance = HTMLRepr()
417 repr = _repr_instance.repr
418 escape = _repr_instance.escape
419
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000420 def page(self, title, contents):
421 """Format an HTML page."""
422 return '''
Tim Peters59ed4482001-10-31 04:20:26 +0000423<!doctype html PUBLIC "-//W3C//DTD HTML 4.0 Transitional//EN">
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000424<html><head><title>Python: %s</title>
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000425</head><body bgcolor="#f0f0f8">
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000426%s
427</body></html>''' % (title, contents)
428
429 def heading(self, title, fgcol, bgcol, extras=''):
430 """Format a page heading."""
431 return '''
Tim Peters59ed4482001-10-31 04:20:26 +0000432<table width="100%%" cellspacing=0 cellpadding=2 border=0 summary="heading">
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000433<tr bgcolor="%s">
Tim Peters2306d242001-09-25 03:18:32 +0000434<td valign=bottom>&nbsp;<br>
435<font color="%s" face="helvetica, arial">&nbsp;<br>%s</font></td
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000436><td align=right valign=bottom
Ka-Ping Yee987ec902001-03-23 13:35:45 +0000437><font color="%s" face="helvetica, arial">%s</font></td></tr></table>
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000438 ''' % (bgcol, fgcol, title, fgcol, extras or '&nbsp;')
439
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000440 def section(self, title, fgcol, bgcol, contents, width=6,
441 prelude='', marginalia=None, gap='&nbsp;'):
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000442 """Format a section with a heading."""
443 if marginalia is None:
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +0000444 marginalia = '<tt>' + '&nbsp;' * width + '</tt>'
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000445 result = '''<p>
Tim Peters59ed4482001-10-31 04:20:26 +0000446<table width="100%%" cellspacing=0 cellpadding=2 border=0 summary="section">
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000447<tr bgcolor="%s">
Tim Peters2306d242001-09-25 03:18:32 +0000448<td colspan=3 valign=bottom>&nbsp;<br>
449<font color="%s" face="helvetica, arial">%s</font></td></tr>
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000450 ''' % (bgcol, fgcol, title)
451 if prelude:
452 result = result + '''
Ka-Ping Yee987ec902001-03-23 13:35:45 +0000453<tr bgcolor="%s"><td rowspan=2>%s</td>
454<td colspan=2>%s</td></tr>
455<tr><td>%s</td>''' % (bgcol, marginalia, prelude, gap)
456 else:
457 result = result + '''
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000458<tr><td bgcolor="%s">%s</td><td>%s</td>''' % (bgcol, marginalia, gap)
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000459
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +0000460 return result + '\n<td width="100%%">%s</td></tr></table>' % contents
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000461
462 def bigsection(self, title, *args):
463 """Format a section with a big heading."""
464 title = '<big><strong>%s</strong></big>' % title
Guido van Rossum68468eb2003-02-27 20:14:51 +0000465 return self.section(title, *args)
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000466
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000467 def preformat(self, text):
468 """Format literal preformatted text."""
469 text = self.escape(expandtabs(text))
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +0000470 return replace(text, '\n\n', '\n \n', '\n\n', '\n \n',
471 ' ', '&nbsp;', '\n', '<br>\n')
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000472
473 def multicolumn(self, list, format, cols=4):
474 """Format a list of items into a multi-column list."""
475 result = ''
476 rows = (len(list)+cols-1)/cols
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000477 for col in range(cols):
478 result = result + '<td width="%d%%" valign=top>' % (100/cols)
479 for i in range(rows*col, rows*col+rows):
480 if i < len(list):
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000481 result = result + format(list[i]) + '<br>\n'
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000482 result = result + '</td>'
Tim Peters59ed4482001-10-31 04:20:26 +0000483 return '<table width="100%%" summary="list"><tr>%s</tr></table>' % result
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000484
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000485 def grey(self, text): return '<font color="#909090">%s</font>' % text
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000486
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000487 def namelink(self, name, *dicts):
488 """Make a link for an identifier, given name-to-URL mappings."""
489 for dict in dicts:
Raymond Hettinger54f02222002-06-01 14:18:47 +0000490 if name in dict:
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000491 return '<a href="%s">%s</a>' % (dict[name], name)
492 return name
493
Ka-Ping Yeeb7a48302001-04-12 20:39:14 +0000494 def classlink(self, object, modname):
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000495 """Make a link for a class."""
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000496 name, module = object.__name__, sys.modules.get(object.__module__)
497 if hasattr(module, name) and getattr(module, name) is object:
Ka-Ping Yeeb7a48302001-04-12 20:39:14 +0000498 return '<a href="%s.html#%s">%s</a>' % (
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000499 module.__name__, name, classname(object, modname))
500 return classname(object, modname)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000501
502 def modulelink(self, object):
503 """Make a link for a module."""
504 return '<a href="%s.html">%s</a>' % (object.__name__, object.__name__)
505
506 def modpkglink(self, (name, path, ispackage, shadowed)):
507 """Make a link for a module or package to display in an index."""
508 if shadowed:
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000509 return self.grey(name)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000510 if path:
511 url = '%s.%s.html' % (path, name)
512 else:
513 url = '%s.html' % name
514 if ispackage:
515 text = '<strong>%s</strong>&nbsp;(package)' % name
516 else:
517 text = name
518 return '<a href="%s">%s</a>' % (url, text)
519
520 def markup(self, text, escape=None, funcs={}, classes={}, methods={}):
521 """Mark up some plain text, given a context of symbols to look for.
522 Each context dictionary maps object names to anchor names."""
523 escape = escape or self.escape
524 results = []
525 here = 0
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000526 pattern = re.compile(r'\b((http|ftp)://\S+[\w/]|'
527 r'RFC[- ]?(\d+)|'
Ka-Ping Yeef78a81b2001-03-27 08:13:42 +0000528 r'PEP[- ]?(\d+)|'
Neil Schemenauerd69711c2002-03-24 23:02:07 +0000529 r'(self\.)?(\w+))')
Guido van Rossum8ca162f2002-04-07 06:36:23 +0000530 while True:
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000531 match = pattern.search(text, here)
532 if not match: break
533 start, end = match.span()
534 results.append(escape(text[here:start]))
535
Ka-Ping Yeef78a81b2001-03-27 08:13:42 +0000536 all, scheme, rfc, pep, selfdot, name = match.groups()
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000537 if scheme:
Neil Schemenauercddc1a02002-03-24 23:11:21 +0000538 url = escape(all).replace('"', '&quot;')
539 results.append('<a href="%s">%s</a>' % (url, url))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000540 elif rfc:
Ka-Ping Yeef78a81b2001-03-27 08:13:42 +0000541 url = 'http://www.rfc-editor.org/rfc/rfc%d.txt' % int(rfc)
542 results.append('<a href="%s">%s</a>' % (url, escape(all)))
543 elif pep:
Andrew M. Kuchling727a5902008-02-05 16:06:57 +0000544 url = 'http://www.python.org/dev/peps/pep-%04d/' % int(pep)
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000545 results.append('<a href="%s">%s</a>' % (url, escape(all)))
546 elif text[end:end+1] == '(':
547 results.append(self.namelink(name, methods, funcs, classes))
548 elif selfdot:
549 results.append('self.<strong>%s</strong>' % name)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000550 else:
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000551 results.append(self.namelink(name, classes))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000552 here = end
553 results.append(escape(text[here:]))
554 return join(results, '')
555
556 # ---------------------------------------------- type-specific routines
557
Ka-Ping Yeeb7a48302001-04-12 20:39:14 +0000558 def formattree(self, tree, modname, parent=None):
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000559 """Produce HTML for a class tree as given by inspect.getclasstree()."""
560 result = ''
561 for entry in tree:
562 if type(entry) is type(()):
563 c, bases = entry
Tim Peters2306d242001-09-25 03:18:32 +0000564 result = result + '<dt><font face="helvetica, arial">'
Ka-Ping Yeeb7a48302001-04-12 20:39:14 +0000565 result = result + self.classlink(c, modname)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000566 if bases and bases != (parent,):
567 parents = []
568 for base in bases:
Ka-Ping Yeeb7a48302001-04-12 20:39:14 +0000569 parents.append(self.classlink(base, modname))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000570 result = result + '(' + join(parents, ', ') + ')'
Tim Peters2306d242001-09-25 03:18:32 +0000571 result = result + '\n</font></dt>'
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000572 elif type(entry) is type([]):
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000573 result = result + '<dd>\n%s</dd>\n' % self.formattree(
Ka-Ping Yeeb7a48302001-04-12 20:39:14 +0000574 entry, modname, c)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000575 return '<dl>\n%s</dl>\n' % result
576
Tim Peters8dd7ade2001-10-18 19:56:17 +0000577 def docmodule(self, object, name=None, mod=None, *ignored):
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000578 """Produce HTML documentation for a module object."""
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000579 name = object.__name__ # ignore the passed-in name
Skip Montanaroa5616d22004-06-11 04:46:12 +0000580 try:
581 all = object.__all__
582 except AttributeError:
583 all = None
Ka-Ping Yee66efbc72001-03-01 13:55:20 +0000584 parts = split(name, '.')
585 links = []
586 for i in range(len(parts)-1):
587 links.append(
588 '<a href="%s.html"><font color="#ffffff">%s</font></a>' %
589 (join(parts[:i+1], '.'), parts[i]))
590 linkedname = join(links + parts[-1:], '.')
591 head = '<big><big><strong>%s</strong></big></big>' % linkedname
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000592 try:
Ka-Ping Yee239432a2001-03-02 02:45:08 +0000593 path = inspect.getabsfile(object)
Ka-Ping Yee6191a232001-04-13 15:00:27 +0000594 url = path
595 if sys.platform == 'win32':
596 import nturl2path
597 url = nturl2path.pathname2url(path)
598 filelink = '<a href="file:%s">%s</a>' % (url, path)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000599 except TypeError:
600 filelink = '(built-in)'
Ka-Ping Yee6f3f9a42001-02-27 22:42:36 +0000601 info = []
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000602 if hasattr(object, '__version__'):
Ka-Ping Yee6f3f9a42001-02-27 22:42:36 +0000603 version = str(object.__version__)
Ka-Ping Yee40c49912001-02-27 22:46:01 +0000604 if version[:11] == '$' + 'Revision: ' and version[-1:] == '$':
605 version = strip(version[11:-1])
Ka-Ping Yee1d384632001-03-01 00:24:32 +0000606 info.append('version %s' % self.escape(version))
Ka-Ping Yee6f3f9a42001-02-27 22:42:36 +0000607 if hasattr(object, '__date__'):
608 info.append(self.escape(str(object.__date__)))
609 if info:
610 head = head + ' (%s)' % join(info, ', ')
Skip Montanaro4997a692003-09-10 16:47:51 +0000611 docloc = self.getdocloc(object)
612 if docloc is not None:
613 docloc = '<br><a href="%(docloc)s">Module Docs</a>' % locals()
614 else:
615 docloc = ''
Ka-Ping Yee66efbc72001-03-01 13:55:20 +0000616 result = self.heading(
Skip Montanaro4997a692003-09-10 16:47:51 +0000617 head, '#ffffff', '#7799ee',
618 '<a href=".">index</a><br>' + filelink + docloc)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000619
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000620 modules = inspect.getmembers(object, inspect.ismodule)
621
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000622 classes, cdict = [], {}
623 for key, value in inspect.getmembers(object, inspect.isclass):
Johannes Gijsbers4c11f602004-08-30 14:13:04 +0000624 # if __all__ exists, believe it. Otherwise use old heuristic.
625 if (all is not None or
626 (inspect.getmodule(value) or object) is object):
Skip Montanaroa5616d22004-06-11 04:46:12 +0000627 if visiblename(key, all):
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000628 classes.append((key, value))
629 cdict[key] = cdict[value] = '#' + key
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000630 for key, value in classes:
631 for base in value.__bases__:
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000632 key, modname = base.__name__, base.__module__
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000633 module = sys.modules.get(modname)
634 if modname != name and module and hasattr(module, key):
635 if getattr(module, key) is base:
Raymond Hettinger54f02222002-06-01 14:18:47 +0000636 if not key in cdict:
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000637 cdict[key] = cdict[base] = modname + '.html#' + key
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000638 funcs, fdict = [], {}
639 for key, value in inspect.getmembers(object, inspect.isroutine):
Johannes Gijsbers4c11f602004-08-30 14:13:04 +0000640 # if __all__ exists, believe it. Otherwise use old heuristic.
641 if (all is not None or
642 inspect.isbuiltin(value) or inspect.getmodule(value) is object):
Skip Montanaroa5616d22004-06-11 04:46:12 +0000643 if visiblename(key, all):
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000644 funcs.append((key, value))
645 fdict[key] = '#-' + key
646 if inspect.isfunction(value): fdict[value] = fdict[key]
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000647 data = []
648 for key, value in inspect.getmembers(object, isdata):
Skip Montanaroa5616d22004-06-11 04:46:12 +0000649 if visiblename(key, all):
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000650 data.append((key, value))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000651
652 doc = self.markup(getdoc(object), self.preformat, fdict, cdict)
653 doc = doc and '<tt>%s</tt>' % doc
Tim Peters2306d242001-09-25 03:18:32 +0000654 result = result + '<p>%s</p>\n' % doc
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000655
656 if hasattr(object, '__path__'):
657 modpkgs = []
Phillip J. Ebyceb30872006-04-18 00:59:55 +0000658 for importer, modname, ispkg in pkgutil.iter_modules(object.__path__):
659 modpkgs.append((modname, name, ispkg, 0))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000660 modpkgs.sort()
661 contents = self.multicolumn(modpkgs, self.modpkglink)
662 result = result + self.bigsection(
663 'Package Contents', '#ffffff', '#aa55cc', contents)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000664 elif modules:
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000665 contents = self.multicolumn(
666 modules, lambda (key, value), s=self: s.modulelink(value))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000667 result = result + self.bigsection(
Georg Brandlb169eaa2008-02-17 21:18:55 +0000668 'Modules', '#ffffff', '#aa55cc', contents)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000669
670 if classes:
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000671 classlist = map(lambda (key, value): value, classes)
Ka-Ping Yeeb7a48302001-04-12 20:39:14 +0000672 contents = [
673 self.formattree(inspect.getclasstree(classlist, 1), name)]
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000674 for key, value in classes:
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +0000675 contents.append(self.document(value, key, name, fdict, cdict))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000676 result = result + self.bigsection(
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000677 'Classes', '#ffffff', '#ee77aa', join(contents))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000678 if funcs:
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000679 contents = []
680 for key, value in funcs:
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +0000681 contents.append(self.document(value, key, name, fdict, cdict))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000682 result = result + self.bigsection(
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000683 'Functions', '#ffffff', '#eeaa77', join(contents))
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000684 if data:
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000685 contents = []
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000686 for key, value in data:
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000687 contents.append(self.document(value, key))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000688 result = result + self.bigsection(
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000689 'Data', '#ffffff', '#55aa55', join(contents, '<br>\n'))
Ka-Ping Yee6f3f9a42001-02-27 22:42:36 +0000690 if hasattr(object, '__author__'):
691 contents = self.markup(str(object.__author__), self.preformat)
692 result = result + self.bigsection(
693 'Author', '#ffffff', '#7799ee', contents)
Ka-Ping Yee6f3f9a42001-02-27 22:42:36 +0000694 if hasattr(object, '__credits__'):
695 contents = self.markup(str(object.__credits__), self.preformat)
696 result = result + self.bigsection(
697 'Credits', '#ffffff', '#7799ee', contents)
698
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000699 return result
700
Tim Peters8dd7ade2001-10-18 19:56:17 +0000701 def docclass(self, object, name=None, mod=None, funcs={}, classes={},
702 *ignored):
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000703 """Produce HTML documentation for a class object."""
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000704 realname = object.__name__
705 name = name or realname
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000706 bases = object.__bases__
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000707
Tim Petersb47879b2001-09-24 04:47:19 +0000708 contents = []
709 push = contents.append
710
Tim Petersfa26f7c2001-09-24 08:05:11 +0000711 # Cute little class to pump out a horizontal rule between sections.
712 class HorizontalRule:
713 def __init__(self):
714 self.needone = 0
715 def maybe(self):
716 if self.needone:
717 push('<hr>\n')
718 self.needone = 1
719 hr = HorizontalRule()
720
Tim Petersc86f6ca2001-09-26 21:31:51 +0000721 # List the mro, if non-trivial.
Raymond Hettinger756b3f32004-01-29 06:37:52 +0000722 mro = deque(inspect.getmro(object))
Tim Petersc86f6ca2001-09-26 21:31:51 +0000723 if len(mro) > 2:
724 hr.maybe()
725 push('<dl><dt>Method resolution order:</dt>\n')
726 for base in mro:
727 push('<dd>%s</dd>\n' % self.classlink(base,
728 object.__module__))
729 push('</dl>\n')
730
Tim Petersb47879b2001-09-24 04:47:19 +0000731 def spill(msg, attrs, predicate):
Tim Petersfa26f7c2001-09-24 08:05:11 +0000732 ok, attrs = _split_list(attrs, predicate)
Tim Petersb47879b2001-09-24 04:47:19 +0000733 if ok:
Tim Petersfa26f7c2001-09-24 08:05:11 +0000734 hr.maybe()
Tim Petersb47879b2001-09-24 04:47:19 +0000735 push(msg)
736 for name, kind, homecls, value in ok:
737 push(self.document(getattr(object, name), name, mod,
738 funcs, classes, mdict, object))
739 push('\n')
740 return attrs
741
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +0000742 def spilldescriptors(msg, attrs, predicate):
Tim Petersfa26f7c2001-09-24 08:05:11 +0000743 ok, attrs = _split_list(attrs, predicate)
Tim Petersb47879b2001-09-24 04:47:19 +0000744 if ok:
Tim Petersfa26f7c2001-09-24 08:05:11 +0000745 hr.maybe()
Tim Petersb47879b2001-09-24 04:47:19 +0000746 push(msg)
747 for name, kind, homecls, value in ok:
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +0000748 push(self._docdescriptor(name, value, mod))
Tim Petersb47879b2001-09-24 04:47:19 +0000749 return attrs
750
Tim Petersfa26f7c2001-09-24 08:05:11 +0000751 def spilldata(msg, attrs, predicate):
752 ok, attrs = _split_list(attrs, predicate)
Tim Petersb47879b2001-09-24 04:47:19 +0000753 if ok:
Tim Petersfa26f7c2001-09-24 08:05:11 +0000754 hr.maybe()
Tim Petersb47879b2001-09-24 04:47:19 +0000755 push(msg)
756 for name, kind, homecls, value in ok:
757 base = self.docother(getattr(object, name), name, mod)
Martin v. Löwise59e2ba2003-05-03 09:09:02 +0000758 if callable(value) or inspect.isdatadescriptor(value):
Guido van Rossum5e355b22002-05-21 20:56:15 +0000759 doc = getattr(value, "__doc__", None)
760 else:
761 doc = None
Tim Petersb47879b2001-09-24 04:47:19 +0000762 if doc is None:
763 push('<dl><dt>%s</dl>\n' % base)
764 else:
765 doc = self.markup(getdoc(value), self.preformat,
766 funcs, classes, mdict)
Tim Peters2306d242001-09-25 03:18:32 +0000767 doc = '<dd><tt>%s</tt>' % doc
Tim Petersb47879b2001-09-24 04:47:19 +0000768 push('<dl><dt>%s%s</dl>\n' % (base, doc))
769 push('\n')
770 return attrs
771
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000772 attrs = filter(lambda (name, kind, cls, value): visiblename(name),
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +0000773 classify_class_attrs(object))
Tim Petersb47879b2001-09-24 04:47:19 +0000774 mdict = {}
775 for key, kind, homecls, value in attrs:
776 mdict[key] = anchor = '#' + name + '-' + key
777 value = getattr(object, key)
778 try:
779 # The value may not be hashable (e.g., a data attr with
780 # a dict or list value).
781 mdict[value] = anchor
782 except TypeError:
783 pass
784
Tim Petersfa26f7c2001-09-24 08:05:11 +0000785 while attrs:
Tim Peters351e3622001-09-27 03:29:51 +0000786 if mro:
Raymond Hettinger756b3f32004-01-29 06:37:52 +0000787 thisclass = mro.popleft()
Tim Peters351e3622001-09-27 03:29:51 +0000788 else:
789 thisclass = attrs[0][2]
Tim Petersfa26f7c2001-09-24 08:05:11 +0000790 attrs, inherited = _split_list(attrs, lambda t: t[2] is thisclass)
791
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000792 if thisclass is __builtin__.object:
793 attrs = inherited
794 continue
795 elif thisclass is object:
796 tag = 'defined here'
Tim Petersb47879b2001-09-24 04:47:19 +0000797 else:
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000798 tag = 'inherited from %s' % self.classlink(thisclass,
799 object.__module__)
Tim Petersb47879b2001-09-24 04:47:19 +0000800 tag += ':<br>\n'
801
802 # Sort attrs by name.
Phillip J. Ebyceb30872006-04-18 00:59:55 +0000803 try:
804 attrs.sort(key=lambda t: t[0])
805 except TypeError:
806 attrs.sort(lambda t1, t2: cmp(t1[0], t2[0])) # 2.3 compat
Tim Petersb47879b2001-09-24 04:47:19 +0000807
808 # Pump out the attrs, segregated by kind.
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000809 attrs = spill('Methods %s' % tag, attrs,
Tim Petersb47879b2001-09-24 04:47:19 +0000810 lambda t: t[1] == 'method')
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000811 attrs = spill('Class methods %s' % tag, attrs,
Tim Petersb47879b2001-09-24 04:47:19 +0000812 lambda t: t[1] == 'class method')
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000813 attrs = spill('Static methods %s' % tag, attrs,
Tim Petersb47879b2001-09-24 04:47:19 +0000814 lambda t: t[1] == 'static method')
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +0000815 attrs = spilldescriptors('Data descriptors %s' % tag, attrs,
816 lambda t: t[1] == 'data descriptor')
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000817 attrs = spilldata('Data and other attributes %s' % tag, attrs,
Tim Petersfa26f7c2001-09-24 08:05:11 +0000818 lambda t: t[1] == 'data')
Tim Petersb47879b2001-09-24 04:47:19 +0000819 assert attrs == []
Tim Peters351e3622001-09-27 03:29:51 +0000820 attrs = inherited
Tim Petersb47879b2001-09-24 04:47:19 +0000821
822 contents = ''.join(contents)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000823
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000824 if name == realname:
825 title = '<a name="%s">class <strong>%s</strong></a>' % (
826 name, realname)
827 else:
828 title = '<strong>%s</strong> = <a name="%s">class %s</a>' % (
829 name, name, realname)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000830 if bases:
831 parents = []
832 for base in bases:
Ka-Ping Yeeb7a48302001-04-12 20:39:14 +0000833 parents.append(self.classlink(base, object.__module__))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000834 title = title + '(%s)' % join(parents, ', ')
Tim Peters2306d242001-09-25 03:18:32 +0000835 doc = self.markup(getdoc(object), self.preformat, funcs, classes, mdict)
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000836 doc = doc and '<tt>%s<br>&nbsp;</tt>' % doc
Tim Petersc86f6ca2001-09-26 21:31:51 +0000837
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000838 return self.section(title, '#000000', '#ffc8d8', contents, 3, doc)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000839
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000840 def formatvalue(self, object):
841 """Format an argument default value as text."""
Tim Peters2306d242001-09-25 03:18:32 +0000842 return self.grey('=' + self.repr(object))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000843
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +0000844 def docroutine(self, object, name=None, mod=None,
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000845 funcs={}, classes={}, methods={}, cl=None):
Ka-Ping Yee66efbc72001-03-01 13:55:20 +0000846 """Produce HTML documentation for a function or method object."""
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000847 realname = object.__name__
848 name = name or realname
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000849 anchor = (cl and cl.__name__ or '') + '-' + name
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000850 note = ''
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000851 skipdocs = 0
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000852 if inspect.ismethod(object):
Ka-Ping Yee6dcfa382001-04-12 20:27:31 +0000853 imclass = object.im_class
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000854 if cl:
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +0000855 if imclass is not cl:
Ka-Ping Yeeb7a48302001-04-12 20:39:14 +0000856 note = ' from ' + self.classlink(imclass, mod)
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000857 else:
Georg Brandl7e2b6bb2007-03-13 22:16:30 +0000858 if object.im_self is not None:
Ka-Ping Yeeb7a48302001-04-12 20:39:14 +0000859 note = ' method of %s instance' % self.classlink(
860 object.im_self.__class__, mod)
861 else:
862 note = ' unbound %s method' % self.classlink(imclass,mod)
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000863 object = object.im_func
864
865 if name == realname:
866 title = '<a name="%s"><strong>%s</strong></a>' % (anchor, realname)
867 else:
Raymond Hettinger54f02222002-06-01 14:18:47 +0000868 if (cl and realname in cl.__dict__ and
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000869 cl.__dict__[realname] is object):
Ka-Ping Yeee280c062001-03-23 14:05:53 +0000870 reallink = '<a href="#%s">%s</a>' % (
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000871 cl.__name__ + '-' + realname, realname)
872 skipdocs = 1
873 else:
874 reallink = realname
875 title = '<a name="%s"><strong>%s</strong></a> = %s' % (
876 anchor, name, reallink)
Tim Peters4bcfa312001-09-20 06:08:24 +0000877 if inspect.isfunction(object):
Ka-Ping Yee66efbc72001-03-01 13:55:20 +0000878 args, varargs, varkw, defaults = inspect.getargspec(object)
879 argspec = inspect.formatargspec(
880 args, varargs, varkw, defaults, formatvalue=self.formatvalue)
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000881 if realname == '<lambda>':
Tim Peters59ed4482001-10-31 04:20:26 +0000882 title = '<strong>%s</strong> <em>lambda</em> ' % name
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000883 argspec = argspec[1:-1] # remove parentheses
Tim Peters4bcfa312001-09-20 06:08:24 +0000884 else:
885 argspec = '(...)'
Ka-Ping Yee66efbc72001-03-01 13:55:20 +0000886
Tim Peters2306d242001-09-25 03:18:32 +0000887 decl = title + argspec + (note and self.grey(
888 '<font face="helvetica, arial">%s</font>' % note))
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000889
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000890 if skipdocs:
Tim Peters2306d242001-09-25 03:18:32 +0000891 return '<dl><dt>%s</dt></dl>\n' % decl
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000892 else:
893 doc = self.markup(
894 getdoc(object), self.preformat, funcs, classes, methods)
Tim Peters2306d242001-09-25 03:18:32 +0000895 doc = doc and '<dd><tt>%s</tt></dd>' % doc
896 return '<dl><dt>%s</dt>%s</dl>\n' % (decl, doc)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000897
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +0000898 def _docdescriptor(self, name, value, mod):
Johannes Gijsbers8de645a2004-11-07 19:16:05 +0000899 results = []
900 push = results.append
901
902 if name:
903 push('<dl><dt><strong>%s</strong></dt>\n' % name)
904 if value.__doc__ is not None:
Ka-Ping Yeebba6acc2005-02-19 22:58:26 +0000905 doc = self.markup(getdoc(value), self.preformat)
Johannes Gijsbers8de645a2004-11-07 19:16:05 +0000906 push('<dd><tt>%s</tt></dd>\n' % doc)
Johannes Gijsbers8de645a2004-11-07 19:16:05 +0000907 push('</dl>\n')
908
909 return ''.join(results)
910
911 def docproperty(self, object, name=None, mod=None, cl=None):
912 """Produce html documentation for a property."""
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +0000913 return self._docdescriptor(name, object, mod)
Johannes Gijsbers8de645a2004-11-07 19:16:05 +0000914
Tim Peters8dd7ade2001-10-18 19:56:17 +0000915 def docother(self, object, name=None, mod=None, *ignored):
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000916 """Produce HTML documentation for a data object."""
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +0000917 lhs = name and '<strong>%s</strong> = ' % name or ''
918 return lhs + self.repr(object)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000919
Barry Warsaw00decd72006-07-27 23:43:15 +0000920 def docdata(self, object, name=None, mod=None, cl=None):
921 """Produce html documentation for a data descriptor."""
922 return self._docdescriptor(name, object, mod)
923
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000924 def index(self, dir, shadowed=None):
925 """Generate an HTML index for a directory of modules."""
926 modpkgs = []
927 if shadowed is None: shadowed = {}
Phillip J. Ebyceb30872006-04-18 00:59:55 +0000928 for importer, name, ispkg in pkgutil.iter_modules([dir]):
929 modpkgs.append((name, '', ispkg, name in shadowed))
930 shadowed[name] = 1
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000931
932 modpkgs.sort()
933 contents = self.multicolumn(modpkgs, self.modpkglink)
934 return self.bigsection(dir, '#ffffff', '#ee77aa', contents)
935
936# -------------------------------------------- text documentation generator
937
938class TextRepr(Repr):
939 """Class for safely making a text representation of a Python object."""
940 def __init__(self):
941 Repr.__init__(self)
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000942 self.maxlist = self.maxtuple = 20
943 self.maxdict = 10
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000944 self.maxstring = self.maxother = 100
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000945
946 def repr1(self, x, level):
Skip Montanaro0fe8fce2003-06-27 15:45:41 +0000947 if hasattr(type(x), '__name__'):
948 methodname = 'repr_' + join(split(type(x).__name__), '_')
949 if hasattr(self, methodname):
950 return getattr(self, methodname)(x, level)
951 return cram(stripid(repr(x)), self.maxother)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000952
Ka-Ping Yeea2fe1032001-03-02 01:19:14 +0000953 def repr_string(self, x, level):
954 test = cram(x, self.maxstring)
955 testrepr = repr(test)
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +0000956 if '\\' in test and '\\' not in replace(testrepr, r'\\', ''):
Ka-Ping Yeea2fe1032001-03-02 01:19:14 +0000957 # Backslashes are only literal in the string and are never
958 # needed to make any special characters, so show a raw string.
959 return 'r' + testrepr[0] + test + testrepr[0]
960 return testrepr
961
Skip Montanarodf708782002-03-07 22:58:02 +0000962 repr_str = repr_string
963
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000964 def repr_instance(self, x, level):
965 try:
Ka-Ping Yee1d384632001-03-01 00:24:32 +0000966 return cram(stripid(repr(x)), self.maxstring)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000967 except:
968 return '<%s instance>' % x.__class__.__name__
969
970class TextDoc(Doc):
971 """Formatter class for text documentation."""
972
973 # ------------------------------------------- text formatting utilities
974
975 _repr_instance = TextRepr()
976 repr = _repr_instance.repr
977
978 def bold(self, text):
979 """Format a string in bold by overstriking."""
980 return join(map(lambda ch: ch + '\b' + ch, text), '')
981
982 def indent(self, text, prefix=' '):
983 """Indent text by prepending a given prefix to each line."""
984 if not text: return ''
985 lines = split(text, '\n')
986 lines = map(lambda line, prefix=prefix: prefix + line, lines)
987 if lines: lines[-1] = rstrip(lines[-1])
988 return join(lines, '\n')
989
990 def section(self, title, contents):
991 """Format a section with a given heading."""
992 return self.bold(title) + '\n' + rstrip(self.indent(contents)) + '\n\n'
993
994 # ---------------------------------------------- type-specific routines
995
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000996 def formattree(self, tree, modname, parent=None, prefix=''):
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000997 """Render in text a class tree as returned by inspect.getclasstree()."""
998 result = ''
999 for entry in tree:
1000 if type(entry) is type(()):
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001001 c, bases = entry
1002 result = result + prefix + classname(c, modname)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001003 if bases and bases != (parent,):
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001004 parents = map(lambda c, m=modname: classname(c, m), bases)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001005 result = result + '(%s)' % join(parents, ', ')
1006 result = result + '\n'
1007 elif type(entry) is type([]):
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001008 result = result + self.formattree(
1009 entry, modname, c, prefix + ' ')
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001010 return result
1011
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001012 def docmodule(self, object, name=None, mod=None):
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001013 """Produce text documentation for a given module object."""
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001014 name = object.__name__ # ignore the passed-in name
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +00001015 synop, desc = splitdoc(getdoc(object))
1016 result = self.section('NAME', name + (synop and ' - ' + synop))
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001017
1018 try:
Skip Montanaroa5616d22004-06-11 04:46:12 +00001019 all = object.__all__
1020 except AttributeError:
1021 all = None
1022
1023 try:
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001024 file = inspect.getabsfile(object)
1025 except TypeError:
1026 file = '(built-in)'
Ka-Ping Yee239432a2001-03-02 02:45:08 +00001027 result = result + self.section('FILE', file)
Skip Montanaro4997a692003-09-10 16:47:51 +00001028
1029 docloc = self.getdocloc(object)
1030 if docloc is not None:
1031 result = result + self.section('MODULE DOCS', docloc)
1032
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +00001033 if desc:
1034 result = result + self.section('DESCRIPTION', desc)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001035
1036 classes = []
1037 for key, value in inspect.getmembers(object, inspect.isclass):
Johannes Gijsbers4c11f602004-08-30 14:13:04 +00001038 # if __all__ exists, believe it. Otherwise use old heuristic.
1039 if (all is not None
1040 or (inspect.getmodule(value) or object) is object):
Skip Montanaroa5616d22004-06-11 04:46:12 +00001041 if visiblename(key, all):
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +00001042 classes.append((key, value))
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001043 funcs = []
1044 for key, value in inspect.getmembers(object, inspect.isroutine):
Johannes Gijsbers4c11f602004-08-30 14:13:04 +00001045 # if __all__ exists, believe it. Otherwise use old heuristic.
1046 if (all is not None or
1047 inspect.isbuiltin(value) or inspect.getmodule(value) is object):
Skip Montanaroa5616d22004-06-11 04:46:12 +00001048 if visiblename(key, all):
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +00001049 funcs.append((key, value))
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001050 data = []
1051 for key, value in inspect.getmembers(object, isdata):
Skip Montanaroa5616d22004-06-11 04:46:12 +00001052 if visiblename(key, all):
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001053 data.append((key, value))
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001054
Georg Brandlf2dae0e2008-01-21 21:05:49 +00001055 modpkgs = []
1056 modpkgs_names = set()
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001057 if hasattr(object, '__path__'):
Phillip J. Ebyceb30872006-04-18 00:59:55 +00001058 for importer, modname, ispkg in pkgutil.iter_modules(object.__path__):
Georg Brandlf2dae0e2008-01-21 21:05:49 +00001059 modpkgs_names.add(modname)
Phillip J. Ebyceb30872006-04-18 00:59:55 +00001060 if ispkg:
1061 modpkgs.append(modname + ' (package)')
1062 else:
1063 modpkgs.append(modname)
1064
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001065 modpkgs.sort()
1066 result = result + self.section(
1067 'PACKAGE CONTENTS', join(modpkgs, '\n'))
1068
Georg Brandlf2dae0e2008-01-21 21:05:49 +00001069 # Detect submodules as sometimes created by C extensions
1070 submodules = []
1071 for key, value in inspect.getmembers(object, inspect.ismodule):
1072 if value.__name__.startswith(name + '.') and key not in modpkgs_names:
1073 submodules.append(key)
1074 if submodules:
1075 submodules.sort()
1076 result = result + self.section(
1077 'SUBMODULES', join(submodules, '\n'))
1078
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001079 if classes:
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001080 classlist = map(lambda (key, value): value, classes)
1081 contents = [self.formattree(
1082 inspect.getclasstree(classlist, 1), name)]
1083 for key, value in classes:
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001084 contents.append(self.document(value, key, name))
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001085 result = result + self.section('CLASSES', join(contents, '\n'))
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001086
1087 if funcs:
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001088 contents = []
1089 for key, value in funcs:
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001090 contents.append(self.document(value, key, name))
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001091 result = result + self.section('FUNCTIONS', join(contents, '\n'))
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001092
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001093 if data:
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001094 contents = []
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001095 for key, value in data:
Georg Brandl8b813db2005-10-01 16:32:31 +00001096 contents.append(self.docother(value, key, name, maxlen=70))
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001097 result = result + self.section('DATA', join(contents, '\n'))
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001098
1099 if hasattr(object, '__version__'):
1100 version = str(object.__version__)
Ka-Ping Yee1d384632001-03-01 00:24:32 +00001101 if version[:11] == '$' + 'Revision: ' and version[-1:] == '$':
1102 version = strip(version[11:-1])
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001103 result = result + self.section('VERSION', version)
Ka-Ping Yee6f3f9a42001-02-27 22:42:36 +00001104 if hasattr(object, '__date__'):
1105 result = result + self.section('DATE', str(object.__date__))
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001106 if hasattr(object, '__author__'):
Ka-Ping Yee6f3f9a42001-02-27 22:42:36 +00001107 result = result + self.section('AUTHOR', str(object.__author__))
1108 if hasattr(object, '__credits__'):
1109 result = result + self.section('CREDITS', str(object.__credits__))
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001110 return result
1111
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001112 def docclass(self, object, name=None, mod=None):
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001113 """Produce text documentation for a given class object."""
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001114 realname = object.__name__
1115 name = name or realname
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001116 bases = object.__bases__
1117
Tim Petersc86f6ca2001-09-26 21:31:51 +00001118 def makename(c, m=object.__module__):
1119 return classname(c, m)
1120
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001121 if name == realname:
1122 title = 'class ' + self.bold(realname)
1123 else:
1124 title = self.bold(name) + ' = class ' + realname
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001125 if bases:
Ka-Ping Yee3bda8792001-03-23 13:17:50 +00001126 parents = map(makename, bases)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001127 title = title + '(%s)' % join(parents, ', ')
1128
1129 doc = getdoc(object)
Tim Peters28355492001-09-23 21:29:55 +00001130 contents = doc and [doc + '\n'] or []
1131 push = contents.append
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001132
Tim Petersc86f6ca2001-09-26 21:31:51 +00001133 # List the mro, if non-trivial.
Raymond Hettinger756b3f32004-01-29 06:37:52 +00001134 mro = deque(inspect.getmro(object))
Tim Petersc86f6ca2001-09-26 21:31:51 +00001135 if len(mro) > 2:
1136 push("Method resolution order:")
1137 for base in mro:
1138 push(' ' + makename(base))
1139 push('')
1140
Tim Petersf4aad8e2001-09-24 22:40:47 +00001141 # Cute little class to pump out a horizontal rule between sections.
1142 class HorizontalRule:
1143 def __init__(self):
1144 self.needone = 0
1145 def maybe(self):
1146 if self.needone:
1147 push('-' * 70)
1148 self.needone = 1
1149 hr = HorizontalRule()
1150
Tim Peters28355492001-09-23 21:29:55 +00001151 def spill(msg, attrs, predicate):
Tim Petersfa26f7c2001-09-24 08:05:11 +00001152 ok, attrs = _split_list(attrs, predicate)
Tim Peters28355492001-09-23 21:29:55 +00001153 if ok:
Tim Petersf4aad8e2001-09-24 22:40:47 +00001154 hr.maybe()
Tim Peters28355492001-09-23 21:29:55 +00001155 push(msg)
1156 for name, kind, homecls, value in ok:
1157 push(self.document(getattr(object, name),
1158 name, mod, object))
1159 return attrs
1160
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +00001161 def spilldescriptors(msg, attrs, predicate):
Tim Petersfa26f7c2001-09-24 08:05:11 +00001162 ok, attrs = _split_list(attrs, predicate)
Tim Peters28355492001-09-23 21:29:55 +00001163 if ok:
Tim Petersf4aad8e2001-09-24 22:40:47 +00001164 hr.maybe()
Tim Peters28355492001-09-23 21:29:55 +00001165 push(msg)
1166 for name, kind, homecls, value in ok:
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +00001167 push(self._docdescriptor(name, value, mod))
Tim Peters28355492001-09-23 21:29:55 +00001168 return attrs
Tim Petersb47879b2001-09-24 04:47:19 +00001169
Tim Petersfa26f7c2001-09-24 08:05:11 +00001170 def spilldata(msg, attrs, predicate):
1171 ok, attrs = _split_list(attrs, predicate)
Tim Peters28355492001-09-23 21:29:55 +00001172 if ok:
Tim Petersf4aad8e2001-09-24 22:40:47 +00001173 hr.maybe()
Tim Peters28355492001-09-23 21:29:55 +00001174 push(msg)
1175 for name, kind, homecls, value in ok:
Martin v. Löwise59e2ba2003-05-03 09:09:02 +00001176 if callable(value) or inspect.isdatadescriptor(value):
Ka-Ping Yeebba6acc2005-02-19 22:58:26 +00001177 doc = getdoc(value)
Guido van Rossum5e355b22002-05-21 20:56:15 +00001178 else:
1179 doc = None
Tim Peters28355492001-09-23 21:29:55 +00001180 push(self.docother(getattr(object, name),
Georg Brandl8b813db2005-10-01 16:32:31 +00001181 name, mod, maxlen=70, doc=doc) + '\n')
Tim Peters28355492001-09-23 21:29:55 +00001182 return attrs
1183
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +00001184 attrs = filter(lambda (name, kind, cls, value): visiblename(name),
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +00001185 classify_class_attrs(object))
Tim Petersfa26f7c2001-09-24 08:05:11 +00001186 while attrs:
Tim Peters351e3622001-09-27 03:29:51 +00001187 if mro:
Raymond Hettinger756b3f32004-01-29 06:37:52 +00001188 thisclass = mro.popleft()
Tim Peters351e3622001-09-27 03:29:51 +00001189 else:
1190 thisclass = attrs[0][2]
Tim Petersfa26f7c2001-09-24 08:05:11 +00001191 attrs, inherited = _split_list(attrs, lambda t: t[2] is thisclass)
1192
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +00001193 if thisclass is __builtin__.object:
1194 attrs = inherited
1195 continue
1196 elif thisclass is object:
Tim Peters28355492001-09-23 21:29:55 +00001197 tag = "defined here"
1198 else:
Tim Petersfa26f7c2001-09-24 08:05:11 +00001199 tag = "inherited from %s" % classname(thisclass,
1200 object.__module__)
Tim Peters28355492001-09-23 21:29:55 +00001201
1202 # Sort attrs by name.
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +00001203 attrs.sort()
Tim Peters28355492001-09-23 21:29:55 +00001204
1205 # Pump out the attrs, segregated by kind.
Tim Petersf4aad8e2001-09-24 22:40:47 +00001206 attrs = spill("Methods %s:\n" % tag, attrs,
Tim Peters28355492001-09-23 21:29:55 +00001207 lambda t: t[1] == 'method')
Tim Petersf4aad8e2001-09-24 22:40:47 +00001208 attrs = spill("Class methods %s:\n" % tag, attrs,
Tim Peters28355492001-09-23 21:29:55 +00001209 lambda t: t[1] == 'class method')
Tim Petersf4aad8e2001-09-24 22:40:47 +00001210 attrs = spill("Static methods %s:\n" % tag, attrs,
Tim Peters28355492001-09-23 21:29:55 +00001211 lambda t: t[1] == 'static method')
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +00001212 attrs = spilldescriptors("Data descriptors %s:\n" % tag, attrs,
1213 lambda t: t[1] == 'data descriptor')
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +00001214 attrs = spilldata("Data and other attributes %s:\n" % tag, attrs,
1215 lambda t: t[1] == 'data')
Tim Peters28355492001-09-23 21:29:55 +00001216 assert attrs == []
Tim Peters351e3622001-09-27 03:29:51 +00001217 attrs = inherited
Tim Peters28355492001-09-23 21:29:55 +00001218
1219 contents = '\n'.join(contents)
1220 if not contents:
1221 return title + '\n'
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001222 return title + '\n' + self.indent(rstrip(contents), ' | ') + '\n'
1223
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001224 def formatvalue(self, object):
1225 """Format an argument default value as text."""
1226 return '=' + self.repr(object)
1227
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001228 def docroutine(self, object, name=None, mod=None, cl=None):
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001229 """Produce text documentation for a function or method object."""
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001230 realname = object.__name__
1231 name = name or realname
1232 note = ''
Ka-Ping Yee3bda8792001-03-23 13:17:50 +00001233 skipdocs = 0
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001234 if inspect.ismethod(object):
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001235 imclass = object.im_class
Ka-Ping Yee3bda8792001-03-23 13:17:50 +00001236 if cl:
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001237 if imclass is not cl:
1238 note = ' from ' + classname(imclass, mod)
Ka-Ping Yee3bda8792001-03-23 13:17:50 +00001239 else:
Georg Brandl7e2b6bb2007-03-13 22:16:30 +00001240 if object.im_self is not None:
Ka-Ping Yeeb7a48302001-04-12 20:39:14 +00001241 note = ' method of %s instance' % classname(
1242 object.im_self.__class__, mod)
1243 else:
1244 note = ' unbound %s method' % classname(imclass,mod)
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001245 object = object.im_func
1246
1247 if name == realname:
1248 title = self.bold(realname)
1249 else:
Raymond Hettinger54f02222002-06-01 14:18:47 +00001250 if (cl and realname in cl.__dict__ and
Ka-Ping Yee3bda8792001-03-23 13:17:50 +00001251 cl.__dict__[realname] is object):
1252 skipdocs = 1
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001253 title = self.bold(name) + ' = ' + realname
Tim Peters4bcfa312001-09-20 06:08:24 +00001254 if inspect.isfunction(object):
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001255 args, varargs, varkw, defaults = inspect.getargspec(object)
1256 argspec = inspect.formatargspec(
1257 args, varargs, varkw, defaults, formatvalue=self.formatvalue)
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001258 if realname == '<lambda>':
Georg Brandl501dd0d2006-02-17 09:45:40 +00001259 title = self.bold(name) + ' lambda '
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001260 argspec = argspec[1:-1] # remove parentheses
Tim Peters4bcfa312001-09-20 06:08:24 +00001261 else:
1262 argspec = '(...)'
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001263 decl = title + argspec + note
1264
Ka-Ping Yee3bda8792001-03-23 13:17:50 +00001265 if skipdocs:
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001266 return decl + '\n'
Ka-Ping Yee3bda8792001-03-23 13:17:50 +00001267 else:
1268 doc = getdoc(object) or ''
1269 return decl + '\n' + (doc and rstrip(self.indent(doc)) + '\n')
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001270
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +00001271 def _docdescriptor(self, name, value, mod):
Johannes Gijsbers8de645a2004-11-07 19:16:05 +00001272 results = []
1273 push = results.append
1274
1275 if name:
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +00001276 push(self.bold(name))
1277 push('\n')
Johannes Gijsbers8de645a2004-11-07 19:16:05 +00001278 doc = getdoc(value) or ''
1279 if doc:
1280 push(self.indent(doc))
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +00001281 push('\n')
1282 return ''.join(results)
Johannes Gijsbers8de645a2004-11-07 19:16:05 +00001283
1284 def docproperty(self, object, name=None, mod=None, cl=None):
1285 """Produce text documentation for a property."""
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +00001286 return self._docdescriptor(name, object, mod)
Johannes Gijsbers8de645a2004-11-07 19:16:05 +00001287
Barry Warsaw00decd72006-07-27 23:43:15 +00001288 def docdata(self, object, name=None, mod=None, cl=None):
1289 """Produce text documentation for a data descriptor."""
1290 return self._docdescriptor(name, object, mod)
1291
Georg Brandl8b813db2005-10-01 16:32:31 +00001292 def docother(self, object, name=None, mod=None, parent=None, maxlen=None, doc=None):
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001293 """Produce text documentation for a data object."""
1294 repr = self.repr(object)
1295 if maxlen:
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001296 line = (name and name + ' = ' or '') + repr
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001297 chop = maxlen - len(line)
1298 if chop < 0: repr = repr[:chop] + '...'
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001299 line = (name and self.bold(name) + ' = ' or '') + repr
Tim Peters28355492001-09-23 21:29:55 +00001300 if doc is not None:
1301 line += '\n' + self.indent(str(doc))
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001302 return line
1303
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001304# --------------------------------------------------------- user interfaces
1305
1306def pager(text):
1307 """The first time this is called, determine what kind of pager to use."""
1308 global pager
1309 pager = getpager()
1310 pager(text)
1311
1312def getpager():
1313 """Decide what method to use for paging through text."""
1314 if type(sys.stdout) is not types.FileType:
1315 return plainpager
1316 if not sys.stdin.isatty() or not sys.stdout.isatty():
1317 return plainpager
Raymond Hettinger54f02222002-06-01 14:18:47 +00001318 if 'PAGER' in os.environ:
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +00001319 if sys.platform == 'win32': # pipes completely broken in Windows
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001320 return lambda text: tempfilepager(plain(text), os.environ['PAGER'])
Raymond Hettingerdbecd932005-02-06 06:57:08 +00001321 elif os.environ.get('TERM') in ('dumb', 'emacs'):
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001322 return lambda text: pipepager(plain(text), os.environ['PAGER'])
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +00001323 else:
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001324 return lambda text: pipepager(text, os.environ['PAGER'])
Ka-Ping Yeea487e4e2005-11-05 04:49:18 +00001325 if os.environ.get('TERM') in ('dumb', 'emacs'):
1326 return plainpager
Andrew MacIntyre54e0eab2002-03-03 03:12:30 +00001327 if sys.platform == 'win32' or sys.platform.startswith('os2'):
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001328 return lambda text: tempfilepager(plain(text), 'more <')
Skip Montanarod404bee2002-09-26 21:44:57 +00001329 if hasattr(os, 'system') and os.system('(less) 2>/dev/null') == 0:
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001330 return lambda text: pipepager(text, 'less')
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001331
1332 import tempfile
Guido van Rossum3b0a3292002-08-09 16:38:32 +00001333 (fd, filename) = tempfile.mkstemp()
1334 os.close(fd)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001335 try:
Georg Brandlb32dea52008-07-16 21:19:28 +00001336 if hasattr(os, 'system') and os.system('more "%s"' % filename) == 0:
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001337 return lambda text: pipepager(text, 'more')
1338 else:
1339 return ttypager
1340 finally:
1341 os.unlink(filename)
1342
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001343def plain(text):
1344 """Remove boldface formatting from text."""
1345 return re.sub('.\b', '', text)
1346
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001347def pipepager(text, cmd):
1348 """Page through text by feeding it to another program."""
1349 pipe = os.popen(cmd, 'w')
1350 try:
1351 pipe.write(text)
1352 pipe.close()
1353 except IOError:
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001354 pass # Ignore broken pipes caused by quitting the pager program.
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001355
1356def tempfilepager(text, cmd):
1357 """Page through text by invoking a program on a temporary file."""
1358 import tempfile
Tim Peters550e4e52003-02-07 01:53:46 +00001359 filename = tempfile.mktemp()
1360 file = open(filename, 'w')
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001361 file.write(text)
1362 file.close()
1363 try:
Georg Brandlb32dea52008-07-16 21:19:28 +00001364 os.system(cmd + ' "' + filename + '"')
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001365 finally:
1366 os.unlink(filename)
1367
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001368def ttypager(text):
1369 """Page through text on a text terminal."""
1370 lines = split(plain(text), '\n')
1371 try:
1372 import tty
1373 fd = sys.stdin.fileno()
1374 old = tty.tcgetattr(fd)
1375 tty.setcbreak(fd)
1376 getchar = lambda: sys.stdin.read(1)
Ka-Ping Yee457aab22001-02-27 23:36:29 +00001377 except (ImportError, AttributeError):
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001378 tty = None
1379 getchar = lambda: sys.stdin.readline()[:-1][:1]
1380
1381 try:
1382 r = inc = os.environ.get('LINES', 25) - 1
1383 sys.stdout.write(join(lines[:inc], '\n') + '\n')
1384 while lines[r:]:
1385 sys.stdout.write('-- more --')
1386 sys.stdout.flush()
1387 c = getchar()
1388
Raymond Hettingerdbecd932005-02-06 06:57:08 +00001389 if c in ('q', 'Q'):
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001390 sys.stdout.write('\r \r')
1391 break
Raymond Hettingerdbecd932005-02-06 06:57:08 +00001392 elif c in ('\r', '\n'):
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001393 sys.stdout.write('\r \r' + lines[r] + '\n')
1394 r = r + 1
1395 continue
Raymond Hettingerdbecd932005-02-06 06:57:08 +00001396 if c in ('b', 'B', '\x1b'):
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001397 r = r - inc - inc
1398 if r < 0: r = 0
1399 sys.stdout.write('\n' + join(lines[r:r+inc], '\n') + '\n')
1400 r = r + inc
1401
1402 finally:
1403 if tty:
1404 tty.tcsetattr(fd, tty.TCSAFLUSH, old)
1405
1406def plainpager(text):
1407 """Simply print unformatted text. This is the ultimate fallback."""
1408 sys.stdout.write(plain(text))
1409
1410def describe(thing):
Ka-Ping Yee3bda8792001-03-23 13:17:50 +00001411 """Produce a short description of the given thing."""
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001412 if inspect.ismodule(thing):
1413 if thing.__name__ in sys.builtin_module_names:
1414 return 'built-in module ' + thing.__name__
1415 if hasattr(thing, '__path__'):
1416 return 'package ' + thing.__name__
1417 else:
1418 return 'module ' + thing.__name__
1419 if inspect.isbuiltin(thing):
1420 return 'built-in function ' + thing.__name__
Barry Warsaw00decd72006-07-27 23:43:15 +00001421 if inspect.isgetsetdescriptor(thing):
1422 return 'getset descriptor %s.%s.%s' % (
1423 thing.__objclass__.__module__, thing.__objclass__.__name__,
1424 thing.__name__)
1425 if inspect.ismemberdescriptor(thing):
1426 return 'member descriptor %s.%s.%s' % (
1427 thing.__objclass__.__module__, thing.__objclass__.__name__,
1428 thing.__name__)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001429 if inspect.isclass(thing):
1430 return 'class ' + thing.__name__
1431 if inspect.isfunction(thing):
1432 return 'function ' + thing.__name__
1433 if inspect.ismethod(thing):
1434 return 'method ' + thing.__name__
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001435 if type(thing) is types.InstanceType:
1436 return 'instance of ' + thing.__class__.__name__
1437 return type(thing).__name__
1438
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001439def locate(path, forceload=0):
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001440 """Locate an object by name or dotted path, importing as necessary."""
Guido van Rossum97dede02003-02-16 01:12:32 +00001441 parts = [part for part in split(path, '.') if part]
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001442 module, n = None, 0
1443 while n < len(parts):
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001444 nextmodule = safeimport(join(parts[:n+1], '.'), forceload)
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001445 if nextmodule: module, n = nextmodule, n + 1
1446 else: break
1447 if module:
1448 object = module
1449 for part in parts[n:]:
1450 try: object = getattr(object, part)
1451 except AttributeError: return None
1452 return object
1453 else:
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001454 if hasattr(__builtin__, path):
1455 return getattr(__builtin__, path)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001456
1457# --------------------------------------- interactive interpreter interface
1458
1459text = TextDoc()
1460html = HTMLDoc()
1461
Ka-Ping Yee8ef1cf32007-01-14 04:25:15 +00001462class _OldStyleClass: pass
1463_OLD_INSTANCE_TYPE = type(_OldStyleClass())
1464
Ka-Ping Yee45daeb02002-08-11 15:11:33 +00001465def resolve(thing, forceload=0):
1466 """Given an object or a path to an object, get the object and its name."""
1467 if isinstance(thing, str):
1468 object = locate(thing, forceload)
1469 if not object:
1470 raise ImportError, 'no Python documentation found for %r' % thing
1471 return object, thing
1472 else:
1473 return thing, getattr(thing, '__name__', None)
1474
Georg Brandl8441f152007-03-13 20:02:57 +00001475def render_doc(thing, title='Python Library Documentation: %s', forceload=0):
1476 """Render text documentation, given an object or a path to an object."""
1477 object, name = resolve(thing, forceload)
1478 desc = describe(object)
1479 module = inspect.getmodule(object)
1480 if name and '.' in name:
1481 desc += ' in ' + name[:name.rfind('.')]
1482 elif module and module is not object:
1483 desc += ' in module ' + module.__name__
1484 if type(object) is _OLD_INSTANCE_TYPE:
1485 # If the passed object is an instance of an old-style class,
1486 # document its available methods instead of its value.
1487 object = object.__class__
1488 elif not (inspect.ismodule(object) or
1489 inspect.isclass(object) or
1490 inspect.isroutine(object) or
1491 inspect.isgetsetdescriptor(object) or
1492 inspect.ismemberdescriptor(object) or
1493 isinstance(object, property)):
1494 # If the passed object is a piece of data or an instance,
1495 # document its available methods instead of its value.
1496 object = type(object)
1497 desc += ' object'
1498 return title % desc + '\n\n' + text.document(object, name)
1499
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001500def doc(thing, title='Python Library Documentation: %s', forceload=0):
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001501 """Display text documentation, given an object or a path to an object."""
Ka-Ping Yee45daeb02002-08-11 15:11:33 +00001502 try:
Georg Brandl8441f152007-03-13 20:02:57 +00001503 pager(render_doc(thing, title, forceload))
Ka-Ping Yee45daeb02002-08-11 15:11:33 +00001504 except (ImportError, ErrorDuringImport), value:
1505 print value
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001506
Ka-Ping Yee45daeb02002-08-11 15:11:33 +00001507def writedoc(thing, forceload=0):
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001508 """Write HTML documentation to a file in the current directory."""
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001509 try:
Ka-Ping Yee45daeb02002-08-11 15:11:33 +00001510 object, name = resolve(thing, forceload)
1511 page = html.page(describe(object), html.document(object, name))
1512 file = open(name + '.html', 'w')
1513 file.write(page)
1514 file.close()
1515 print 'wrote', name + '.html'
1516 except (ImportError, ErrorDuringImport), value:
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001517 print value
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001518
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001519def writedocs(dir, pkgpath='', done=None):
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001520 """Write out HTML documentation for all modules in a directory tree."""
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001521 if done is None: done = {}
Phillip J. Ebyceb30872006-04-18 00:59:55 +00001522 for importer, modname, ispkg in pkgutil.walk_packages([dir], pkgpath):
1523 writedoc(modname)
1524 return
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001525
1526class Helper:
Georg Brandl681001e2008-06-01 20:33:55 +00001527
1528 # These dictionaries map a topic name to either an alias, or a tuple
1529 # (label, seealso-items). The "label" is the label of the corresponding
1530 # section in the .rst file under Doc/ and an index into the dictionary
1531 # in pydoc_topics.py.
1532 #
1533 # CAUTION: if you change one of these dictionaries, be sure to adapt the
1534 # list of needed labels in Doc/tools/sphinxext/pyspecific.py and
1535 # regenerate the pydoc_topics.py file by running
1536 # make pydoc-topics
1537 # in Doc/ and copying the output file into the Lib/ directory.
1538
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001539 keywords = {
1540 'and': 'BOOLEAN',
Georg Brandlb6a87542007-03-13 10:06:48 +00001541 'as': 'with',
Georg Brandl681001e2008-06-01 20:33:55 +00001542 'assert': ('assert', ''),
1543 'break': ('break', 'while for'),
1544 'class': ('class', 'CLASSES SPECIALMETHODS'),
1545 'continue': ('continue', 'while for'),
1546 'def': ('function', ''),
1547 'del': ('del', 'BASICMETHODS'),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001548 'elif': 'if',
Georg Brandl681001e2008-06-01 20:33:55 +00001549 'else': ('else', 'while for'),
1550 'except': 'except',
1551 'exec': ('exec', ''),
1552 'finally': 'finally',
1553 'for': ('for', 'break continue while'),
1554 'from': 'from',
1555 'global': ('global', 'NAMESPACES'),
1556 'if': ('if', 'TRUTHVALUE'),
1557 'import': ('import', 'MODULES'),
1558 'in': ('in', 'SEQUENCEMETHODS2'),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001559 'is': 'COMPARISON',
Georg Brandl681001e2008-06-01 20:33:55 +00001560 'lambda': ('lambda', 'FUNCTIONS'),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001561 'not': 'BOOLEAN',
1562 'or': 'BOOLEAN',
Georg Brandl681001e2008-06-01 20:33:55 +00001563 'pass': ('pass', ''),
1564 'print': ('print', ''),
1565 'raise': ('raise', 'EXCEPTIONS'),
1566 'return': ('return', 'FUNCTIONS'),
1567 'try': ('try', 'EXCEPTIONS'),
1568 'while': ('while', 'break continue if TRUTHVALUE'),
1569 'with': ('with', 'CONTEXTMANAGERS EXCEPTIONS yield'),
1570 'yield': ('yield', ''),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001571 }
1572
1573 topics = {
Georg Brandl681001e2008-06-01 20:33:55 +00001574 'TYPES': ('types', 'STRINGS UNICODE NUMBERS SEQUENCES MAPPINGS '
1575 'FUNCTIONS CLASSES MODULES FILES inspect'),
1576 'STRINGS': ('strings', 'str UNICODE SEQUENCES STRINGMETHODS FORMATTING '
1577 'TYPES'),
1578 'STRINGMETHODS': ('string-methods', 'STRINGS FORMATTING'),
1579 'FORMATTING': ('formatstrings', 'OPERATORS'),
1580 'UNICODE': ('strings', 'encodings unicode SEQUENCES STRINGMETHODS '
1581 'FORMATTING TYPES'),
1582 'NUMBERS': ('numbers', 'INTEGER FLOAT COMPLEX TYPES'),
1583 'INTEGER': ('integers', 'int range'),
1584 'FLOAT': ('floating', 'float math'),
1585 'COMPLEX': ('imaginary', 'complex cmath'),
1586 'SEQUENCES': ('typesseq', 'STRINGMETHODS FORMATTING xrange LISTS'),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001587 'MAPPINGS': 'DICTIONARIES',
Georg Brandl681001e2008-06-01 20:33:55 +00001588 'FUNCTIONS': ('typesfunctions', 'def TYPES'),
1589 'METHODS': ('typesmethods', 'class def CLASSES TYPES'),
1590 'CODEOBJECTS': ('bltin-code-objects', 'compile FUNCTIONS TYPES'),
1591 'TYPEOBJECTS': ('bltin-type-objects', 'types TYPES'),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001592 'FRAMEOBJECTS': 'TYPES',
1593 'TRACEBACKS': 'TYPES',
Georg Brandl681001e2008-06-01 20:33:55 +00001594 'NONE': ('bltin-null-object', ''),
1595 'ELLIPSIS': ('bltin-ellipsis-object', 'SLICINGS'),
1596 'FILES': ('bltin-file-objects', ''),
1597 'SPECIALATTRIBUTES': ('specialattrs', ''),
1598 'CLASSES': ('types', 'class SPECIALMETHODS PRIVATENAMES'),
1599 'MODULES': ('typesmodules', 'import'),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001600 'PACKAGES': 'import',
Georg Brandl681001e2008-06-01 20:33:55 +00001601 'EXPRESSIONS': ('operator-summary', 'lambda or and not in is BOOLEAN '
1602 'COMPARISON BITWISE SHIFTING BINARY FORMATTING POWER '
1603 'UNARY ATTRIBUTES SUBSCRIPTS SLICINGS CALLS TUPLES '
1604 'LISTS DICTIONARIES BACKQUOTES'),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001605 'OPERATORS': 'EXPRESSIONS',
1606 'PRECEDENCE': 'EXPRESSIONS',
Georg Brandl681001e2008-06-01 20:33:55 +00001607 'OBJECTS': ('objects', 'TYPES'),
1608 'SPECIALMETHODS': ('specialnames', 'BASICMETHODS ATTRIBUTEMETHODS '
1609 'CALLABLEMETHODS SEQUENCEMETHODS1 MAPPINGMETHODS '
1610 'SEQUENCEMETHODS2 NUMBERMETHODS CLASSES'),
1611 'BASICMETHODS': ('customization', 'cmp hash repr str SPECIALMETHODS'),
1612 'ATTRIBUTEMETHODS': ('attribute-access', 'ATTRIBUTES SPECIALMETHODS'),
1613 'CALLABLEMETHODS': ('callable-types', 'CALLS SPECIALMETHODS'),
1614 'SEQUENCEMETHODS1': ('sequence-types', 'SEQUENCES SEQUENCEMETHODS2 '
1615 'SPECIALMETHODS'),
1616 'SEQUENCEMETHODS2': ('sequence-methods', 'SEQUENCES SEQUENCEMETHODS1 '
1617 'SPECIALMETHODS'),
1618 'MAPPINGMETHODS': ('sequence-types', 'MAPPINGS SPECIALMETHODS'),
1619 'NUMBERMETHODS': ('numeric-types', 'NUMBERS AUGMENTEDASSIGNMENT '
1620 'SPECIALMETHODS'),
1621 'EXECUTION': ('execmodel', 'NAMESPACES DYNAMICFEATURES EXCEPTIONS'),
1622 'NAMESPACES': ('naming', 'global ASSIGNMENT DELETION DYNAMICFEATURES'),
1623 'DYNAMICFEATURES': ('dynamic-features', ''),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001624 'SCOPING': 'NAMESPACES',
1625 'FRAMES': 'NAMESPACES',
Georg Brandl681001e2008-06-01 20:33:55 +00001626 'EXCEPTIONS': ('exceptions', 'try except finally raise'),
1627 'COERCIONS': ('coercion-rules','CONVERSIONS'),
1628 'CONVERSIONS': ('conversions', 'COERCIONS'),
1629 'IDENTIFIERS': ('identifiers', 'keywords SPECIALIDENTIFIERS'),
1630 'SPECIALIDENTIFIERS': ('id-classes', ''),
1631 'PRIVATENAMES': ('atom-identifiers', ''),
1632 'LITERALS': ('atom-literals', 'STRINGS BACKQUOTES NUMBERS '
1633 'TUPLELITERALS LISTLITERALS DICTIONARYLITERALS'),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001634 'TUPLES': 'SEQUENCES',
Georg Brandl681001e2008-06-01 20:33:55 +00001635 'TUPLELITERALS': ('exprlists', 'TUPLES LITERALS'),
1636 'LISTS': ('typesseq-mutable', 'LISTLITERALS'),
1637 'LISTLITERALS': ('lists', 'LISTS LITERALS'),
1638 'DICTIONARIES': ('typesmapping', 'DICTIONARYLITERALS'),
1639 'DICTIONARYLITERALS': ('dict', 'DICTIONARIES LITERALS'),
1640 'BACKQUOTES': ('string-conversions', 'repr str STRINGS LITERALS'),
1641 'ATTRIBUTES': ('attribute-references', 'getattr hasattr setattr '
1642 'ATTRIBUTEMETHODS'),
1643 'SUBSCRIPTS': ('subscriptions', 'SEQUENCEMETHODS1'),
1644 'SLICINGS': ('slicings', 'SEQUENCEMETHODS2'),
1645 'CALLS': ('calls', 'EXPRESSIONS'),
1646 'POWER': ('power', 'EXPRESSIONS'),
1647 'UNARY': ('unary', 'EXPRESSIONS'),
1648 'BINARY': ('binary', 'EXPRESSIONS'),
1649 'SHIFTING': ('shifting', 'EXPRESSIONS'),
1650 'BITWISE': ('bitwise', 'EXPRESSIONS'),
1651 'COMPARISON': ('comparisons', 'EXPRESSIONS BASICMETHODS'),
1652 'BOOLEAN': ('booleans', 'EXPRESSIONS TRUTHVALUE'),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001653 'ASSERTION': 'assert',
Georg Brandl681001e2008-06-01 20:33:55 +00001654 'ASSIGNMENT': ('assignment', 'AUGMENTEDASSIGNMENT'),
1655 'AUGMENTEDASSIGNMENT': ('augassign', 'NUMBERMETHODS'),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001656 'DELETION': 'del',
1657 'PRINTING': 'print',
1658 'RETURNING': 'return',
1659 'IMPORTING': 'import',
1660 'CONDITIONAL': 'if',
Georg Brandl681001e2008-06-01 20:33:55 +00001661 'LOOPING': ('compound', 'for while break continue'),
1662 'TRUTHVALUE': ('truth', 'if while and or not BASICMETHODS'),
1663 'DEBUGGING': ('debugger', 'pdb'),
1664 'CONTEXTMANAGERS': ('context-managers', 'with'),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001665 }
1666
1667 def __init__(self, input, output):
1668 self.input = input
1669 self.output = output
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001670
Ka-Ping Yee79c009d2001-04-13 10:53:25 +00001671 def __repr__(self):
Ka-Ping Yee9bc576b2001-04-13 13:57:31 +00001672 if inspect.stack()[1][3] == '?':
Ka-Ping Yee79c009d2001-04-13 10:53:25 +00001673 self()
1674 return ''
Ka-Ping Yee9bc576b2001-04-13 13:57:31 +00001675 return '<pydoc.Helper instance>'
Ka-Ping Yee79c009d2001-04-13 10:53:25 +00001676
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001677 def __call__(self, request=None):
1678 if request is not None:
1679 self.help(request)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001680 else:
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001681 self.intro()
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001682 self.interact()
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001683 self.output.write('''
Fred Drakee61967f2001-05-10 18:41:02 +00001684You are now leaving help and returning to the Python interpreter.
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001685If you want to ask for help on a particular object directly from the
1686interpreter, you can type "help(object)". Executing "help('string')"
1687has the same effect as typing a particular string at the help> prompt.
1688''')
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001689
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001690 def interact(self):
1691 self.output.write('\n')
Guido van Rossum8ca162f2002-04-07 06:36:23 +00001692 while True:
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001693 try:
Johannes Gijsberse7691d32004-08-17 13:21:53 +00001694 request = self.getline('help> ')
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001695 if not request: break
Johannes Gijsberse7691d32004-08-17 13:21:53 +00001696 except (KeyboardInterrupt, EOFError):
1697 break
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001698 request = strip(replace(request, '"', '', "'", ''))
Raymond Hettingerdbecd932005-02-06 06:57:08 +00001699 if lower(request) in ('q', 'quit'): break
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001700 self.help(request)
1701
Johannes Gijsberse7691d32004-08-17 13:21:53 +00001702 def getline(self, prompt):
1703 """Read one line, using raw_input when available."""
1704 if self.input is sys.stdin:
1705 return raw_input(prompt)
1706 else:
1707 self.output.write(prompt)
1708 self.output.flush()
1709 return self.input.readline()
1710
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001711 def help(self, request):
1712 if type(request) is type(''):
1713 if request == 'help': self.intro()
1714 elif request == 'keywords': self.listkeywords()
1715 elif request == 'topics': self.listtopics()
1716 elif request == 'modules': self.listmodules()
1717 elif request[:8] == 'modules ':
1718 self.listmodules(split(request)[1])
Raymond Hettinger54f02222002-06-01 14:18:47 +00001719 elif request in self.keywords: self.showtopic(request)
1720 elif request in self.topics: self.showtopic(request)
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001721 elif request: doc(request, 'Help on %s:')
1722 elif isinstance(request, Helper): self()
1723 else: doc(request, 'Help on %s:')
1724 self.output.write('\n')
1725
1726 def intro(self):
1727 self.output.write('''
1728Welcome to Python %s! This is the online help utility.
1729
1730If this is your first time using Python, you should definitely check out
Georg Brandl0751d1a2008-01-21 17:13:03 +00001731the tutorial on the Internet at http://docs.python.org/tutorial/.
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001732
1733Enter the name of any module, keyword, or topic to get help on writing
1734Python programs and using Python modules. To quit this help utility and
1735return to the interpreter, just type "quit".
1736
1737To get a list of available modules, keywords, or topics, type "modules",
1738"keywords", or "topics". Each module also comes with a one-line summary
1739of what it does; to list the modules whose summaries contain a given word
1740such as "spam", type "modules spam".
1741''' % sys.version[:3])
1742
1743 def list(self, items, columns=4, width=80):
1744 items = items[:]
1745 items.sort()
1746 colw = width / columns
1747 rows = (len(items) + columns - 1) / columns
1748 for row in range(rows):
1749 for col in range(columns):
1750 i = col * rows + row
1751 if i < len(items):
1752 self.output.write(items[i])
1753 if col < columns - 1:
1754 self.output.write(' ' + ' ' * (colw-1 - len(items[i])))
1755 self.output.write('\n')
1756
1757 def listkeywords(self):
1758 self.output.write('''
1759Here is a list of the Python keywords. Enter any keyword to get more help.
1760
1761''')
1762 self.list(self.keywords.keys())
1763
1764 def listtopics(self):
1765 self.output.write('''
1766Here is a list of available topics. Enter any topic name to get more help.
1767
1768''')
1769 self.list(self.topics.keys())
1770
1771 def showtopic(self, topic):
Georg Brandl681001e2008-06-01 20:33:55 +00001772 try:
1773 import pydoc_topics
1774 except ImportError:
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001775 self.output.write('''
Georg Brandl681001e2008-06-01 20:33:55 +00001776Sorry, topic and keyword documentation is not available because the
1777module "pydoc_topics" could not be found.
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001778''')
1779 return
1780 target = self.topics.get(topic, self.keywords.get(topic))
1781 if not target:
1782 self.output.write('no documentation found for %s\n' % repr(topic))
1783 return
1784 if type(target) is type(''):
1785 return self.showtopic(target)
1786
Georg Brandl681001e2008-06-01 20:33:55 +00001787 label, xrefs = target
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001788 try:
Georg Brandl681001e2008-06-01 20:33:55 +00001789 doc = pydoc_topics.topics[label]
1790 except KeyError:
1791 self.output.write('no documentation found for %s\n' % repr(topic))
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001792 return
Georg Brandl681001e2008-06-01 20:33:55 +00001793 pager(strip(doc) + '\n')
Ka-Ping Yeeda793892001-04-13 11:02:51 +00001794 if xrefs:
Georg Brandl681001e2008-06-01 20:33:55 +00001795 import StringIO, formatter
Ka-Ping Yeeda793892001-04-13 11:02:51 +00001796 buffer = StringIO.StringIO()
1797 formatter.DumbWriter(buffer).send_flowing_data(
1798 'Related help topics: ' + join(split(xrefs), ', ') + '\n')
1799 self.output.write('\n%s\n' % buffer.getvalue())
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001800
1801 def listmodules(self, key=''):
1802 if key:
1803 self.output.write('''
1804Here is a list of matching modules. Enter any module name to get more help.
1805
1806''')
1807 apropos(key)
1808 else:
1809 self.output.write('''
1810Please wait a moment while I gather a list of all available modules...
1811
1812''')
1813 modules = {}
1814 def callback(path, modname, desc, modules=modules):
1815 if modname and modname[-9:] == '.__init__':
1816 modname = modname[:-9] + ' (package)'
1817 if find(modname, '.') < 0:
1818 modules[modname] = 1
Ka-Ping Yee9e0f1162008-01-13 11:25:13 +00001819 def onerror(modname):
1820 callback(None, modname, None)
1821 ModuleScanner().run(callback, onerror=onerror)
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001822 self.list(modules.keys())
1823 self.output.write('''
1824Enter any module name to get more help. Or, type "modules spam" to search
1825for modules whose descriptions contain the word "spam".
1826''')
1827
1828help = Helper(sys.stdin, sys.stdout)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001829
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001830class Scanner:
1831 """A generic tree iterator."""
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001832 def __init__(self, roots, children, descendp):
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001833 self.roots = roots[:]
1834 self.state = []
1835 self.children = children
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001836 self.descendp = descendp
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001837
1838 def next(self):
1839 if not self.state:
1840 if not self.roots:
1841 return None
1842 root = self.roots.pop(0)
1843 self.state = [(root, self.children(root))]
1844 node, children = self.state[-1]
1845 if not children:
1846 self.state.pop()
1847 return self.next()
1848 child = children.pop(0)
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001849 if self.descendp(child):
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001850 self.state.append((child, self.children(child)))
1851 return child
1852
Phillip J. Ebyceb30872006-04-18 00:59:55 +00001853
1854class ModuleScanner:
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001855 """An interruptible scanner that searches module synopses."""
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001856
Ka-Ping Yee9e0f1162008-01-13 11:25:13 +00001857 def run(self, callback, key=None, completer=None, onerror=None):
Ka-Ping Yee66246962001-04-12 11:59:50 +00001858 if key: key = lower(key)
Guido van Rossum8ca162f2002-04-07 06:36:23 +00001859 self.quit = False
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001860 seen = {}
1861
1862 for modname in sys.builtin_module_names:
Ka-Ping Yee239432a2001-03-02 02:45:08 +00001863 if modname != '__main__':
1864 seen[modname] = 1
Ka-Ping Yee66246962001-04-12 11:59:50 +00001865 if key is None:
1866 callback(None, modname, '')
1867 else:
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001868 desc = split(__import__(modname).__doc__ or '', '\n')[0]
Ka-Ping Yee66246962001-04-12 11:59:50 +00001869 if find(lower(modname + ' - ' + desc), key) >= 0:
1870 callback(None, modname, desc)
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001871
Ka-Ping Yee9e0f1162008-01-13 11:25:13 +00001872 for importer, modname, ispkg in pkgutil.walk_packages(onerror=onerror):
Phillip J. Ebyceb30872006-04-18 00:59:55 +00001873 if self.quit:
1874 break
1875 if key is None:
1876 callback(None, modname, '')
1877 else:
1878 loader = importer.find_module(modname)
1879 if hasattr(loader,'get_source'):
1880 import StringIO
1881 desc = source_synopsis(
1882 StringIO.StringIO(loader.get_source(modname))
1883 ) or ''
1884 if hasattr(loader,'get_filename'):
1885 path = loader.get_filename(modname)
Ka-Ping Yee66246962001-04-12 11:59:50 +00001886 else:
Phillip J. Ebyceb30872006-04-18 00:59:55 +00001887 path = None
1888 else:
1889 module = loader.load_module(modname)
1890 desc = (module.__doc__ or '').splitlines()[0]
1891 path = getattr(module,'__file__',None)
1892 if find(lower(modname + ' - ' + desc), key) >= 0:
1893 callback(path, modname, desc)
1894
1895 if completer:
1896 completer()
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001897
1898def apropos(key):
1899 """Print all the one-line module summaries that contain a substring."""
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001900 def callback(path, modname, desc):
1901 if modname[-9:] == '.__init__':
1902 modname = modname[:-9] + ' (package)'
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +00001903 print modname, desc and '- ' + desc
1904 try: import warnings
1905 except ImportError: pass
1906 else: warnings.filterwarnings('ignore') # ignore problems during import
Ka-Ping Yee66246962001-04-12 11:59:50 +00001907 ModuleScanner().run(callback, key)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001908
1909# --------------------------------------------------- web browser interface
1910
Ka-Ping Yee66246962001-04-12 11:59:50 +00001911def serve(port, callback=None, completer=None):
Ka-Ping Yeefd540692001-04-12 12:54:36 +00001912 import BaseHTTPServer, mimetools, select
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001913
1914 # Patch up mimetools.Message so it doesn't break if rfc822 is reloaded.
1915 class Message(mimetools.Message):
1916 def __init__(self, fp, seekable=1):
1917 Message = self.__class__
1918 Message.__bases__[0].__bases__[0].__init__(self, fp, seekable)
1919 self.encodingheader = self.getheader('content-transfer-encoding')
1920 self.typeheader = self.getheader('content-type')
1921 self.parsetype()
1922 self.parseplist()
1923
1924 class DocHandler(BaseHTTPServer.BaseHTTPRequestHandler):
1925 def send_document(self, title, contents):
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001926 try:
1927 self.send_response(200)
1928 self.send_header('Content-Type', 'text/html')
1929 self.end_headers()
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001930 self.wfile.write(html.page(title, contents))
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001931 except IOError: pass
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001932
1933 def do_GET(self):
1934 path = self.path
1935 if path[-5:] == '.html': path = path[:-5]
1936 if path[:1] == '/': path = path[1:]
1937 if path and path != '.':
1938 try:
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001939 obj = locate(path, forceload=1)
Ka-Ping Yee3bda8792001-03-23 13:17:50 +00001940 except ErrorDuringImport, value:
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001941 self.send_document(path, html.escape(str(value)))
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001942 return
Ka-Ping Yee3bda8792001-03-23 13:17:50 +00001943 if obj:
1944 self.send_document(describe(obj), html.document(obj, path))
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001945 else:
1946 self.send_document(path,
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001947'no Python documentation found for %s' % repr(path))
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001948 else:
1949 heading = html.heading(
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001950'<big><big><strong>Python: Index of Modules</strong></big></big>',
1951'#ffffff', '#7799ee')
Ka-Ping Yee239432a2001-03-02 02:45:08 +00001952 def bltinlink(name):
1953 return '<a href="%s.html">%s</a>' % (name, name)
Ka-Ping Yee3bda8792001-03-23 13:17:50 +00001954 names = filter(lambda x: x != '__main__',
1955 sys.builtin_module_names)
Ka-Ping Yee239432a2001-03-02 02:45:08 +00001956 contents = html.multicolumn(names, bltinlink)
1957 indices = ['<p>' + html.bigsection(
1958 'Built-in Modules', '#ffffff', '#ee77aa', contents)]
1959
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001960 seen = {}
Phillip J. Ebyceb30872006-04-18 00:59:55 +00001961 for dir in sys.path:
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001962 indices.append(html.index(dir, seen))
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001963 contents = heading + join(indices) + '''<p align=right>
Tim Peters2306d242001-09-25 03:18:32 +00001964<font color="#909090" face="helvetica, arial"><strong>
1965pydoc</strong> by Ka-Ping Yee &lt;ping@lfw.org&gt;</font>'''
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001966 self.send_document('Index of Modules', contents)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001967
1968 def log_message(self, *args): pass
1969
Ka-Ping Yeefd540692001-04-12 12:54:36 +00001970 class DocServer(BaseHTTPServer.HTTPServer):
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001971 def __init__(self, port, callback):
Ka-Ping Yeedb8ed152001-03-02 05:58:17 +00001972 host = (sys.platform == 'mac') and '127.0.0.1' or 'localhost'
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001973 self.address = ('', port)
Ka-Ping Yeedb8ed152001-03-02 05:58:17 +00001974 self.url = 'http://%s:%d/' % (host, port)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001975 self.callback = callback
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001976 self.base.__init__(self, self.address, self.handler)
1977
1978 def serve_until_quit(self):
1979 import select
Guido van Rossum8ca162f2002-04-07 06:36:23 +00001980 self.quit = False
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001981 while not self.quit:
1982 rd, wr, ex = select.select([self.socket.fileno()], [], [], 1)
1983 if rd: self.handle_request()
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001984
1985 def server_activate(self):
1986 self.base.server_activate(self)
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001987 if self.callback: self.callback(self)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001988
1989 DocServer.base = BaseHTTPServer.HTTPServer
1990 DocServer.handler = DocHandler
1991 DocHandler.MessageClass = Message
1992 try:
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001993 try:
1994 DocServer(port, callback).serve_until_quit()
1995 except (KeyboardInterrupt, select.error):
1996 pass
1997 finally:
Ka-Ping Yee66246962001-04-12 11:59:50 +00001998 if completer: completer()
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001999
2000# ----------------------------------------------------- graphical interface
2001
2002def gui():
2003 """Graphical interface (starts web server and pops up a control window)."""
2004 class GUI:
2005 def __init__(self, window, port=7464):
2006 self.window = window
2007 self.server = None
2008 self.scanner = None
2009
Georg Brandl6634bf22008-05-20 07:13:37 +00002010 import Tkinter
2011 self.server_frm = Tkinter.Frame(window)
2012 self.title_lbl = Tkinter.Label(self.server_frm,
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002013 text='Starting server...\n ')
Georg Brandl6634bf22008-05-20 07:13:37 +00002014 self.open_btn = Tkinter.Button(self.server_frm,
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002015 text='open browser', command=self.open, state='disabled')
Georg Brandl6634bf22008-05-20 07:13:37 +00002016 self.quit_btn = Tkinter.Button(self.server_frm,
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002017 text='quit serving', command=self.quit, state='disabled')
2018
Georg Brandl6634bf22008-05-20 07:13:37 +00002019 self.search_frm = Tkinter.Frame(window)
2020 self.search_lbl = Tkinter.Label(self.search_frm, text='Search for')
2021 self.search_ent = Tkinter.Entry(self.search_frm)
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002022 self.search_ent.bind('<Return>', self.search)
Georg Brandl6634bf22008-05-20 07:13:37 +00002023 self.stop_btn = Tkinter.Button(self.search_frm,
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002024 text='stop', pady=0, command=self.stop, state='disabled')
2025 if sys.platform == 'win32':
Ka-Ping Yeec92cdf72001-03-02 05:54:35 +00002026 # Trying to hide and show this button crashes under Windows.
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002027 self.stop_btn.pack(side='right')
2028
2029 self.window.title('pydoc')
2030 self.window.protocol('WM_DELETE_WINDOW', self.quit)
2031 self.title_lbl.pack(side='top', fill='x')
2032 self.open_btn.pack(side='left', fill='x', expand=1)
2033 self.quit_btn.pack(side='right', fill='x', expand=1)
2034 self.server_frm.pack(side='top', fill='x')
2035
2036 self.search_lbl.pack(side='left')
2037 self.search_ent.pack(side='right', fill='x', expand=1)
2038 self.search_frm.pack(side='top', fill='x')
2039 self.search_ent.focus_set()
2040
Ka-Ping Yeec92cdf72001-03-02 05:54:35 +00002041 font = ('helvetica', sys.platform == 'win32' and 8 or 10)
Georg Brandl6634bf22008-05-20 07:13:37 +00002042 self.result_lst = Tkinter.Listbox(window, font=font, height=6)
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002043 self.result_lst.bind('<Button-1>', self.select)
2044 self.result_lst.bind('<Double-Button-1>', self.goto)
Georg Brandl6634bf22008-05-20 07:13:37 +00002045 self.result_scr = Tkinter.Scrollbar(window,
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002046 orient='vertical', command=self.result_lst.yview)
2047 self.result_lst.config(yscrollcommand=self.result_scr.set)
2048
Georg Brandl6634bf22008-05-20 07:13:37 +00002049 self.result_frm = Tkinter.Frame(window)
2050 self.goto_btn = Tkinter.Button(self.result_frm,
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002051 text='go to selected', command=self.goto)
Georg Brandl6634bf22008-05-20 07:13:37 +00002052 self.hide_btn = Tkinter.Button(self.result_frm,
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002053 text='hide results', command=self.hide)
2054 self.goto_btn.pack(side='left', fill='x', expand=1)
2055 self.hide_btn.pack(side='right', fill='x', expand=1)
2056
2057 self.window.update()
2058 self.minwidth = self.window.winfo_width()
2059 self.minheight = self.window.winfo_height()
2060 self.bigminheight = (self.server_frm.winfo_reqheight() +
2061 self.search_frm.winfo_reqheight() +
2062 self.result_lst.winfo_reqheight() +
2063 self.result_frm.winfo_reqheight())
2064 self.bigwidth, self.bigheight = self.minwidth, self.bigminheight
2065 self.expanded = 0
2066 self.window.wm_geometry('%dx%d' % (self.minwidth, self.minheight))
2067 self.window.wm_minsize(self.minwidth, self.minheight)
Martin v. Löwis5b26abb2002-12-28 09:23:09 +00002068 self.window.tk.willdispatch()
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002069
2070 import threading
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00002071 threading.Thread(
2072 target=serve, args=(port, self.ready, self.quit)).start()
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002073
2074 def ready(self, server):
2075 self.server = server
2076 self.title_lbl.config(
2077 text='Python documentation server at\n' + server.url)
2078 self.open_btn.config(state='normal')
2079 self.quit_btn.config(state='normal')
2080
Ka-Ping Yee239432a2001-03-02 02:45:08 +00002081 def open(self, event=None, url=None):
2082 url = url or self.server.url
2083 try:
2084 import webbrowser
2085 webbrowser.open(url)
2086 except ImportError: # pre-webbrowser.py compatibility
Ka-Ping Yeec92cdf72001-03-02 05:54:35 +00002087 if sys.platform == 'win32':
Ka-Ping Yee239432a2001-03-02 02:45:08 +00002088 os.system('start "%s"' % url)
2089 elif sys.platform == 'mac':
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +00002090 try: import ic
Ka-Ping Yee239432a2001-03-02 02:45:08 +00002091 except ImportError: pass
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +00002092 else: ic.launchurl(url)
Ka-Ping Yee239432a2001-03-02 02:45:08 +00002093 else:
2094 rc = os.system('netscape -remote "openURL(%s)" &' % url)
2095 if rc: os.system('netscape "%s" &' % url)
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002096
2097 def quit(self, event=None):
2098 if self.server:
2099 self.server.quit = 1
2100 self.window.quit()
2101
2102 def search(self, event=None):
2103 key = self.search_ent.get()
2104 self.stop_btn.pack(side='right')
2105 self.stop_btn.config(state='normal')
2106 self.search_lbl.config(text='Searching for "%s"...' % key)
2107 self.search_ent.forget()
2108 self.search_lbl.pack(side='left')
2109 self.result_lst.delete(0, 'end')
2110 self.goto_btn.config(state='disabled')
2111 self.expand()
2112
2113 import threading
2114 if self.scanner:
2115 self.scanner.quit = 1
2116 self.scanner = ModuleScanner()
2117 threading.Thread(target=self.scanner.run,
Ka-Ping Yee6dcfa382001-04-12 20:27:31 +00002118 args=(self.update, key, self.done)).start()
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002119
2120 def update(self, path, modname, desc):
2121 if modname[-9:] == '.__init__':
2122 modname = modname[:-9] + ' (package)'
2123 self.result_lst.insert('end',
2124 modname + ' - ' + (desc or '(no description)'))
2125
2126 def stop(self, event=None):
2127 if self.scanner:
2128 self.scanner.quit = 1
2129 self.scanner = None
2130
2131 def done(self):
2132 self.scanner = None
2133 self.search_lbl.config(text='Search for')
2134 self.search_lbl.pack(side='left')
2135 self.search_ent.pack(side='right', fill='x', expand=1)
2136 if sys.platform != 'win32': self.stop_btn.forget()
2137 self.stop_btn.config(state='disabled')
2138
2139 def select(self, event=None):
2140 self.goto_btn.config(state='normal')
2141
2142 def goto(self, event=None):
2143 selection = self.result_lst.curselection()
2144 if selection:
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002145 modname = split(self.result_lst.get(selection[0]))[0]
Ka-Ping Yee239432a2001-03-02 02:45:08 +00002146 self.open(url=self.server.url + modname + '.html')
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002147
2148 def collapse(self):
2149 if not self.expanded: return
2150 self.result_frm.forget()
2151 self.result_scr.forget()
2152 self.result_lst.forget()
2153 self.bigwidth = self.window.winfo_width()
2154 self.bigheight = self.window.winfo_height()
2155 self.window.wm_geometry('%dx%d' % (self.minwidth, self.minheight))
2156 self.window.wm_minsize(self.minwidth, self.minheight)
2157 self.expanded = 0
2158
2159 def expand(self):
2160 if self.expanded: return
2161 self.result_frm.pack(side='bottom', fill='x')
2162 self.result_scr.pack(side='right', fill='y')
2163 self.result_lst.pack(side='top', fill='both', expand=1)
2164 self.window.wm_geometry('%dx%d' % (self.bigwidth, self.bigheight))
2165 self.window.wm_minsize(self.minwidth, self.bigminheight)
2166 self.expanded = 1
2167
2168 def hide(self, event=None):
2169 self.stop()
2170 self.collapse()
2171
Georg Brandl6634bf22008-05-20 07:13:37 +00002172 import Tkinter
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002173 try:
Georg Brandl6634bf22008-05-20 07:13:37 +00002174 root = Tkinter.Tk()
Martin v. Löwise09bd932004-08-22 16:13:26 +00002175 # Tk will crash if pythonw.exe has an XP .manifest
2176 # file and the root has is not destroyed explicitly.
2177 # If the problem is ever fixed in Tk, the explicit
2178 # destroy can go.
2179 try:
2180 gui = GUI(root)
2181 root.mainloop()
2182 finally:
2183 root.destroy()
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002184 except KeyboardInterrupt:
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002185 pass
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002186
2187# -------------------------------------------------- command-line interface
2188
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00002189def ispath(x):
Ka-Ping Yee45daeb02002-08-11 15:11:33 +00002190 return isinstance(x, str) and find(x, os.sep) >= 0
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00002191
Ka-Ping Yee1d384632001-03-01 00:24:32 +00002192def cli():
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002193 """Command-line interface (looks at sys.argv to decide what to do)."""
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002194 import getopt
2195 class BadUsage: pass
2196
Ka-Ping Yee3bda8792001-03-23 13:17:50 +00002197 # Scripts don't get the current directory in their path by default.
Ka-Ping Yeef78a81b2001-03-27 08:13:42 +00002198 scriptdir = os.path.dirname(sys.argv[0])
2199 if scriptdir in sys.path:
2200 sys.path.remove(scriptdir)
Ka-Ping Yee3bda8792001-03-23 13:17:50 +00002201 sys.path.insert(0, '.')
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002202
Ka-Ping Yee3bda8792001-03-23 13:17:50 +00002203 try:
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002204 opts, args = getopt.getopt(sys.argv[1:], 'gk:p:w')
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002205 writing = 0
2206
2207 for opt, val in opts:
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002208 if opt == '-g':
2209 gui()
2210 return
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002211 if opt == '-k':
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002212 apropos(val)
2213 return
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002214 if opt == '-p':
2215 try:
2216 port = int(val)
2217 except ValueError:
2218 raise BadUsage
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002219 def ready(server):
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00002220 print 'pydoc server ready at %s' % server.url
2221 def stopped():
2222 print 'pydoc server stopped'
2223 serve(port, ready, stopped)
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002224 return
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002225 if opt == '-w':
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002226 writing = 1
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002227
2228 if not args: raise BadUsage
2229 for arg in args:
Ka-Ping Yee45daeb02002-08-11 15:11:33 +00002230 if ispath(arg) and not os.path.exists(arg):
2231 print 'file %r does not exist' % arg
2232 break
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002233 try:
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00002234 if ispath(arg) and os.path.isfile(arg):
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002235 arg = importfile(arg)
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00002236 if writing:
2237 if ispath(arg) and os.path.isdir(arg):
2238 writedocs(arg)
2239 else:
2240 writedoc(arg)
2241 else:
Martin v. Löwisb8c084e2003-06-14 09:03:46 +00002242 help.help(arg)
Ka-Ping Yee3bda8792001-03-23 13:17:50 +00002243 except ErrorDuringImport, value:
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00002244 print value
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002245
2246 except (getopt.error, BadUsage):
Martin v. Löwisf9b08b82003-10-31 13:05:21 +00002247 cmd = os.path.basename(sys.argv[0])
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002248 print """pydoc - the Python documentation tool
2249
2250%s <name> ...
2251 Show text documentation on something. <name> may be the name of a
Martin v. Löwisb8c084e2003-06-14 09:03:46 +00002252 Python keyword, topic, function, module, or package, or a dotted
2253 reference to a class or function within a module or module in a
2254 package. If <name> contains a '%s', it is used as the path to a
2255 Python source file to document. If name is 'keywords', 'topics',
2256 or 'modules', a listing of these things is displayed.
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002257
2258%s -k <keyword>
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002259 Search for a keyword in the synopsis lines of all available modules.
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002260
2261%s -p <port>
2262 Start an HTTP server on the given port on the local machine.
2263
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002264%s -g
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +00002265 Pop up a graphical interface for finding and serving documentation.
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002266
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002267%s -w <name> ...
2268 Write out the HTML documentation for a module to a file in the current
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +00002269 directory. If <name> contains a '%s', it is treated as a filename; if
2270 it names a directory, documentation is written for all the contents.
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002271""" % (cmd, os.sep, cmd, cmd, cmd, cmd, os.sep)
Ka-Ping Yee1d384632001-03-01 00:24:32 +00002272
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002273if __name__ == '__main__': cli()