blob: 86a6aa02c1ebc347bcd8655f541f8fc891566eea [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
R. David Murrayc4c84e12009-06-23 20:37:26 +000058from traceback import extract_tb
Phillip J. Ebyceb30872006-04-18 00:59:55 +000059try:
60 from collections import deque
61except ImportError:
62 # Python 2.3 compatibility
63 class deque(list):
64 def popleft(self):
65 return self.pop(0)
Ka-Ping Yeedd175342001-02-27 14:43:46 +000066
67# --------------------------------------------------------- common routines
68
Ka-Ping Yeedd175342001-02-27 14:43:46 +000069def pathdirs():
70 """Convert sys.path into a list of absolute, existing, unique paths."""
71 dirs = []
Ka-Ping Yee1d384632001-03-01 00:24:32 +000072 normdirs = []
Ka-Ping Yeedd175342001-02-27 14:43:46 +000073 for dir in sys.path:
74 dir = os.path.abspath(dir or '.')
Ka-Ping Yee1d384632001-03-01 00:24:32 +000075 normdir = os.path.normcase(dir)
76 if normdir not in normdirs and os.path.isdir(dir):
Ka-Ping Yeedd175342001-02-27 14:43:46 +000077 dirs.append(dir)
Ka-Ping Yee1d384632001-03-01 00:24:32 +000078 normdirs.append(normdir)
Ka-Ping Yeedd175342001-02-27 14:43:46 +000079 return dirs
80
81def getdoc(object):
82 """Get the doc string or comments for an object."""
Ka-Ping Yee3bda8792001-03-23 13:17:50 +000083 result = inspect.getdoc(object) or inspect.getcomments(object)
Ka-Ping Yee239432a2001-03-02 02:45:08 +000084 return result and re.sub('^ *\n', '', rstrip(result)) or ''
Ka-Ping Yeedd175342001-02-27 14:43:46 +000085
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +000086def splitdoc(doc):
87 """Split a doc string into a synopsis line (if any) and the rest."""
88 lines = split(strip(doc), '\n')
89 if len(lines) == 1:
90 return lines[0], ''
91 elif len(lines) >= 2 and not rstrip(lines[1]):
92 return lines[0], join(lines[2:], '\n')
93 return '', join(lines, '\n')
94
Ka-Ping Yeedd175342001-02-27 14:43:46 +000095def classname(object, modname):
96 """Get a class name and qualify it with a module name if necessary."""
97 name = object.__name__
98 if object.__module__ != modname:
99 name = object.__module__ + '.' + name
100 return name
101
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000102def isdata(object):
Georg Brandl94432422005-07-22 21:52:25 +0000103 """Check if an object is of a type that probably means it's data."""
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000104 return not (inspect.ismodule(object) or inspect.isclass(object) or
105 inspect.isroutine(object) or inspect.isframe(object) or
106 inspect.istraceback(object) or inspect.iscode(object))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000107
108def replace(text, *pairs):
109 """Do a series of global replacements on a string."""
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +0000110 while pairs:
111 text = join(split(text, pairs[0]), pairs[1])
112 pairs = pairs[2:]
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000113 return text
114
115def cram(text, maxlen):
116 """Omit part of a string if needed to make it fit in a maximum length."""
117 if len(text) > maxlen:
Raymond Hettingerfca3bb62002-10-21 04:44:11 +0000118 pre = max(0, (maxlen-3)//2)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000119 post = max(0, maxlen-3-pre)
120 return text[:pre] + '...' + text[len(text)-post:]
121 return text
122
Brett Cannon84601f12004-06-19 01:22:48 +0000123_re_stripid = re.compile(r' at 0x[0-9a-f]{6,16}(>+)$', re.IGNORECASE)
Ka-Ping Yee1d384632001-03-01 00:24:32 +0000124def stripid(text):
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000125 """Remove the hexadecimal id from a Python object representation."""
Brett Cannonc6c1f472004-06-19 01:02:51 +0000126 # The behaviour of %p is implementation-dependent in terms of case.
Ezio Melotti5f943ed2010-02-16 23:29:44 +0000127 return _re_stripid.sub(r'\1', text)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000128
Brett Cannonc6c1f472004-06-19 01:02:51 +0000129def _is_some_method(obj):
130 return inspect.ismethod(obj) or inspect.ismethoddescriptor(obj)
Tim Peters536d2262001-09-20 05:13:38 +0000131
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000132def allmethods(cl):
133 methods = {}
Tim Peters536d2262001-09-20 05:13:38 +0000134 for key, value in inspect.getmembers(cl, _is_some_method):
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000135 methods[key] = 1
136 for base in cl.__bases__:
137 methods.update(allmethods(base)) # all your base are belong to us
138 for key in methods.keys():
139 methods[key] = getattr(cl, key)
140 return methods
141
Tim Petersfa26f7c2001-09-24 08:05:11 +0000142def _split_list(s, predicate):
143 """Split sequence s via predicate, and return pair ([true], [false]).
144
145 The return value is a 2-tuple of lists,
146 ([x for x in s if predicate(x)],
147 [x for x in s if not predicate(x)])
148 """
149
Tim Peters28355492001-09-23 21:29:55 +0000150 yes = []
151 no = []
Tim Petersfa26f7c2001-09-24 08:05:11 +0000152 for x in s:
153 if predicate(x):
154 yes.append(x)
Tim Peters28355492001-09-23 21:29:55 +0000155 else:
Tim Petersfa26f7c2001-09-24 08:05:11 +0000156 no.append(x)
Tim Peters28355492001-09-23 21:29:55 +0000157 return yes, no
158
Skip Montanaroa5616d22004-06-11 04:46:12 +0000159def visiblename(name, all=None):
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000160 """Decide whether to show documentation on a variable."""
161 # Certain special names are redundant.
Nick Coghlanc060b0e2008-07-02 13:09:19 +0000162 _hidden_names = ('__builtins__', '__doc__', '__file__', '__path__',
163 '__module__', '__name__', '__slots__', '__package__')
164 if name in _hidden_names: return 0
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000165 # Private names are hidden, but special names are displayed.
166 if name.startswith('__') and name.endswith('__'): return 1
Skip Montanaroa5616d22004-06-11 04:46:12 +0000167 if all is not None:
168 # only document that which the programmer exported in __all__
169 return name in all
170 else:
171 return not name.startswith('_')
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000172
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +0000173def classify_class_attrs(object):
174 """Wrap inspect.classify_class_attrs, with fixup for data descriptors."""
Brett Cannonaca98b02008-08-03 00:58:51 +0000175 def fixup(data):
176 name, kind, cls, value = data
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +0000177 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."""
Brett Cannonaca98b02008-08-03 00:58:51 +0000233 def __init__(self, filename, exc_info):
234 exc, value, tb = exc_info
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000235 self.filename = filename
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000236 self.exc = exc
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000237 self.value = value
238 self.tb = tb
239
240 def __str__(self):
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000241 exc = self.exc
242 if type(exc) is types.ClassType:
243 exc = exc.__name__
244 return 'problem in %s - %s: %s' % (self.filename, exc, self.value)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000245
246def importfile(path):
247 """Import a Python source file or compiled file given its path."""
248 magic = imp.get_magic()
249 file = open(path, 'r')
250 if file.read(len(magic)) == magic:
251 kind = imp.PY_COMPILED
252 else:
253 kind = imp.PY_SOURCE
254 file.close()
255 filename = os.path.basename(path)
256 name, ext = os.path.splitext(filename)
257 file = open(path, 'r')
258 try:
259 module = imp.load_module(name, file, path, (ext, 'r', kind))
260 except:
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000261 raise ErrorDuringImport(path, sys.exc_info())
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000262 file.close()
263 return module
264
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000265def safeimport(path, forceload=0, cache={}):
266 """Import a module; handle errors; return None if the module isn't found.
267
268 If the module *is* found but an exception occurs, it's wrapped in an
269 ErrorDuringImport exception and reraised. Unlike __import__, if a
270 package path is specified, the module at the end of the path is returned,
271 not the package at the beginning. If the optional 'forceload' argument
272 is 1, we reload the module from disk (unless it's a dynamic extension)."""
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000273 try:
Ka-Ping Yee9a2dcf82005-11-05 05:04:41 +0000274 # If forceload is 1 and the module has been previously loaded from
275 # disk, we always have to reload the module. Checking the file's
276 # mtime isn't good enough (e.g. the module could contain a class
277 # that inherits from another module that has changed).
278 if forceload and path in sys.modules:
279 if path not in sys.builtin_module_names:
280 # Avoid simply calling reload() because it leaves names in
281 # the currently loaded module lying around if they're not
282 # defined in the new source file. Instead, remove the
283 # module from sys.modules and re-import. Also remove any
284 # submodules because they won't appear in the newly loaded
285 # module's namespace if they're already in sys.modules.
286 subs = [m for m in sys.modules if m.startswith(path + '.')]
287 for key in [path] + subs:
288 # Prevent garbage collection.
289 cache[key] = sys.modules[key]
290 del sys.modules[key]
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000291 module = __import__(path)
292 except:
293 # Did the error occur before or after the module was found?
294 (exc, value, tb) = info = sys.exc_info()
Raymond Hettinger54f02222002-06-01 14:18:47 +0000295 if path in sys.modules:
Fred Drakedb390c12005-10-28 14:39:47 +0000296 # An error occurred while executing the imported module.
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000297 raise ErrorDuringImport(sys.modules[path].__file__, info)
298 elif exc is SyntaxError:
299 # A SyntaxError occurred before we could execute the module.
300 raise ErrorDuringImport(value.filename, info)
R. David Murrayc4c84e12009-06-23 20:37:26 +0000301 elif exc is ImportError and extract_tb(tb)[-1][2]=='safeimport':
302 # The import error occurred directly in this function,
303 # which means there is no such module in the path.
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000304 return None
305 else:
306 # Some other error occurred during the importing process.
307 raise ErrorDuringImport(path, sys.exc_info())
308 for part in split(path, '.')[1:]:
309 try: module = getattr(module, part)
310 except AttributeError: return None
311 return module
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000312
313# ---------------------------------------------------- formatter base class
314
315class Doc:
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000316 def document(self, object, name=None, *args):
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000317 """Generate documentation for an object."""
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000318 args = (object, name) + args
Brett Cannon28a4f0f2003-06-11 23:38:55 +0000319 # 'try' clause is to attempt to handle the possibility that inspect
320 # identifies something in a way that pydoc itself has issues handling;
321 # think 'super' and how it is a descriptor (which raises the exception
322 # by lacking a __name__ attribute) and an instance.
Barry Warsaw00decd72006-07-27 23:43:15 +0000323 if inspect.isgetsetdescriptor(object): return self.docdata(*args)
324 if inspect.ismemberdescriptor(object): return self.docdata(*args)
Brett Cannon28a4f0f2003-06-11 23:38:55 +0000325 try:
326 if inspect.ismodule(object): return self.docmodule(*args)
327 if inspect.isclass(object): return self.docclass(*args)
328 if inspect.isroutine(object): return self.docroutine(*args)
329 except AttributeError:
330 pass
Johannes Gijsbers8de645a2004-11-07 19:16:05 +0000331 if isinstance(object, property): return self.docproperty(*args)
Guido van Rossum68468eb2003-02-27 20:14:51 +0000332 return self.docother(*args)
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000333
334 def fail(self, object, name=None, *args):
335 """Raise an exception for unimplemented types."""
336 message = "don't know how to document object%s of type %s" % (
337 name and ' ' + repr(name), type(object).__name__)
338 raise TypeError, message
339
Barry Warsaw00decd72006-07-27 23:43:15 +0000340 docmodule = docclass = docroutine = docother = docproperty = docdata = fail
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000341
Skip Montanaro4997a692003-09-10 16:47:51 +0000342 def getdocloc(self, object):
343 """Return the location of module docs or None"""
344
345 try:
346 file = inspect.getabsfile(object)
347 except TypeError:
348 file = '(built-in)'
349
350 docloc = os.environ.get("PYTHONDOCS",
Georg Brandl0751d1a2008-01-21 17:13:03 +0000351 "http://docs.python.org/library")
Skip Montanaro4997a692003-09-10 16:47:51 +0000352 basedir = os.path.join(sys.exec_prefix, "lib",
353 "python"+sys.version[0:3])
354 if (isinstance(object, type(os)) and
355 (object.__name__ in ('errno', 'exceptions', 'gc', 'imp',
356 'marshal', 'posix', 'signal', 'sys',
357 'thread', 'zipimport') or
358 (file.startswith(basedir) and
Brian Curtina5d96792010-03-31 03:14:27 +0000359 not file.startswith(os.path.join(basedir, 'site-packages')))) and
Brian Curtin74ee91d2010-04-01 04:03:45 +0000360 object.__name__ not in ('xml.etree', 'test.pydoc_mod')):
Skip Montanaro4997a692003-09-10 16:47:51 +0000361 if docloc.startswith("http://"):
Georg Brandl0751d1a2008-01-21 17:13:03 +0000362 docloc = "%s/%s" % (docloc.rstrip("/"), object.__name__)
Skip Montanaro4997a692003-09-10 16:47:51 +0000363 else:
Georg Brandl0751d1a2008-01-21 17:13:03 +0000364 docloc = os.path.join(docloc, object.__name__ + ".html")
Skip Montanaro4997a692003-09-10 16:47:51 +0000365 else:
366 docloc = None
367 return docloc
368
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000369# -------------------------------------------- HTML documentation generator
370
371class HTMLRepr(Repr):
372 """Class for safely making an HTML representation of a Python object."""
373 def __init__(self):
374 Repr.__init__(self)
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000375 self.maxlist = self.maxtuple = 20
376 self.maxdict = 10
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000377 self.maxstring = self.maxother = 100
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000378
379 def escape(self, text):
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +0000380 return replace(text, '&', '&amp;', '<', '&lt;', '>', '&gt;')
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000381
382 def repr(self, object):
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000383 return Repr.repr(self, object)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000384
385 def repr1(self, x, level):
Skip Montanaro0fe8fce2003-06-27 15:45:41 +0000386 if hasattr(type(x), '__name__'):
387 methodname = 'repr_' + join(split(type(x).__name__), '_')
388 if hasattr(self, methodname):
389 return getattr(self, methodname)(x, level)
390 return self.escape(cram(stripid(repr(x)), self.maxother))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000391
392 def repr_string(self, x, level):
Ka-Ping Yeea2fe1032001-03-02 01:19:14 +0000393 test = cram(x, self.maxstring)
394 testrepr = repr(test)
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +0000395 if '\\' in test and '\\' not in replace(testrepr, r'\\', ''):
Ka-Ping Yeea2fe1032001-03-02 01:19:14 +0000396 # Backslashes are only literal in the string and are never
397 # needed to make any special characters, so show a raw string.
398 return 'r' + testrepr[0] + self.escape(test) + testrepr[0]
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +0000399 return re.sub(r'((\\[\\abfnrtv\'"]|\\[0-9]..|\\x..|\\u....)+)',
Ka-Ping Yeea2fe1032001-03-02 01:19:14 +0000400 r'<font color="#c040c0">\1</font>',
401 self.escape(testrepr))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000402
Skip Montanarodf708782002-03-07 22:58:02 +0000403 repr_str = repr_string
404
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000405 def repr_instance(self, x, level):
406 try:
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000407 return self.escape(cram(stripid(repr(x)), self.maxstring))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000408 except:
409 return self.escape('<%s instance>' % x.__class__.__name__)
410
411 repr_unicode = repr_string
412
413class HTMLDoc(Doc):
414 """Formatter class for HTML documentation."""
415
416 # ------------------------------------------- HTML formatting utilities
417
418 _repr_instance = HTMLRepr()
419 repr = _repr_instance.repr
420 escape = _repr_instance.escape
421
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000422 def page(self, title, contents):
423 """Format an HTML page."""
424 return '''
Matthias Klose4bdc4c92009-04-10 12:38:09 +0000425<!DOCTYPE html PUBLIC "-//W3C//DTD HTML 4.0 Transitional//EN">
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000426<html><head><title>Python: %s</title>
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000427</head><body bgcolor="#f0f0f8">
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000428%s
429</body></html>''' % (title, contents)
430
431 def heading(self, title, fgcol, bgcol, extras=''):
432 """Format a page heading."""
433 return '''
Tim Peters59ed4482001-10-31 04:20:26 +0000434<table width="100%%" cellspacing=0 cellpadding=2 border=0 summary="heading">
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000435<tr bgcolor="%s">
Tim Peters2306d242001-09-25 03:18:32 +0000436<td valign=bottom>&nbsp;<br>
437<font color="%s" face="helvetica, arial">&nbsp;<br>%s</font></td
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000438><td align=right valign=bottom
Ka-Ping Yee987ec902001-03-23 13:35:45 +0000439><font color="%s" face="helvetica, arial">%s</font></td></tr></table>
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000440 ''' % (bgcol, fgcol, title, fgcol, extras or '&nbsp;')
441
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000442 def section(self, title, fgcol, bgcol, contents, width=6,
443 prelude='', marginalia=None, gap='&nbsp;'):
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000444 """Format a section with a heading."""
445 if marginalia is None:
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +0000446 marginalia = '<tt>' + '&nbsp;' * width + '</tt>'
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000447 result = '''<p>
Tim Peters59ed4482001-10-31 04:20:26 +0000448<table width="100%%" cellspacing=0 cellpadding=2 border=0 summary="section">
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000449<tr bgcolor="%s">
Tim Peters2306d242001-09-25 03:18:32 +0000450<td colspan=3 valign=bottom>&nbsp;<br>
451<font color="%s" face="helvetica, arial">%s</font></td></tr>
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000452 ''' % (bgcol, fgcol, title)
453 if prelude:
454 result = result + '''
Ka-Ping Yee987ec902001-03-23 13:35:45 +0000455<tr bgcolor="%s"><td rowspan=2>%s</td>
456<td colspan=2>%s</td></tr>
457<tr><td>%s</td>''' % (bgcol, marginalia, prelude, gap)
458 else:
459 result = result + '''
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000460<tr><td bgcolor="%s">%s</td><td>%s</td>''' % (bgcol, marginalia, gap)
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000461
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +0000462 return result + '\n<td width="100%%">%s</td></tr></table>' % contents
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000463
464 def bigsection(self, title, *args):
465 """Format a section with a big heading."""
466 title = '<big><strong>%s</strong></big>' % title
Guido van Rossum68468eb2003-02-27 20:14:51 +0000467 return self.section(title, *args)
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000468
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000469 def preformat(self, text):
470 """Format literal preformatted text."""
471 text = self.escape(expandtabs(text))
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +0000472 return replace(text, '\n\n', '\n \n', '\n\n', '\n \n',
473 ' ', '&nbsp;', '\n', '<br>\n')
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000474
475 def multicolumn(self, list, format, cols=4):
476 """Format a list of items into a multi-column list."""
477 result = ''
478 rows = (len(list)+cols-1)/cols
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000479 for col in range(cols):
480 result = result + '<td width="%d%%" valign=top>' % (100/cols)
481 for i in range(rows*col, rows*col+rows):
482 if i < len(list):
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000483 result = result + format(list[i]) + '<br>\n'
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000484 result = result + '</td>'
Tim Peters59ed4482001-10-31 04:20:26 +0000485 return '<table width="100%%" summary="list"><tr>%s</tr></table>' % result
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000486
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000487 def grey(self, text): return '<font color="#909090">%s</font>' % text
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000488
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000489 def namelink(self, name, *dicts):
490 """Make a link for an identifier, given name-to-URL mappings."""
491 for dict in dicts:
Raymond Hettinger54f02222002-06-01 14:18:47 +0000492 if name in dict:
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000493 return '<a href="%s">%s</a>' % (dict[name], name)
494 return name
495
Ka-Ping Yeeb7a48302001-04-12 20:39:14 +0000496 def classlink(self, object, modname):
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000497 """Make a link for a class."""
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000498 name, module = object.__name__, sys.modules.get(object.__module__)
499 if hasattr(module, name) and getattr(module, name) is object:
Ka-Ping Yeeb7a48302001-04-12 20:39:14 +0000500 return '<a href="%s.html#%s">%s</a>' % (
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000501 module.__name__, name, classname(object, modname))
502 return classname(object, modname)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000503
504 def modulelink(self, object):
505 """Make a link for a module."""
506 return '<a href="%s.html">%s</a>' % (object.__name__, object.__name__)
507
Brett Cannonaca98b02008-08-03 00:58:51 +0000508 def modpkglink(self, data):
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000509 """Make a link for a module or package to display in an index."""
Brett Cannonaca98b02008-08-03 00:58:51 +0000510 name, path, ispackage, shadowed = data
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000511 if shadowed:
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000512 return self.grey(name)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000513 if path:
514 url = '%s.%s.html' % (path, name)
515 else:
516 url = '%s.html' % name
517 if ispackage:
518 text = '<strong>%s</strong>&nbsp;(package)' % name
519 else:
520 text = name
521 return '<a href="%s">%s</a>' % (url, text)
522
523 def markup(self, text, escape=None, funcs={}, classes={}, methods={}):
524 """Mark up some plain text, given a context of symbols to look for.
525 Each context dictionary maps object names to anchor names."""
526 escape = escape or self.escape
527 results = []
528 here = 0
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000529 pattern = re.compile(r'\b((http|ftp)://\S+[\w/]|'
530 r'RFC[- ]?(\d+)|'
Ka-Ping Yeef78a81b2001-03-27 08:13:42 +0000531 r'PEP[- ]?(\d+)|'
Neil Schemenauerd69711c2002-03-24 23:02:07 +0000532 r'(self\.)?(\w+))')
Guido van Rossum8ca162f2002-04-07 06:36:23 +0000533 while True:
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000534 match = pattern.search(text, here)
535 if not match: break
536 start, end = match.span()
537 results.append(escape(text[here:start]))
538
Ka-Ping Yeef78a81b2001-03-27 08:13:42 +0000539 all, scheme, rfc, pep, selfdot, name = match.groups()
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000540 if scheme:
Neil Schemenauercddc1a02002-03-24 23:11:21 +0000541 url = escape(all).replace('"', '&quot;')
542 results.append('<a href="%s">%s</a>' % (url, url))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000543 elif rfc:
Ka-Ping Yeef78a81b2001-03-27 08:13:42 +0000544 url = 'http://www.rfc-editor.org/rfc/rfc%d.txt' % int(rfc)
545 results.append('<a href="%s">%s</a>' % (url, escape(all)))
546 elif pep:
Andrew M. Kuchling727a5902008-02-05 16:06:57 +0000547 url = 'http://www.python.org/dev/peps/pep-%04d/' % int(pep)
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000548 results.append('<a href="%s">%s</a>' % (url, escape(all)))
549 elif text[end:end+1] == '(':
550 results.append(self.namelink(name, methods, funcs, classes))
551 elif selfdot:
552 results.append('self.<strong>%s</strong>' % name)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000553 else:
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000554 results.append(self.namelink(name, classes))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000555 here = end
556 results.append(escape(text[here:]))
557 return join(results, '')
558
559 # ---------------------------------------------- type-specific routines
560
Ka-Ping Yeeb7a48302001-04-12 20:39:14 +0000561 def formattree(self, tree, modname, parent=None):
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000562 """Produce HTML for a class tree as given by inspect.getclasstree()."""
563 result = ''
564 for entry in tree:
565 if type(entry) is type(()):
566 c, bases = entry
Tim Peters2306d242001-09-25 03:18:32 +0000567 result = result + '<dt><font face="helvetica, arial">'
Ka-Ping Yeeb7a48302001-04-12 20:39:14 +0000568 result = result + self.classlink(c, modname)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000569 if bases and bases != (parent,):
570 parents = []
571 for base in bases:
Ka-Ping Yeeb7a48302001-04-12 20:39:14 +0000572 parents.append(self.classlink(base, modname))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000573 result = result + '(' + join(parents, ', ') + ')'
Tim Peters2306d242001-09-25 03:18:32 +0000574 result = result + '\n</font></dt>'
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000575 elif type(entry) is type([]):
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000576 result = result + '<dd>\n%s</dd>\n' % self.formattree(
Ka-Ping Yeeb7a48302001-04-12 20:39:14 +0000577 entry, modname, c)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000578 return '<dl>\n%s</dl>\n' % result
579
Tim Peters8dd7ade2001-10-18 19:56:17 +0000580 def docmodule(self, object, name=None, mod=None, *ignored):
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000581 """Produce HTML documentation for a module object."""
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000582 name = object.__name__ # ignore the passed-in name
Skip Montanaroa5616d22004-06-11 04:46:12 +0000583 try:
584 all = object.__all__
585 except AttributeError:
586 all = None
Ka-Ping Yee66efbc72001-03-01 13:55:20 +0000587 parts = split(name, '.')
588 links = []
589 for i in range(len(parts)-1):
590 links.append(
591 '<a href="%s.html"><font color="#ffffff">%s</font></a>' %
592 (join(parts[:i+1], '.'), parts[i]))
593 linkedname = join(links + parts[-1:], '.')
594 head = '<big><big><strong>%s</strong></big></big>' % linkedname
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000595 try:
Ka-Ping Yee239432a2001-03-02 02:45:08 +0000596 path = inspect.getabsfile(object)
Ka-Ping Yee6191a232001-04-13 15:00:27 +0000597 url = path
598 if sys.platform == 'win32':
599 import nturl2path
600 url = nturl2path.pathname2url(path)
601 filelink = '<a href="file:%s">%s</a>' % (url, path)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000602 except TypeError:
603 filelink = '(built-in)'
Ka-Ping Yee6f3f9a42001-02-27 22:42:36 +0000604 info = []
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000605 if hasattr(object, '__version__'):
Ka-Ping Yee6f3f9a42001-02-27 22:42:36 +0000606 version = str(object.__version__)
Ka-Ping Yee40c49912001-02-27 22:46:01 +0000607 if version[:11] == '$' + 'Revision: ' and version[-1:] == '$':
608 version = strip(version[11:-1])
Ka-Ping Yee1d384632001-03-01 00:24:32 +0000609 info.append('version %s' % self.escape(version))
Ka-Ping Yee6f3f9a42001-02-27 22:42:36 +0000610 if hasattr(object, '__date__'):
611 info.append(self.escape(str(object.__date__)))
612 if info:
613 head = head + ' (%s)' % join(info, ', ')
Skip Montanaro4997a692003-09-10 16:47:51 +0000614 docloc = self.getdocloc(object)
615 if docloc is not None:
616 docloc = '<br><a href="%(docloc)s">Module Docs</a>' % locals()
617 else:
618 docloc = ''
Ka-Ping Yee66efbc72001-03-01 13:55:20 +0000619 result = self.heading(
Skip Montanaro4997a692003-09-10 16:47:51 +0000620 head, '#ffffff', '#7799ee',
621 '<a href=".">index</a><br>' + filelink + docloc)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000622
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000623 modules = inspect.getmembers(object, inspect.ismodule)
624
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000625 classes, cdict = [], {}
626 for key, value in inspect.getmembers(object, inspect.isclass):
Johannes Gijsbers4c11f602004-08-30 14:13:04 +0000627 # if __all__ exists, believe it. Otherwise use old heuristic.
628 if (all is not None or
629 (inspect.getmodule(value) or object) is object):
Skip Montanaroa5616d22004-06-11 04:46:12 +0000630 if visiblename(key, all):
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000631 classes.append((key, value))
632 cdict[key] = cdict[value] = '#' + key
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000633 for key, value in classes:
634 for base in value.__bases__:
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000635 key, modname = base.__name__, base.__module__
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000636 module = sys.modules.get(modname)
637 if modname != name and module and hasattr(module, key):
638 if getattr(module, key) is base:
Raymond Hettinger54f02222002-06-01 14:18:47 +0000639 if not key in cdict:
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000640 cdict[key] = cdict[base] = modname + '.html#' + key
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000641 funcs, fdict = [], {}
642 for key, value in inspect.getmembers(object, inspect.isroutine):
Johannes Gijsbers4c11f602004-08-30 14:13:04 +0000643 # if __all__ exists, believe it. Otherwise use old heuristic.
644 if (all is not None or
645 inspect.isbuiltin(value) or inspect.getmodule(value) is object):
Skip Montanaroa5616d22004-06-11 04:46:12 +0000646 if visiblename(key, all):
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000647 funcs.append((key, value))
648 fdict[key] = '#-' + key
649 if inspect.isfunction(value): fdict[value] = fdict[key]
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000650 data = []
651 for key, value in inspect.getmembers(object, isdata):
Skip Montanaroa5616d22004-06-11 04:46:12 +0000652 if visiblename(key, all):
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000653 data.append((key, value))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000654
655 doc = self.markup(getdoc(object), self.preformat, fdict, cdict)
656 doc = doc and '<tt>%s</tt>' % doc
Tim Peters2306d242001-09-25 03:18:32 +0000657 result = result + '<p>%s</p>\n' % doc
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000658
659 if hasattr(object, '__path__'):
660 modpkgs = []
Phillip J. Ebyceb30872006-04-18 00:59:55 +0000661 for importer, modname, ispkg in pkgutil.iter_modules(object.__path__):
662 modpkgs.append((modname, name, ispkg, 0))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000663 modpkgs.sort()
664 contents = self.multicolumn(modpkgs, self.modpkglink)
665 result = result + self.bigsection(
666 'Package Contents', '#ffffff', '#aa55cc', contents)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000667 elif modules:
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000668 contents = self.multicolumn(
Brett Cannonaca98b02008-08-03 00:58:51 +0000669 modules, lambda key_value, s=self: s.modulelink(key_value[1]))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000670 result = result + self.bigsection(
Georg Brandlb169eaa2008-02-17 21:18:55 +0000671 'Modules', '#ffffff', '#aa55cc', contents)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000672
673 if classes:
Brett Cannonaca98b02008-08-03 00:58:51 +0000674 classlist = map(lambda key_value: key_value[1], classes)
Ka-Ping Yeeb7a48302001-04-12 20:39:14 +0000675 contents = [
676 self.formattree(inspect.getclasstree(classlist, 1), name)]
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000677 for key, value in classes:
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +0000678 contents.append(self.document(value, key, name, fdict, cdict))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000679 result = result + self.bigsection(
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000680 'Classes', '#ffffff', '#ee77aa', join(contents))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000681 if funcs:
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000682 contents = []
683 for key, value in funcs:
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +0000684 contents.append(self.document(value, key, name, fdict, cdict))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000685 result = result + self.bigsection(
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000686 'Functions', '#ffffff', '#eeaa77', join(contents))
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000687 if data:
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000688 contents = []
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000689 for key, value in data:
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000690 contents.append(self.document(value, key))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000691 result = result + self.bigsection(
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000692 'Data', '#ffffff', '#55aa55', join(contents, '<br>\n'))
Ka-Ping Yee6f3f9a42001-02-27 22:42:36 +0000693 if hasattr(object, '__author__'):
694 contents = self.markup(str(object.__author__), self.preformat)
695 result = result + self.bigsection(
696 'Author', '#ffffff', '#7799ee', contents)
Ka-Ping Yee6f3f9a42001-02-27 22:42:36 +0000697 if hasattr(object, '__credits__'):
698 contents = self.markup(str(object.__credits__), self.preformat)
699 result = result + self.bigsection(
700 'Credits', '#ffffff', '#7799ee', contents)
701
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000702 return result
703
Tim Peters8dd7ade2001-10-18 19:56:17 +0000704 def docclass(self, object, name=None, mod=None, funcs={}, classes={},
705 *ignored):
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000706 """Produce HTML documentation for a class object."""
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000707 realname = object.__name__
708 name = name or realname
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000709 bases = object.__bases__
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000710
Tim Petersb47879b2001-09-24 04:47:19 +0000711 contents = []
712 push = contents.append
713
Tim Petersfa26f7c2001-09-24 08:05:11 +0000714 # Cute little class to pump out a horizontal rule between sections.
715 class HorizontalRule:
716 def __init__(self):
717 self.needone = 0
718 def maybe(self):
719 if self.needone:
720 push('<hr>\n')
721 self.needone = 1
722 hr = HorizontalRule()
723
Tim Petersc86f6ca2001-09-26 21:31:51 +0000724 # List the mro, if non-trivial.
Raymond Hettinger756b3f32004-01-29 06:37:52 +0000725 mro = deque(inspect.getmro(object))
Tim Petersc86f6ca2001-09-26 21:31:51 +0000726 if len(mro) > 2:
727 hr.maybe()
728 push('<dl><dt>Method resolution order:</dt>\n')
729 for base in mro:
730 push('<dd>%s</dd>\n' % self.classlink(base,
731 object.__module__))
732 push('</dl>\n')
733
Tim Petersb47879b2001-09-24 04:47:19 +0000734 def spill(msg, attrs, predicate):
Tim Petersfa26f7c2001-09-24 08:05:11 +0000735 ok, attrs = _split_list(attrs, predicate)
Tim Petersb47879b2001-09-24 04:47:19 +0000736 if ok:
Tim Petersfa26f7c2001-09-24 08:05:11 +0000737 hr.maybe()
Tim Petersb47879b2001-09-24 04:47:19 +0000738 push(msg)
739 for name, kind, homecls, value in ok:
740 push(self.document(getattr(object, name), name, mod,
741 funcs, classes, mdict, object))
742 push('\n')
743 return attrs
744
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +0000745 def spilldescriptors(msg, attrs, predicate):
Tim Petersfa26f7c2001-09-24 08:05:11 +0000746 ok, attrs = _split_list(attrs, predicate)
Tim Petersb47879b2001-09-24 04:47:19 +0000747 if ok:
Tim Petersfa26f7c2001-09-24 08:05:11 +0000748 hr.maybe()
Tim Petersb47879b2001-09-24 04:47:19 +0000749 push(msg)
750 for name, kind, homecls, value in ok:
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +0000751 push(self._docdescriptor(name, value, mod))
Tim Petersb47879b2001-09-24 04:47:19 +0000752 return attrs
753
Tim Petersfa26f7c2001-09-24 08:05:11 +0000754 def spilldata(msg, attrs, predicate):
755 ok, attrs = _split_list(attrs, predicate)
Tim Petersb47879b2001-09-24 04:47:19 +0000756 if ok:
Tim Petersfa26f7c2001-09-24 08:05:11 +0000757 hr.maybe()
Tim Petersb47879b2001-09-24 04:47:19 +0000758 push(msg)
759 for name, kind, homecls, value in ok:
760 base = self.docother(getattr(object, name), name, mod)
Brett Cannonaca98b02008-08-03 00:58:51 +0000761 if (hasattr(value, '__call__') or
762 inspect.isdatadescriptor(value)):
Guido van Rossum5e355b22002-05-21 20:56:15 +0000763 doc = getattr(value, "__doc__", None)
764 else:
765 doc = None
Tim Petersb47879b2001-09-24 04:47:19 +0000766 if doc is None:
767 push('<dl><dt>%s</dl>\n' % base)
768 else:
769 doc = self.markup(getdoc(value), self.preformat,
770 funcs, classes, mdict)
Tim Peters2306d242001-09-25 03:18:32 +0000771 doc = '<dd><tt>%s</tt>' % doc
Tim Petersb47879b2001-09-24 04:47:19 +0000772 push('<dl><dt>%s%s</dl>\n' % (base, doc))
773 push('\n')
774 return attrs
775
Brett Cannonaca98b02008-08-03 00:58:51 +0000776 attrs = filter(lambda data: visiblename(data[0]),
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +0000777 classify_class_attrs(object))
Tim Petersb47879b2001-09-24 04:47:19 +0000778 mdict = {}
779 for key, kind, homecls, value in attrs:
780 mdict[key] = anchor = '#' + name + '-' + key
781 value = getattr(object, key)
782 try:
783 # The value may not be hashable (e.g., a data attr with
784 # a dict or list value).
785 mdict[value] = anchor
786 except TypeError:
787 pass
788
Tim Petersfa26f7c2001-09-24 08:05:11 +0000789 while attrs:
Tim Peters351e3622001-09-27 03:29:51 +0000790 if mro:
Raymond Hettinger756b3f32004-01-29 06:37:52 +0000791 thisclass = mro.popleft()
Tim Peters351e3622001-09-27 03:29:51 +0000792 else:
793 thisclass = attrs[0][2]
Tim Petersfa26f7c2001-09-24 08:05:11 +0000794 attrs, inherited = _split_list(attrs, lambda t: t[2] is thisclass)
795
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000796 if thisclass is __builtin__.object:
797 attrs = inherited
798 continue
799 elif thisclass is object:
800 tag = 'defined here'
Tim Petersb47879b2001-09-24 04:47:19 +0000801 else:
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000802 tag = 'inherited from %s' % self.classlink(thisclass,
803 object.__module__)
Tim Petersb47879b2001-09-24 04:47:19 +0000804 tag += ':<br>\n'
805
806 # Sort attrs by name.
Phillip J. Ebyceb30872006-04-18 00:59:55 +0000807 try:
808 attrs.sort(key=lambda t: t[0])
809 except TypeError:
810 attrs.sort(lambda t1, t2: cmp(t1[0], t2[0])) # 2.3 compat
Tim Petersb47879b2001-09-24 04:47:19 +0000811
812 # Pump out the attrs, segregated by kind.
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000813 attrs = spill('Methods %s' % tag, attrs,
Tim Petersb47879b2001-09-24 04:47:19 +0000814 lambda t: t[1] == 'method')
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000815 attrs = spill('Class methods %s' % tag, attrs,
Tim Petersb47879b2001-09-24 04:47:19 +0000816 lambda t: t[1] == 'class method')
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000817 attrs = spill('Static methods %s' % tag, attrs,
Tim Petersb47879b2001-09-24 04:47:19 +0000818 lambda t: t[1] == 'static method')
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +0000819 attrs = spilldescriptors('Data descriptors %s' % tag, attrs,
820 lambda t: t[1] == 'data descriptor')
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000821 attrs = spilldata('Data and other attributes %s' % tag, attrs,
Tim Petersfa26f7c2001-09-24 08:05:11 +0000822 lambda t: t[1] == 'data')
Tim Petersb47879b2001-09-24 04:47:19 +0000823 assert attrs == []
Tim Peters351e3622001-09-27 03:29:51 +0000824 attrs = inherited
Tim Petersb47879b2001-09-24 04:47:19 +0000825
826 contents = ''.join(contents)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000827
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000828 if name == realname:
829 title = '<a name="%s">class <strong>%s</strong></a>' % (
830 name, realname)
831 else:
832 title = '<strong>%s</strong> = <a name="%s">class %s</a>' % (
833 name, name, realname)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000834 if bases:
835 parents = []
836 for base in bases:
Ka-Ping Yeeb7a48302001-04-12 20:39:14 +0000837 parents.append(self.classlink(base, object.__module__))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000838 title = title + '(%s)' % join(parents, ', ')
Tim Peters2306d242001-09-25 03:18:32 +0000839 doc = self.markup(getdoc(object), self.preformat, funcs, classes, mdict)
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000840 doc = doc and '<tt>%s<br>&nbsp;</tt>' % doc
Tim Petersc86f6ca2001-09-26 21:31:51 +0000841
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000842 return self.section(title, '#000000', '#ffc8d8', contents, 3, doc)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000843
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000844 def formatvalue(self, object):
845 """Format an argument default value as text."""
Tim Peters2306d242001-09-25 03:18:32 +0000846 return self.grey('=' + self.repr(object))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000847
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +0000848 def docroutine(self, object, name=None, mod=None,
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000849 funcs={}, classes={}, methods={}, cl=None):
Ka-Ping Yee66efbc72001-03-01 13:55:20 +0000850 """Produce HTML documentation for a function or method object."""
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000851 realname = object.__name__
852 name = name or realname
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000853 anchor = (cl and cl.__name__ or '') + '-' + name
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000854 note = ''
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000855 skipdocs = 0
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000856 if inspect.ismethod(object):
Ka-Ping Yee6dcfa382001-04-12 20:27:31 +0000857 imclass = object.im_class
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000858 if cl:
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +0000859 if imclass is not cl:
Ka-Ping Yeeb7a48302001-04-12 20:39:14 +0000860 note = ' from ' + self.classlink(imclass, mod)
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000861 else:
Georg Brandl7e2b6bb2007-03-13 22:16:30 +0000862 if object.im_self is not None:
Ka-Ping Yeeb7a48302001-04-12 20:39:14 +0000863 note = ' method of %s instance' % self.classlink(
864 object.im_self.__class__, mod)
865 else:
866 note = ' unbound %s method' % self.classlink(imclass,mod)
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000867 object = object.im_func
868
869 if name == realname:
870 title = '<a name="%s"><strong>%s</strong></a>' % (anchor, realname)
871 else:
Raymond Hettinger54f02222002-06-01 14:18:47 +0000872 if (cl and realname in cl.__dict__ and
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000873 cl.__dict__[realname] is object):
Ka-Ping Yeee280c062001-03-23 14:05:53 +0000874 reallink = '<a href="#%s">%s</a>' % (
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000875 cl.__name__ + '-' + realname, realname)
876 skipdocs = 1
877 else:
878 reallink = realname
879 title = '<a name="%s"><strong>%s</strong></a> = %s' % (
880 anchor, name, reallink)
Tim Peters4bcfa312001-09-20 06:08:24 +0000881 if inspect.isfunction(object):
Ka-Ping Yee66efbc72001-03-01 13:55:20 +0000882 args, varargs, varkw, defaults = inspect.getargspec(object)
883 argspec = inspect.formatargspec(
884 args, varargs, varkw, defaults, formatvalue=self.formatvalue)
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000885 if realname == '<lambda>':
Tim Peters59ed4482001-10-31 04:20:26 +0000886 title = '<strong>%s</strong> <em>lambda</em> ' % name
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000887 argspec = argspec[1:-1] # remove parentheses
Tim Peters4bcfa312001-09-20 06:08:24 +0000888 else:
889 argspec = '(...)'
Ka-Ping Yee66efbc72001-03-01 13:55:20 +0000890
Tim Peters2306d242001-09-25 03:18:32 +0000891 decl = title + argspec + (note and self.grey(
892 '<font face="helvetica, arial">%s</font>' % note))
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000893
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000894 if skipdocs:
Tim Peters2306d242001-09-25 03:18:32 +0000895 return '<dl><dt>%s</dt></dl>\n' % decl
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000896 else:
897 doc = self.markup(
898 getdoc(object), self.preformat, funcs, classes, methods)
Tim Peters2306d242001-09-25 03:18:32 +0000899 doc = doc and '<dd><tt>%s</tt></dd>' % doc
900 return '<dl><dt>%s</dt>%s</dl>\n' % (decl, doc)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000901
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +0000902 def _docdescriptor(self, name, value, mod):
Johannes Gijsbers8de645a2004-11-07 19:16:05 +0000903 results = []
904 push = results.append
905
906 if name:
907 push('<dl><dt><strong>%s</strong></dt>\n' % name)
908 if value.__doc__ is not None:
Ka-Ping Yeebba6acc2005-02-19 22:58:26 +0000909 doc = self.markup(getdoc(value), self.preformat)
Johannes Gijsbers8de645a2004-11-07 19:16:05 +0000910 push('<dd><tt>%s</tt></dd>\n' % doc)
Johannes Gijsbers8de645a2004-11-07 19:16:05 +0000911 push('</dl>\n')
912
913 return ''.join(results)
914
915 def docproperty(self, object, name=None, mod=None, cl=None):
916 """Produce html documentation for a property."""
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +0000917 return self._docdescriptor(name, object, mod)
Johannes Gijsbers8de645a2004-11-07 19:16:05 +0000918
Tim Peters8dd7ade2001-10-18 19:56:17 +0000919 def docother(self, object, name=None, mod=None, *ignored):
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000920 """Produce HTML documentation for a data object."""
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +0000921 lhs = name and '<strong>%s</strong> = ' % name or ''
922 return lhs + self.repr(object)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000923
Barry Warsaw00decd72006-07-27 23:43:15 +0000924 def docdata(self, object, name=None, mod=None, cl=None):
925 """Produce html documentation for a data descriptor."""
926 return self._docdescriptor(name, object, mod)
927
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000928 def index(self, dir, shadowed=None):
929 """Generate an HTML index for a directory of modules."""
930 modpkgs = []
931 if shadowed is None: shadowed = {}
Phillip J. Ebyceb30872006-04-18 00:59:55 +0000932 for importer, name, ispkg in pkgutil.iter_modules([dir]):
933 modpkgs.append((name, '', ispkg, name in shadowed))
934 shadowed[name] = 1
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000935
936 modpkgs.sort()
937 contents = self.multicolumn(modpkgs, self.modpkglink)
938 return self.bigsection(dir, '#ffffff', '#ee77aa', contents)
939
940# -------------------------------------------- text documentation generator
941
942class TextRepr(Repr):
943 """Class for safely making a text representation of a Python object."""
944 def __init__(self):
945 Repr.__init__(self)
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000946 self.maxlist = self.maxtuple = 20
947 self.maxdict = 10
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000948 self.maxstring = self.maxother = 100
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000949
950 def repr1(self, x, level):
Skip Montanaro0fe8fce2003-06-27 15:45:41 +0000951 if hasattr(type(x), '__name__'):
952 methodname = 'repr_' + join(split(type(x).__name__), '_')
953 if hasattr(self, methodname):
954 return getattr(self, methodname)(x, level)
955 return cram(stripid(repr(x)), self.maxother)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000956
Ka-Ping Yeea2fe1032001-03-02 01:19:14 +0000957 def repr_string(self, x, level):
958 test = cram(x, self.maxstring)
959 testrepr = repr(test)
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +0000960 if '\\' in test and '\\' not in replace(testrepr, r'\\', ''):
Ka-Ping Yeea2fe1032001-03-02 01:19:14 +0000961 # Backslashes are only literal in the string and are never
962 # needed to make any special characters, so show a raw string.
963 return 'r' + testrepr[0] + test + testrepr[0]
964 return testrepr
965
Skip Montanarodf708782002-03-07 22:58:02 +0000966 repr_str = repr_string
967
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000968 def repr_instance(self, x, level):
969 try:
Ka-Ping Yee1d384632001-03-01 00:24:32 +0000970 return cram(stripid(repr(x)), self.maxstring)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000971 except:
972 return '<%s instance>' % x.__class__.__name__
973
974class TextDoc(Doc):
975 """Formatter class for text documentation."""
976
977 # ------------------------------------------- text formatting utilities
978
979 _repr_instance = TextRepr()
980 repr = _repr_instance.repr
981
982 def bold(self, text):
983 """Format a string in bold by overstriking."""
984 return join(map(lambda ch: ch + '\b' + ch, text), '')
985
986 def indent(self, text, prefix=' '):
987 """Indent text by prepending a given prefix to each line."""
988 if not text: return ''
989 lines = split(text, '\n')
990 lines = map(lambda line, prefix=prefix: prefix + line, lines)
991 if lines: lines[-1] = rstrip(lines[-1])
992 return join(lines, '\n')
993
994 def section(self, title, contents):
995 """Format a section with a given heading."""
996 return self.bold(title) + '\n' + rstrip(self.indent(contents)) + '\n\n'
997
998 # ---------------------------------------------- type-specific routines
999
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001000 def formattree(self, tree, modname, parent=None, prefix=''):
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001001 """Render in text a class tree as returned by inspect.getclasstree()."""
1002 result = ''
1003 for entry in tree:
1004 if type(entry) is type(()):
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001005 c, bases = entry
1006 result = result + prefix + classname(c, modname)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001007 if bases and bases != (parent,):
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001008 parents = map(lambda c, m=modname: classname(c, m), bases)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001009 result = result + '(%s)' % join(parents, ', ')
1010 result = result + '\n'
1011 elif type(entry) is type([]):
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001012 result = result + self.formattree(
1013 entry, modname, c, prefix + ' ')
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001014 return result
1015
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001016 def docmodule(self, object, name=None, mod=None):
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001017 """Produce text documentation for a given module object."""
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001018 name = object.__name__ # ignore the passed-in name
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +00001019 synop, desc = splitdoc(getdoc(object))
1020 result = self.section('NAME', name + (synop and ' - ' + synop))
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001021
1022 try:
Skip Montanaroa5616d22004-06-11 04:46:12 +00001023 all = object.__all__
1024 except AttributeError:
1025 all = None
1026
1027 try:
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001028 file = inspect.getabsfile(object)
1029 except TypeError:
1030 file = '(built-in)'
Ka-Ping Yee239432a2001-03-02 02:45:08 +00001031 result = result + self.section('FILE', file)
Skip Montanaro4997a692003-09-10 16:47:51 +00001032
1033 docloc = self.getdocloc(object)
1034 if docloc is not None:
1035 result = result + self.section('MODULE DOCS', docloc)
1036
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +00001037 if desc:
1038 result = result + self.section('DESCRIPTION', desc)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001039
1040 classes = []
1041 for key, value in inspect.getmembers(object, inspect.isclass):
Johannes Gijsbers4c11f602004-08-30 14:13:04 +00001042 # if __all__ exists, believe it. Otherwise use old heuristic.
1043 if (all is not None
1044 or (inspect.getmodule(value) or object) is object):
Skip Montanaroa5616d22004-06-11 04:46:12 +00001045 if visiblename(key, all):
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +00001046 classes.append((key, value))
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001047 funcs = []
1048 for key, value in inspect.getmembers(object, inspect.isroutine):
Johannes Gijsbers4c11f602004-08-30 14:13:04 +00001049 # if __all__ exists, believe it. Otherwise use old heuristic.
1050 if (all is not None or
1051 inspect.isbuiltin(value) or inspect.getmodule(value) is object):
Skip Montanaroa5616d22004-06-11 04:46:12 +00001052 if visiblename(key, all):
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +00001053 funcs.append((key, value))
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001054 data = []
1055 for key, value in inspect.getmembers(object, isdata):
Skip Montanaroa5616d22004-06-11 04:46:12 +00001056 if visiblename(key, all):
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001057 data.append((key, value))
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001058
Georg Brandlf2dae0e2008-01-21 21:05:49 +00001059 modpkgs = []
1060 modpkgs_names = set()
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001061 if hasattr(object, '__path__'):
Phillip J. Ebyceb30872006-04-18 00:59:55 +00001062 for importer, modname, ispkg in pkgutil.iter_modules(object.__path__):
Georg Brandlf2dae0e2008-01-21 21:05:49 +00001063 modpkgs_names.add(modname)
Phillip J. Ebyceb30872006-04-18 00:59:55 +00001064 if ispkg:
1065 modpkgs.append(modname + ' (package)')
1066 else:
1067 modpkgs.append(modname)
1068
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001069 modpkgs.sort()
1070 result = result + self.section(
1071 'PACKAGE CONTENTS', join(modpkgs, '\n'))
1072
Georg Brandlf2dae0e2008-01-21 21:05:49 +00001073 # Detect submodules as sometimes created by C extensions
1074 submodules = []
1075 for key, value in inspect.getmembers(object, inspect.ismodule):
1076 if value.__name__.startswith(name + '.') and key not in modpkgs_names:
1077 submodules.append(key)
1078 if submodules:
1079 submodules.sort()
1080 result = result + self.section(
1081 'SUBMODULES', join(submodules, '\n'))
1082
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001083 if classes:
Brett Cannonaca98b02008-08-03 00:58:51 +00001084 classlist = map(lambda key_value: key_value[1], classes)
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001085 contents = [self.formattree(
1086 inspect.getclasstree(classlist, 1), name)]
1087 for key, value in classes:
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001088 contents.append(self.document(value, key, name))
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001089 result = result + self.section('CLASSES', join(contents, '\n'))
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001090
1091 if funcs:
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001092 contents = []
1093 for key, value in funcs:
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001094 contents.append(self.document(value, key, name))
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001095 result = result + self.section('FUNCTIONS', join(contents, '\n'))
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001096
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001097 if data:
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001098 contents = []
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001099 for key, value in data:
Georg Brandl8b813db2005-10-01 16:32:31 +00001100 contents.append(self.docother(value, key, name, maxlen=70))
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001101 result = result + self.section('DATA', join(contents, '\n'))
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001102
1103 if hasattr(object, '__version__'):
1104 version = str(object.__version__)
Ka-Ping Yee1d384632001-03-01 00:24:32 +00001105 if version[:11] == '$' + 'Revision: ' and version[-1:] == '$':
1106 version = strip(version[11:-1])
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001107 result = result + self.section('VERSION', version)
Ka-Ping Yee6f3f9a42001-02-27 22:42:36 +00001108 if hasattr(object, '__date__'):
1109 result = result + self.section('DATE', str(object.__date__))
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001110 if hasattr(object, '__author__'):
Ka-Ping Yee6f3f9a42001-02-27 22:42:36 +00001111 result = result + self.section('AUTHOR', str(object.__author__))
1112 if hasattr(object, '__credits__'):
1113 result = result + self.section('CREDITS', str(object.__credits__))
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001114 return result
1115
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001116 def docclass(self, object, name=None, mod=None):
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001117 """Produce text documentation for a given class object."""
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001118 realname = object.__name__
1119 name = name or realname
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001120 bases = object.__bases__
1121
Tim Petersc86f6ca2001-09-26 21:31:51 +00001122 def makename(c, m=object.__module__):
1123 return classname(c, m)
1124
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001125 if name == realname:
1126 title = 'class ' + self.bold(realname)
1127 else:
1128 title = self.bold(name) + ' = class ' + realname
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001129 if bases:
Ka-Ping Yee3bda8792001-03-23 13:17:50 +00001130 parents = map(makename, bases)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001131 title = title + '(%s)' % join(parents, ', ')
1132
1133 doc = getdoc(object)
Tim Peters28355492001-09-23 21:29:55 +00001134 contents = doc and [doc + '\n'] or []
1135 push = contents.append
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001136
Tim Petersc86f6ca2001-09-26 21:31:51 +00001137 # List the mro, if non-trivial.
Raymond Hettinger756b3f32004-01-29 06:37:52 +00001138 mro = deque(inspect.getmro(object))
Tim Petersc86f6ca2001-09-26 21:31:51 +00001139 if len(mro) > 2:
1140 push("Method resolution order:")
1141 for base in mro:
1142 push(' ' + makename(base))
1143 push('')
1144
Tim Petersf4aad8e2001-09-24 22:40:47 +00001145 # Cute little class to pump out a horizontal rule between sections.
1146 class HorizontalRule:
1147 def __init__(self):
1148 self.needone = 0
1149 def maybe(self):
1150 if self.needone:
1151 push('-' * 70)
1152 self.needone = 1
1153 hr = HorizontalRule()
1154
Tim Peters28355492001-09-23 21:29:55 +00001155 def spill(msg, attrs, predicate):
Tim Petersfa26f7c2001-09-24 08:05:11 +00001156 ok, attrs = _split_list(attrs, predicate)
Tim Peters28355492001-09-23 21:29:55 +00001157 if ok:
Tim Petersf4aad8e2001-09-24 22:40:47 +00001158 hr.maybe()
Tim Peters28355492001-09-23 21:29:55 +00001159 push(msg)
1160 for name, kind, homecls, value in ok:
1161 push(self.document(getattr(object, name),
1162 name, mod, object))
1163 return attrs
1164
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +00001165 def spilldescriptors(msg, attrs, predicate):
Tim Petersfa26f7c2001-09-24 08:05:11 +00001166 ok, attrs = _split_list(attrs, predicate)
Tim Peters28355492001-09-23 21:29:55 +00001167 if ok:
Tim Petersf4aad8e2001-09-24 22:40:47 +00001168 hr.maybe()
Tim Peters28355492001-09-23 21:29:55 +00001169 push(msg)
1170 for name, kind, homecls, value in ok:
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +00001171 push(self._docdescriptor(name, value, mod))
Tim Peters28355492001-09-23 21:29:55 +00001172 return attrs
Tim Petersb47879b2001-09-24 04:47:19 +00001173
Tim Petersfa26f7c2001-09-24 08:05:11 +00001174 def spilldata(msg, attrs, predicate):
1175 ok, attrs = _split_list(attrs, predicate)
Tim Peters28355492001-09-23 21:29:55 +00001176 if ok:
Tim Petersf4aad8e2001-09-24 22:40:47 +00001177 hr.maybe()
Tim Peters28355492001-09-23 21:29:55 +00001178 push(msg)
1179 for name, kind, homecls, value in ok:
Brett Cannonaca98b02008-08-03 00:58:51 +00001180 if (hasattr(value, '__call__') or
1181 inspect.isdatadescriptor(value)):
Ka-Ping Yeebba6acc2005-02-19 22:58:26 +00001182 doc = getdoc(value)
Guido van Rossum5e355b22002-05-21 20:56:15 +00001183 else:
1184 doc = None
Tim Peters28355492001-09-23 21:29:55 +00001185 push(self.docother(getattr(object, name),
Georg Brandl8b813db2005-10-01 16:32:31 +00001186 name, mod, maxlen=70, doc=doc) + '\n')
Tim Peters28355492001-09-23 21:29:55 +00001187 return attrs
1188
Brett Cannonaca98b02008-08-03 00:58:51 +00001189 attrs = filter(lambda data: visiblename(data[0]),
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +00001190 classify_class_attrs(object))
Tim Petersfa26f7c2001-09-24 08:05:11 +00001191 while attrs:
Tim Peters351e3622001-09-27 03:29:51 +00001192 if mro:
Raymond Hettinger756b3f32004-01-29 06:37:52 +00001193 thisclass = mro.popleft()
Tim Peters351e3622001-09-27 03:29:51 +00001194 else:
1195 thisclass = attrs[0][2]
Tim Petersfa26f7c2001-09-24 08:05:11 +00001196 attrs, inherited = _split_list(attrs, lambda t: t[2] is thisclass)
1197
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +00001198 if thisclass is __builtin__.object:
1199 attrs = inherited
1200 continue
1201 elif thisclass is object:
Tim Peters28355492001-09-23 21:29:55 +00001202 tag = "defined here"
1203 else:
Tim Petersfa26f7c2001-09-24 08:05:11 +00001204 tag = "inherited from %s" % classname(thisclass,
1205 object.__module__)
Tim Peters28355492001-09-23 21:29:55 +00001206
1207 # Sort attrs by name.
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +00001208 attrs.sort()
Tim Peters28355492001-09-23 21:29:55 +00001209
1210 # Pump out the attrs, segregated by kind.
Tim Petersf4aad8e2001-09-24 22:40:47 +00001211 attrs = spill("Methods %s:\n" % tag, attrs,
Tim Peters28355492001-09-23 21:29:55 +00001212 lambda t: t[1] == 'method')
Tim Petersf4aad8e2001-09-24 22:40:47 +00001213 attrs = spill("Class methods %s:\n" % tag, attrs,
Tim Peters28355492001-09-23 21:29:55 +00001214 lambda t: t[1] == 'class method')
Tim Petersf4aad8e2001-09-24 22:40:47 +00001215 attrs = spill("Static methods %s:\n" % tag, attrs,
Tim Peters28355492001-09-23 21:29:55 +00001216 lambda t: t[1] == 'static method')
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +00001217 attrs = spilldescriptors("Data descriptors %s:\n" % tag, attrs,
1218 lambda t: t[1] == 'data descriptor')
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +00001219 attrs = spilldata("Data and other attributes %s:\n" % tag, attrs,
1220 lambda t: t[1] == 'data')
Tim Peters28355492001-09-23 21:29:55 +00001221 assert attrs == []
Tim Peters351e3622001-09-27 03:29:51 +00001222 attrs = inherited
Tim Peters28355492001-09-23 21:29:55 +00001223
1224 contents = '\n'.join(contents)
1225 if not contents:
1226 return title + '\n'
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001227 return title + '\n' + self.indent(rstrip(contents), ' | ') + '\n'
1228
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001229 def formatvalue(self, object):
1230 """Format an argument default value as text."""
1231 return '=' + self.repr(object)
1232
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001233 def docroutine(self, object, name=None, mod=None, cl=None):
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001234 """Produce text documentation for a function or method object."""
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001235 realname = object.__name__
1236 name = name or realname
1237 note = ''
Ka-Ping Yee3bda8792001-03-23 13:17:50 +00001238 skipdocs = 0
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001239 if inspect.ismethod(object):
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001240 imclass = object.im_class
Ka-Ping Yee3bda8792001-03-23 13:17:50 +00001241 if cl:
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001242 if imclass is not cl:
1243 note = ' from ' + classname(imclass, mod)
Ka-Ping Yee3bda8792001-03-23 13:17:50 +00001244 else:
Georg Brandl7e2b6bb2007-03-13 22:16:30 +00001245 if object.im_self is not None:
Ka-Ping Yeeb7a48302001-04-12 20:39:14 +00001246 note = ' method of %s instance' % classname(
1247 object.im_self.__class__, mod)
1248 else:
1249 note = ' unbound %s method' % classname(imclass,mod)
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001250 object = object.im_func
1251
1252 if name == realname:
1253 title = self.bold(realname)
1254 else:
Raymond Hettinger54f02222002-06-01 14:18:47 +00001255 if (cl and realname in cl.__dict__ and
Ka-Ping Yee3bda8792001-03-23 13:17:50 +00001256 cl.__dict__[realname] is object):
1257 skipdocs = 1
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001258 title = self.bold(name) + ' = ' + realname
Tim Peters4bcfa312001-09-20 06:08:24 +00001259 if inspect.isfunction(object):
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001260 args, varargs, varkw, defaults = inspect.getargspec(object)
1261 argspec = inspect.formatargspec(
1262 args, varargs, varkw, defaults, formatvalue=self.formatvalue)
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001263 if realname == '<lambda>':
Georg Brandl501dd0d2006-02-17 09:45:40 +00001264 title = self.bold(name) + ' lambda '
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001265 argspec = argspec[1:-1] # remove parentheses
Tim Peters4bcfa312001-09-20 06:08:24 +00001266 else:
1267 argspec = '(...)'
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001268 decl = title + argspec + note
1269
Ka-Ping Yee3bda8792001-03-23 13:17:50 +00001270 if skipdocs:
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001271 return decl + '\n'
Ka-Ping Yee3bda8792001-03-23 13:17:50 +00001272 else:
1273 doc = getdoc(object) or ''
1274 return decl + '\n' + (doc and rstrip(self.indent(doc)) + '\n')
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001275
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +00001276 def _docdescriptor(self, name, value, mod):
Johannes Gijsbers8de645a2004-11-07 19:16:05 +00001277 results = []
1278 push = results.append
1279
1280 if name:
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +00001281 push(self.bold(name))
1282 push('\n')
Johannes Gijsbers8de645a2004-11-07 19:16:05 +00001283 doc = getdoc(value) or ''
1284 if doc:
1285 push(self.indent(doc))
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +00001286 push('\n')
1287 return ''.join(results)
Johannes Gijsbers8de645a2004-11-07 19:16:05 +00001288
1289 def docproperty(self, object, name=None, mod=None, cl=None):
1290 """Produce text documentation for a property."""
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +00001291 return self._docdescriptor(name, object, mod)
Johannes Gijsbers8de645a2004-11-07 19:16:05 +00001292
Barry Warsaw00decd72006-07-27 23:43:15 +00001293 def docdata(self, object, name=None, mod=None, cl=None):
1294 """Produce text documentation for a data descriptor."""
1295 return self._docdescriptor(name, object, mod)
1296
Georg Brandl8b813db2005-10-01 16:32:31 +00001297 def docother(self, object, name=None, mod=None, parent=None, maxlen=None, doc=None):
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001298 """Produce text documentation for a data object."""
1299 repr = self.repr(object)
1300 if maxlen:
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001301 line = (name and name + ' = ' or '') + repr
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001302 chop = maxlen - len(line)
1303 if chop < 0: repr = repr[:chop] + '...'
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001304 line = (name and self.bold(name) + ' = ' or '') + repr
Tim Peters28355492001-09-23 21:29:55 +00001305 if doc is not None:
1306 line += '\n' + self.indent(str(doc))
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001307 return line
1308
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001309# --------------------------------------------------------- user interfaces
1310
1311def pager(text):
1312 """The first time this is called, determine what kind of pager to use."""
1313 global pager
1314 pager = getpager()
1315 pager(text)
1316
1317def getpager():
1318 """Decide what method to use for paging through text."""
1319 if type(sys.stdout) is not types.FileType:
1320 return plainpager
1321 if not sys.stdin.isatty() or not sys.stdout.isatty():
1322 return plainpager
Raymond Hettinger54f02222002-06-01 14:18:47 +00001323 if 'PAGER' in os.environ:
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +00001324 if sys.platform == 'win32': # pipes completely broken in Windows
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001325 return lambda text: tempfilepager(plain(text), os.environ['PAGER'])
Raymond Hettingerdbecd932005-02-06 06:57:08 +00001326 elif os.environ.get('TERM') in ('dumb', 'emacs'):
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001327 return lambda text: pipepager(plain(text), os.environ['PAGER'])
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +00001328 else:
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001329 return lambda text: pipepager(text, os.environ['PAGER'])
Ka-Ping Yeea487e4e2005-11-05 04:49:18 +00001330 if os.environ.get('TERM') in ('dumb', 'emacs'):
1331 return plainpager
Andrew MacIntyre54e0eab2002-03-03 03:12:30 +00001332 if sys.platform == 'win32' or sys.platform.startswith('os2'):
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001333 return lambda text: tempfilepager(plain(text), 'more <')
Skip Montanarod404bee2002-09-26 21:44:57 +00001334 if hasattr(os, 'system') and os.system('(less) 2>/dev/null') == 0:
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001335 return lambda text: pipepager(text, 'less')
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001336
1337 import tempfile
Guido van Rossum3b0a3292002-08-09 16:38:32 +00001338 (fd, filename) = tempfile.mkstemp()
1339 os.close(fd)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001340 try:
Georg Brandlb32dea52008-07-16 21:19:28 +00001341 if hasattr(os, 'system') and os.system('more "%s"' % filename) == 0:
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001342 return lambda text: pipepager(text, 'more')
1343 else:
1344 return ttypager
1345 finally:
1346 os.unlink(filename)
1347
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001348def plain(text):
1349 """Remove boldface formatting from text."""
1350 return re.sub('.\b', '', text)
1351
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001352def pipepager(text, cmd):
1353 """Page through text by feeding it to another program."""
1354 pipe = os.popen(cmd, 'w')
1355 try:
1356 pipe.write(text)
1357 pipe.close()
1358 except IOError:
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001359 pass # Ignore broken pipes caused by quitting the pager program.
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001360
1361def tempfilepager(text, cmd):
1362 """Page through text by invoking a program on a temporary file."""
1363 import tempfile
Tim Peters550e4e52003-02-07 01:53:46 +00001364 filename = tempfile.mktemp()
1365 file = open(filename, 'w')
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001366 file.write(text)
1367 file.close()
1368 try:
Georg Brandlb32dea52008-07-16 21:19:28 +00001369 os.system(cmd + ' "' + filename + '"')
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001370 finally:
1371 os.unlink(filename)
1372
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001373def ttypager(text):
1374 """Page through text on a text terminal."""
1375 lines = split(plain(text), '\n')
1376 try:
1377 import tty
1378 fd = sys.stdin.fileno()
1379 old = tty.tcgetattr(fd)
1380 tty.setcbreak(fd)
1381 getchar = lambda: sys.stdin.read(1)
Ka-Ping Yee457aab22001-02-27 23:36:29 +00001382 except (ImportError, AttributeError):
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001383 tty = None
1384 getchar = lambda: sys.stdin.readline()[:-1][:1]
1385
1386 try:
1387 r = inc = os.environ.get('LINES', 25) - 1
1388 sys.stdout.write(join(lines[:inc], '\n') + '\n')
1389 while lines[r:]:
1390 sys.stdout.write('-- more --')
1391 sys.stdout.flush()
1392 c = getchar()
1393
Raymond Hettingerdbecd932005-02-06 06:57:08 +00001394 if c in ('q', 'Q'):
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001395 sys.stdout.write('\r \r')
1396 break
Raymond Hettingerdbecd932005-02-06 06:57:08 +00001397 elif c in ('\r', '\n'):
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001398 sys.stdout.write('\r \r' + lines[r] + '\n')
1399 r = r + 1
1400 continue
Raymond Hettingerdbecd932005-02-06 06:57:08 +00001401 if c in ('b', 'B', '\x1b'):
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001402 r = r - inc - inc
1403 if r < 0: r = 0
1404 sys.stdout.write('\n' + join(lines[r:r+inc], '\n') + '\n')
1405 r = r + inc
1406
1407 finally:
1408 if tty:
1409 tty.tcsetattr(fd, tty.TCSAFLUSH, old)
1410
1411def plainpager(text):
1412 """Simply print unformatted text. This is the ultimate fallback."""
1413 sys.stdout.write(plain(text))
1414
1415def describe(thing):
Ka-Ping Yee3bda8792001-03-23 13:17:50 +00001416 """Produce a short description of the given thing."""
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001417 if inspect.ismodule(thing):
1418 if thing.__name__ in sys.builtin_module_names:
1419 return 'built-in module ' + thing.__name__
1420 if hasattr(thing, '__path__'):
1421 return 'package ' + thing.__name__
1422 else:
1423 return 'module ' + thing.__name__
1424 if inspect.isbuiltin(thing):
1425 return 'built-in function ' + thing.__name__
Barry Warsaw00decd72006-07-27 23:43:15 +00001426 if inspect.isgetsetdescriptor(thing):
1427 return 'getset descriptor %s.%s.%s' % (
1428 thing.__objclass__.__module__, thing.__objclass__.__name__,
1429 thing.__name__)
1430 if inspect.ismemberdescriptor(thing):
1431 return 'member descriptor %s.%s.%s' % (
1432 thing.__objclass__.__module__, thing.__objclass__.__name__,
1433 thing.__name__)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001434 if inspect.isclass(thing):
1435 return 'class ' + thing.__name__
1436 if inspect.isfunction(thing):
1437 return 'function ' + thing.__name__
1438 if inspect.ismethod(thing):
1439 return 'method ' + thing.__name__
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001440 if type(thing) is types.InstanceType:
1441 return 'instance of ' + thing.__class__.__name__
1442 return type(thing).__name__
1443
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001444def locate(path, forceload=0):
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001445 """Locate an object by name or dotted path, importing as necessary."""
Guido van Rossum97dede02003-02-16 01:12:32 +00001446 parts = [part for part in split(path, '.') if part]
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001447 module, n = None, 0
1448 while n < len(parts):
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001449 nextmodule = safeimport(join(parts[:n+1], '.'), forceload)
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001450 if nextmodule: module, n = nextmodule, n + 1
1451 else: break
1452 if module:
1453 object = module
1454 for part in parts[n:]:
1455 try: object = getattr(object, part)
1456 except AttributeError: return None
1457 return object
1458 else:
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001459 if hasattr(__builtin__, path):
1460 return getattr(__builtin__, path)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001461
1462# --------------------------------------- interactive interpreter interface
1463
1464text = TextDoc()
1465html = HTMLDoc()
1466
Ka-Ping Yee8ef1cf32007-01-14 04:25:15 +00001467class _OldStyleClass: pass
1468_OLD_INSTANCE_TYPE = type(_OldStyleClass())
1469
Ka-Ping Yee45daeb02002-08-11 15:11:33 +00001470def resolve(thing, forceload=0):
1471 """Given an object or a path to an object, get the object and its name."""
1472 if isinstance(thing, str):
1473 object = locate(thing, forceload)
1474 if not object:
1475 raise ImportError, 'no Python documentation found for %r' % thing
1476 return object, thing
1477 else:
1478 return thing, getattr(thing, '__name__', None)
1479
Georg Brandl8441f152007-03-13 20:02:57 +00001480def render_doc(thing, title='Python Library Documentation: %s', forceload=0):
1481 """Render text documentation, given an object or a path to an object."""
1482 object, name = resolve(thing, forceload)
1483 desc = describe(object)
1484 module = inspect.getmodule(object)
1485 if name and '.' in name:
1486 desc += ' in ' + name[:name.rfind('.')]
1487 elif module and module is not object:
1488 desc += ' in module ' + module.__name__
1489 if type(object) is _OLD_INSTANCE_TYPE:
1490 # If the passed object is an instance of an old-style class,
1491 # document its available methods instead of its value.
1492 object = object.__class__
1493 elif not (inspect.ismodule(object) or
1494 inspect.isclass(object) or
1495 inspect.isroutine(object) or
1496 inspect.isgetsetdescriptor(object) or
1497 inspect.ismemberdescriptor(object) or
1498 isinstance(object, property)):
1499 # If the passed object is a piece of data or an instance,
1500 # document its available methods instead of its value.
1501 object = type(object)
1502 desc += ' object'
1503 return title % desc + '\n\n' + text.document(object, name)
1504
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001505def doc(thing, title='Python Library Documentation: %s', forceload=0):
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001506 """Display text documentation, given an object or a path to an object."""
Ka-Ping Yee45daeb02002-08-11 15:11:33 +00001507 try:
Georg Brandl8441f152007-03-13 20:02:57 +00001508 pager(render_doc(thing, title, forceload))
Ka-Ping Yee45daeb02002-08-11 15:11:33 +00001509 except (ImportError, ErrorDuringImport), value:
1510 print value
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001511
Ka-Ping Yee45daeb02002-08-11 15:11:33 +00001512def writedoc(thing, forceload=0):
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001513 """Write HTML documentation to a file in the current directory."""
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001514 try:
Ka-Ping Yee45daeb02002-08-11 15:11:33 +00001515 object, name = resolve(thing, forceload)
1516 page = html.page(describe(object), html.document(object, name))
1517 file = open(name + '.html', 'w')
1518 file.write(page)
1519 file.close()
1520 print 'wrote', name + '.html'
1521 except (ImportError, ErrorDuringImport), value:
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001522 print value
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001523
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001524def writedocs(dir, pkgpath='', done=None):
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001525 """Write out HTML documentation for all modules in a directory tree."""
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001526 if done is None: done = {}
Phillip J. Ebyceb30872006-04-18 00:59:55 +00001527 for importer, modname, ispkg in pkgutil.walk_packages([dir], pkgpath):
1528 writedoc(modname)
1529 return
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001530
1531class Helper:
Georg Brandl681001e2008-06-01 20:33:55 +00001532
1533 # These dictionaries map a topic name to either an alias, or a tuple
1534 # (label, seealso-items). The "label" is the label of the corresponding
1535 # section in the .rst file under Doc/ and an index into the dictionary
1536 # in pydoc_topics.py.
1537 #
1538 # CAUTION: if you change one of these dictionaries, be sure to adapt the
1539 # list of needed labels in Doc/tools/sphinxext/pyspecific.py and
1540 # regenerate the pydoc_topics.py file by running
1541 # make pydoc-topics
1542 # in Doc/ and copying the output file into the Lib/ directory.
1543
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001544 keywords = {
1545 'and': 'BOOLEAN',
Georg Brandlb6a87542007-03-13 10:06:48 +00001546 'as': 'with',
Georg Brandl681001e2008-06-01 20:33:55 +00001547 'assert': ('assert', ''),
1548 'break': ('break', 'while for'),
1549 'class': ('class', 'CLASSES SPECIALMETHODS'),
1550 'continue': ('continue', 'while for'),
1551 'def': ('function', ''),
1552 'del': ('del', 'BASICMETHODS'),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001553 'elif': 'if',
Georg Brandl681001e2008-06-01 20:33:55 +00001554 'else': ('else', 'while for'),
Georg Brandl5c669db2008-08-30 19:03:43 +00001555 'except': 'try',
Georg Brandl681001e2008-06-01 20:33:55 +00001556 'exec': ('exec', ''),
Georg Brandl5c669db2008-08-30 19:03:43 +00001557 'finally': 'try',
Georg Brandl681001e2008-06-01 20:33:55 +00001558 'for': ('for', 'break continue while'),
Georg Brandl5c669db2008-08-30 19:03:43 +00001559 'from': 'import',
Georg Brandl681001e2008-06-01 20:33:55 +00001560 'global': ('global', 'NAMESPACES'),
1561 'if': ('if', 'TRUTHVALUE'),
1562 'import': ('import', 'MODULES'),
1563 'in': ('in', 'SEQUENCEMETHODS2'),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001564 'is': 'COMPARISON',
Georg Brandl681001e2008-06-01 20:33:55 +00001565 'lambda': ('lambda', 'FUNCTIONS'),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001566 'not': 'BOOLEAN',
1567 'or': 'BOOLEAN',
Georg Brandl681001e2008-06-01 20:33:55 +00001568 'pass': ('pass', ''),
1569 'print': ('print', ''),
1570 'raise': ('raise', 'EXCEPTIONS'),
1571 'return': ('return', 'FUNCTIONS'),
1572 'try': ('try', 'EXCEPTIONS'),
1573 'while': ('while', 'break continue if TRUTHVALUE'),
1574 'with': ('with', 'CONTEXTMANAGERS EXCEPTIONS yield'),
1575 'yield': ('yield', ''),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001576 }
Georg Brandl47fe9812009-01-01 15:46:10 +00001577 # Either add symbols to this dictionary or to the symbols dictionary
1578 # directly: Whichever is easier. They are merged later.
1579 _symbols_inverse = {
1580 'STRINGS' : ("'", "'''", "r'", "u'", '"""', '"', 'r"', 'u"'),
1581 'OPERATORS' : ('+', '-', '*', '**', '/', '//', '%', '<<', '>>', '&',
1582 '|', '^', '~', '<', '>', '<=', '>=', '==', '!=', '<>'),
1583 'COMPARISON' : ('<', '>', '<=', '>=', '==', '!=', '<>'),
1584 'UNARY' : ('-', '~'),
1585 'AUGMENTEDASSIGNMENT' : ('+=', '-=', '*=', '/=', '%=', '&=', '|=',
1586 '^=', '<<=', '>>=', '**=', '//='),
1587 'BITWISE' : ('<<', '>>', '&', '|', '^', '~'),
1588 'COMPLEX' : ('j', 'J')
1589 }
1590 symbols = {
1591 '%': 'OPERATORS FORMATTING',
1592 '**': 'POWER',
1593 ',': 'TUPLES LISTS FUNCTIONS',
1594 '.': 'ATTRIBUTES FLOAT MODULES OBJECTS',
1595 '...': 'ELLIPSIS',
1596 ':': 'SLICINGS DICTIONARYLITERALS',
1597 '@': 'def class',
1598 '\\': 'STRINGS',
1599 '_': 'PRIVATENAMES',
1600 '__': 'PRIVATENAMES SPECIALMETHODS',
1601 '`': 'BACKQUOTES',
1602 '(': 'TUPLES FUNCTIONS CALLS',
1603 ')': 'TUPLES FUNCTIONS CALLS',
1604 '[': 'LISTS SUBSCRIPTS SLICINGS',
1605 ']': 'LISTS SUBSCRIPTS SLICINGS'
1606 }
1607 for topic, symbols_ in _symbols_inverse.iteritems():
1608 for symbol in symbols_:
1609 topics = symbols.get(symbol, topic)
1610 if topic not in topics:
1611 topics = topics + ' ' + topic
1612 symbols[symbol] = topics
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001613
1614 topics = {
Georg Brandl681001e2008-06-01 20:33:55 +00001615 'TYPES': ('types', 'STRINGS UNICODE NUMBERS SEQUENCES MAPPINGS '
1616 'FUNCTIONS CLASSES MODULES FILES inspect'),
1617 'STRINGS': ('strings', 'str UNICODE SEQUENCES STRINGMETHODS FORMATTING '
1618 'TYPES'),
1619 'STRINGMETHODS': ('string-methods', 'STRINGS FORMATTING'),
1620 'FORMATTING': ('formatstrings', 'OPERATORS'),
1621 'UNICODE': ('strings', 'encodings unicode SEQUENCES STRINGMETHODS '
1622 'FORMATTING TYPES'),
1623 'NUMBERS': ('numbers', 'INTEGER FLOAT COMPLEX TYPES'),
1624 'INTEGER': ('integers', 'int range'),
1625 'FLOAT': ('floating', 'float math'),
1626 'COMPLEX': ('imaginary', 'complex cmath'),
1627 'SEQUENCES': ('typesseq', 'STRINGMETHODS FORMATTING xrange LISTS'),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001628 'MAPPINGS': 'DICTIONARIES',
Georg Brandl681001e2008-06-01 20:33:55 +00001629 'FUNCTIONS': ('typesfunctions', 'def TYPES'),
1630 'METHODS': ('typesmethods', 'class def CLASSES TYPES'),
1631 'CODEOBJECTS': ('bltin-code-objects', 'compile FUNCTIONS TYPES'),
1632 'TYPEOBJECTS': ('bltin-type-objects', 'types TYPES'),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001633 'FRAMEOBJECTS': 'TYPES',
1634 'TRACEBACKS': 'TYPES',
Georg Brandl681001e2008-06-01 20:33:55 +00001635 'NONE': ('bltin-null-object', ''),
1636 'ELLIPSIS': ('bltin-ellipsis-object', 'SLICINGS'),
1637 'FILES': ('bltin-file-objects', ''),
1638 'SPECIALATTRIBUTES': ('specialattrs', ''),
1639 'CLASSES': ('types', 'class SPECIALMETHODS PRIVATENAMES'),
1640 'MODULES': ('typesmodules', 'import'),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001641 'PACKAGES': 'import',
Georg Brandl681001e2008-06-01 20:33:55 +00001642 'EXPRESSIONS': ('operator-summary', 'lambda or and not in is BOOLEAN '
1643 'COMPARISON BITWISE SHIFTING BINARY FORMATTING POWER '
1644 'UNARY ATTRIBUTES SUBSCRIPTS SLICINGS CALLS TUPLES '
1645 'LISTS DICTIONARIES BACKQUOTES'),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001646 'OPERATORS': 'EXPRESSIONS',
1647 'PRECEDENCE': 'EXPRESSIONS',
Georg Brandl681001e2008-06-01 20:33:55 +00001648 'OBJECTS': ('objects', 'TYPES'),
1649 'SPECIALMETHODS': ('specialnames', 'BASICMETHODS ATTRIBUTEMETHODS '
1650 'CALLABLEMETHODS SEQUENCEMETHODS1 MAPPINGMETHODS '
1651 'SEQUENCEMETHODS2 NUMBERMETHODS CLASSES'),
1652 'BASICMETHODS': ('customization', 'cmp hash repr str SPECIALMETHODS'),
1653 'ATTRIBUTEMETHODS': ('attribute-access', 'ATTRIBUTES SPECIALMETHODS'),
1654 'CALLABLEMETHODS': ('callable-types', 'CALLS SPECIALMETHODS'),
1655 'SEQUENCEMETHODS1': ('sequence-types', 'SEQUENCES SEQUENCEMETHODS2 '
1656 'SPECIALMETHODS'),
1657 'SEQUENCEMETHODS2': ('sequence-methods', 'SEQUENCES SEQUENCEMETHODS1 '
1658 'SPECIALMETHODS'),
1659 'MAPPINGMETHODS': ('sequence-types', 'MAPPINGS SPECIALMETHODS'),
1660 'NUMBERMETHODS': ('numeric-types', 'NUMBERS AUGMENTEDASSIGNMENT '
1661 'SPECIALMETHODS'),
1662 'EXECUTION': ('execmodel', 'NAMESPACES DYNAMICFEATURES EXCEPTIONS'),
1663 'NAMESPACES': ('naming', 'global ASSIGNMENT DELETION DYNAMICFEATURES'),
1664 'DYNAMICFEATURES': ('dynamic-features', ''),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001665 'SCOPING': 'NAMESPACES',
1666 'FRAMES': 'NAMESPACES',
Georg Brandl681001e2008-06-01 20:33:55 +00001667 'EXCEPTIONS': ('exceptions', 'try except finally raise'),
1668 'COERCIONS': ('coercion-rules','CONVERSIONS'),
1669 'CONVERSIONS': ('conversions', 'COERCIONS'),
1670 'IDENTIFIERS': ('identifiers', 'keywords SPECIALIDENTIFIERS'),
1671 'SPECIALIDENTIFIERS': ('id-classes', ''),
1672 'PRIVATENAMES': ('atom-identifiers', ''),
1673 'LITERALS': ('atom-literals', 'STRINGS BACKQUOTES NUMBERS '
1674 'TUPLELITERALS LISTLITERALS DICTIONARYLITERALS'),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001675 'TUPLES': 'SEQUENCES',
Georg Brandl681001e2008-06-01 20:33:55 +00001676 'TUPLELITERALS': ('exprlists', 'TUPLES LITERALS'),
1677 'LISTS': ('typesseq-mutable', 'LISTLITERALS'),
1678 'LISTLITERALS': ('lists', 'LISTS LITERALS'),
1679 'DICTIONARIES': ('typesmapping', 'DICTIONARYLITERALS'),
1680 'DICTIONARYLITERALS': ('dict', 'DICTIONARIES LITERALS'),
1681 'BACKQUOTES': ('string-conversions', 'repr str STRINGS LITERALS'),
1682 'ATTRIBUTES': ('attribute-references', 'getattr hasattr setattr '
1683 'ATTRIBUTEMETHODS'),
1684 'SUBSCRIPTS': ('subscriptions', 'SEQUENCEMETHODS1'),
1685 'SLICINGS': ('slicings', 'SEQUENCEMETHODS2'),
1686 'CALLS': ('calls', 'EXPRESSIONS'),
1687 'POWER': ('power', 'EXPRESSIONS'),
1688 'UNARY': ('unary', 'EXPRESSIONS'),
1689 'BINARY': ('binary', 'EXPRESSIONS'),
1690 'SHIFTING': ('shifting', 'EXPRESSIONS'),
1691 'BITWISE': ('bitwise', 'EXPRESSIONS'),
1692 'COMPARISON': ('comparisons', 'EXPRESSIONS BASICMETHODS'),
1693 'BOOLEAN': ('booleans', 'EXPRESSIONS TRUTHVALUE'),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001694 'ASSERTION': 'assert',
Georg Brandl681001e2008-06-01 20:33:55 +00001695 'ASSIGNMENT': ('assignment', 'AUGMENTEDASSIGNMENT'),
1696 'AUGMENTEDASSIGNMENT': ('augassign', 'NUMBERMETHODS'),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001697 'DELETION': 'del',
1698 'PRINTING': 'print',
1699 'RETURNING': 'return',
1700 'IMPORTING': 'import',
1701 'CONDITIONAL': 'if',
Georg Brandl681001e2008-06-01 20:33:55 +00001702 'LOOPING': ('compound', 'for while break continue'),
1703 'TRUTHVALUE': ('truth', 'if while and or not BASICMETHODS'),
1704 'DEBUGGING': ('debugger', 'pdb'),
1705 'CONTEXTMANAGERS': ('context-managers', 'with'),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001706 }
1707
Georg Brandl03c1cff2010-08-01 22:05:31 +00001708 def __init__(self, input=None, output=None):
1709 self._input = input
1710 self._output = output
1711
1712 input = property(lambda self: self._input or sys.stdin)
1713 output = property(lambda self: self._output or sys.stdout)
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001714
Ka-Ping Yee79c009d2001-04-13 10:53:25 +00001715 def __repr__(self):
Ka-Ping Yee9bc576b2001-04-13 13:57:31 +00001716 if inspect.stack()[1][3] == '?':
Ka-Ping Yee79c009d2001-04-13 10:53:25 +00001717 self()
1718 return ''
Ka-Ping Yee9bc576b2001-04-13 13:57:31 +00001719 return '<pydoc.Helper instance>'
Ka-Ping Yee79c009d2001-04-13 10:53:25 +00001720
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001721 def __call__(self, request=None):
1722 if request is not None:
1723 self.help(request)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001724 else:
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001725 self.intro()
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001726 self.interact()
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001727 self.output.write('''
Fred Drakee61967f2001-05-10 18:41:02 +00001728You are now leaving help and returning to the Python interpreter.
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001729If you want to ask for help on a particular object directly from the
1730interpreter, you can type "help(object)". Executing "help('string')"
1731has the same effect as typing a particular string at the help> prompt.
1732''')
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001733
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001734 def interact(self):
1735 self.output.write('\n')
Guido van Rossum8ca162f2002-04-07 06:36:23 +00001736 while True:
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001737 try:
Johannes Gijsberse7691d32004-08-17 13:21:53 +00001738 request = self.getline('help> ')
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001739 if not request: break
Johannes Gijsberse7691d32004-08-17 13:21:53 +00001740 except (KeyboardInterrupt, EOFError):
1741 break
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001742 request = strip(replace(request, '"', '', "'", ''))
Raymond Hettingerdbecd932005-02-06 06:57:08 +00001743 if lower(request) in ('q', 'quit'): break
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001744 self.help(request)
1745
Johannes Gijsberse7691d32004-08-17 13:21:53 +00001746 def getline(self, prompt):
1747 """Read one line, using raw_input when available."""
1748 if self.input is sys.stdin:
1749 return raw_input(prompt)
1750 else:
1751 self.output.write(prompt)
1752 self.output.flush()
1753 return self.input.readline()
1754
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001755 def help(self, request):
1756 if type(request) is type(''):
R. David Murray2edb85d2009-05-27 20:16:24 +00001757 request = request.strip()
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001758 if request == 'help': self.intro()
1759 elif request == 'keywords': self.listkeywords()
Georg Brandl47fe9812009-01-01 15:46:10 +00001760 elif request == 'symbols': self.listsymbols()
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001761 elif request == 'topics': self.listtopics()
1762 elif request == 'modules': self.listmodules()
1763 elif request[:8] == 'modules ':
1764 self.listmodules(split(request)[1])
Georg Brandl47fe9812009-01-01 15:46:10 +00001765 elif request in self.symbols: self.showsymbol(request)
Raymond Hettinger54f02222002-06-01 14:18:47 +00001766 elif request in self.keywords: self.showtopic(request)
1767 elif request in self.topics: self.showtopic(request)
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001768 elif request: doc(request, 'Help on %s:')
1769 elif isinstance(request, Helper): self()
1770 else: doc(request, 'Help on %s:')
1771 self.output.write('\n')
1772
1773 def intro(self):
1774 self.output.write('''
1775Welcome to Python %s! This is the online help utility.
1776
1777If this is your first time using Python, you should definitely check out
Georg Brandl0751d1a2008-01-21 17:13:03 +00001778the tutorial on the Internet at http://docs.python.org/tutorial/.
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001779
1780Enter the name of any module, keyword, or topic to get help on writing
1781Python programs and using Python modules. To quit this help utility and
1782return to the interpreter, just type "quit".
1783
1784To get a list of available modules, keywords, or topics, type "modules",
1785"keywords", or "topics". Each module also comes with a one-line summary
1786of what it does; to list the modules whose summaries contain a given word
1787such as "spam", type "modules spam".
1788''' % sys.version[:3])
1789
1790 def list(self, items, columns=4, width=80):
1791 items = items[:]
1792 items.sort()
1793 colw = width / columns
1794 rows = (len(items) + columns - 1) / columns
1795 for row in range(rows):
1796 for col in range(columns):
1797 i = col * rows + row
1798 if i < len(items):
1799 self.output.write(items[i])
1800 if col < columns - 1:
1801 self.output.write(' ' + ' ' * (colw-1 - len(items[i])))
1802 self.output.write('\n')
1803
1804 def listkeywords(self):
1805 self.output.write('''
1806Here is a list of the Python keywords. Enter any keyword to get more help.
1807
1808''')
1809 self.list(self.keywords.keys())
1810
Georg Brandl47fe9812009-01-01 15:46:10 +00001811 def listsymbols(self):
1812 self.output.write('''
1813Here is a list of the punctuation symbols which Python assigns special meaning
1814to. Enter any symbol to get more help.
1815
1816''')
1817 self.list(self.symbols.keys())
1818
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001819 def listtopics(self):
1820 self.output.write('''
1821Here is a list of available topics. Enter any topic name to get more help.
1822
1823''')
1824 self.list(self.topics.keys())
1825
Georg Brandl47fe9812009-01-01 15:46:10 +00001826 def showtopic(self, topic, more_xrefs=''):
Georg Brandl681001e2008-06-01 20:33:55 +00001827 try:
1828 import pydoc_topics
1829 except ImportError:
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001830 self.output.write('''
Georg Brandl681001e2008-06-01 20:33:55 +00001831Sorry, topic and keyword documentation is not available because the
1832module "pydoc_topics" could not be found.
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001833''')
1834 return
1835 target = self.topics.get(topic, self.keywords.get(topic))
1836 if not target:
1837 self.output.write('no documentation found for %s\n' % repr(topic))
1838 return
1839 if type(target) is type(''):
Georg Brandl47fe9812009-01-01 15:46:10 +00001840 return self.showtopic(target, more_xrefs)
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001841
Georg Brandl681001e2008-06-01 20:33:55 +00001842 label, xrefs = target
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001843 try:
Georg Brandl681001e2008-06-01 20:33:55 +00001844 doc = pydoc_topics.topics[label]
1845 except KeyError:
1846 self.output.write('no documentation found for %s\n' % repr(topic))
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001847 return
Georg Brandl681001e2008-06-01 20:33:55 +00001848 pager(strip(doc) + '\n')
Georg Brandl47fe9812009-01-01 15:46:10 +00001849 if more_xrefs:
1850 xrefs = (xrefs or '') + ' ' + more_xrefs
Ka-Ping Yeeda793892001-04-13 11:02:51 +00001851 if xrefs:
Georg Brandl681001e2008-06-01 20:33:55 +00001852 import StringIO, formatter
Ka-Ping Yeeda793892001-04-13 11:02:51 +00001853 buffer = StringIO.StringIO()
1854 formatter.DumbWriter(buffer).send_flowing_data(
1855 'Related help topics: ' + join(split(xrefs), ', ') + '\n')
1856 self.output.write('\n%s\n' % buffer.getvalue())
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001857
Georg Brandl47fe9812009-01-01 15:46:10 +00001858 def showsymbol(self, symbol):
1859 target = self.symbols[symbol]
1860 topic, _, xrefs = target.partition(' ')
1861 self.showtopic(topic, xrefs)
1862
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001863 def listmodules(self, key=''):
1864 if key:
1865 self.output.write('''
1866Here is a list of matching modules. Enter any module name to get more help.
1867
1868''')
1869 apropos(key)
1870 else:
1871 self.output.write('''
1872Please wait a moment while I gather a list of all available modules...
1873
1874''')
1875 modules = {}
1876 def callback(path, modname, desc, modules=modules):
1877 if modname and modname[-9:] == '.__init__':
1878 modname = modname[:-9] + ' (package)'
1879 if find(modname, '.') < 0:
1880 modules[modname] = 1
Ka-Ping Yee9e0f1162008-01-13 11:25:13 +00001881 def onerror(modname):
1882 callback(None, modname, None)
1883 ModuleScanner().run(callback, onerror=onerror)
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001884 self.list(modules.keys())
1885 self.output.write('''
1886Enter any module name to get more help. Or, type "modules spam" to search
1887for modules whose descriptions contain the word "spam".
1888''')
1889
Georg Brandl03c1cff2010-08-01 22:05:31 +00001890help = Helper()
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001891
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001892class Scanner:
1893 """A generic tree iterator."""
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001894 def __init__(self, roots, children, descendp):
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001895 self.roots = roots[:]
1896 self.state = []
1897 self.children = children
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001898 self.descendp = descendp
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001899
1900 def next(self):
1901 if not self.state:
1902 if not self.roots:
1903 return None
1904 root = self.roots.pop(0)
1905 self.state = [(root, self.children(root))]
1906 node, children = self.state[-1]
1907 if not children:
1908 self.state.pop()
1909 return self.next()
1910 child = children.pop(0)
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001911 if self.descendp(child):
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001912 self.state.append((child, self.children(child)))
1913 return child
1914
Phillip J. Ebyceb30872006-04-18 00:59:55 +00001915
1916class ModuleScanner:
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001917 """An interruptible scanner that searches module synopses."""
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001918
Ka-Ping Yee9e0f1162008-01-13 11:25:13 +00001919 def run(self, callback, key=None, completer=None, onerror=None):
Ka-Ping Yee66246962001-04-12 11:59:50 +00001920 if key: key = lower(key)
Guido van Rossum8ca162f2002-04-07 06:36:23 +00001921 self.quit = False
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001922 seen = {}
1923
1924 for modname in sys.builtin_module_names:
Ka-Ping Yee239432a2001-03-02 02:45:08 +00001925 if modname != '__main__':
1926 seen[modname] = 1
Ka-Ping Yee66246962001-04-12 11:59:50 +00001927 if key is None:
1928 callback(None, modname, '')
1929 else:
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001930 desc = split(__import__(modname).__doc__ or '', '\n')[0]
Ka-Ping Yee66246962001-04-12 11:59:50 +00001931 if find(lower(modname + ' - ' + desc), key) >= 0:
1932 callback(None, modname, desc)
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001933
Ka-Ping Yee9e0f1162008-01-13 11:25:13 +00001934 for importer, modname, ispkg in pkgutil.walk_packages(onerror=onerror):
Phillip J. Ebyceb30872006-04-18 00:59:55 +00001935 if self.quit:
1936 break
1937 if key is None:
1938 callback(None, modname, '')
1939 else:
1940 loader = importer.find_module(modname)
1941 if hasattr(loader,'get_source'):
1942 import StringIO
1943 desc = source_synopsis(
1944 StringIO.StringIO(loader.get_source(modname))
1945 ) or ''
1946 if hasattr(loader,'get_filename'):
1947 path = loader.get_filename(modname)
Ka-Ping Yee66246962001-04-12 11:59:50 +00001948 else:
Phillip J. Ebyceb30872006-04-18 00:59:55 +00001949 path = None
1950 else:
1951 module = loader.load_module(modname)
1952 desc = (module.__doc__ or '').splitlines()[0]
1953 path = getattr(module,'__file__',None)
1954 if find(lower(modname + ' - ' + desc), key) >= 0:
1955 callback(path, modname, desc)
1956
1957 if completer:
1958 completer()
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001959
1960def apropos(key):
1961 """Print all the one-line module summaries that contain a substring."""
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001962 def callback(path, modname, desc):
1963 if modname[-9:] == '.__init__':
1964 modname = modname[:-9] + ' (package)'
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +00001965 print modname, desc and '- ' + desc
1966 try: import warnings
1967 except ImportError: pass
1968 else: warnings.filterwarnings('ignore') # ignore problems during import
Ka-Ping Yee66246962001-04-12 11:59:50 +00001969 ModuleScanner().run(callback, key)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001970
1971# --------------------------------------------------- web browser interface
1972
Ka-Ping Yee66246962001-04-12 11:59:50 +00001973def serve(port, callback=None, completer=None):
Ka-Ping Yeefd540692001-04-12 12:54:36 +00001974 import BaseHTTPServer, mimetools, select
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001975
1976 # Patch up mimetools.Message so it doesn't break if rfc822 is reloaded.
1977 class Message(mimetools.Message):
1978 def __init__(self, fp, seekable=1):
1979 Message = self.__class__
1980 Message.__bases__[0].__bases__[0].__init__(self, fp, seekable)
1981 self.encodingheader = self.getheader('content-transfer-encoding')
1982 self.typeheader = self.getheader('content-type')
1983 self.parsetype()
1984 self.parseplist()
1985
1986 class DocHandler(BaseHTTPServer.BaseHTTPRequestHandler):
1987 def send_document(self, title, contents):
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001988 try:
1989 self.send_response(200)
1990 self.send_header('Content-Type', 'text/html')
1991 self.end_headers()
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001992 self.wfile.write(html.page(title, contents))
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001993 except IOError: pass
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001994
1995 def do_GET(self):
1996 path = self.path
1997 if path[-5:] == '.html': path = path[:-5]
1998 if path[:1] == '/': path = path[1:]
1999 if path and path != '.':
2000 try:
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00002001 obj = locate(path, forceload=1)
Ka-Ping Yee3bda8792001-03-23 13:17:50 +00002002 except ErrorDuringImport, value:
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00002003 self.send_document(path, html.escape(str(value)))
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002004 return
Ka-Ping Yee3bda8792001-03-23 13:17:50 +00002005 if obj:
2006 self.send_document(describe(obj), html.document(obj, path))
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002007 else:
2008 self.send_document(path,
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00002009'no Python documentation found for %s' % repr(path))
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002010 else:
2011 heading = html.heading(
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002012'<big><big><strong>Python: Index of Modules</strong></big></big>',
2013'#ffffff', '#7799ee')
Ka-Ping Yee239432a2001-03-02 02:45:08 +00002014 def bltinlink(name):
2015 return '<a href="%s.html">%s</a>' % (name, name)
Ka-Ping Yee3bda8792001-03-23 13:17:50 +00002016 names = filter(lambda x: x != '__main__',
2017 sys.builtin_module_names)
Ka-Ping Yee239432a2001-03-02 02:45:08 +00002018 contents = html.multicolumn(names, bltinlink)
2019 indices = ['<p>' + html.bigsection(
2020 'Built-in Modules', '#ffffff', '#ee77aa', contents)]
2021
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002022 seen = {}
Phillip J. Ebyceb30872006-04-18 00:59:55 +00002023 for dir in sys.path:
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002024 indices.append(html.index(dir, seen))
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00002025 contents = heading + join(indices) + '''<p align=right>
Tim Peters2306d242001-09-25 03:18:32 +00002026<font color="#909090" face="helvetica, arial"><strong>
2027pydoc</strong> by Ka-Ping Yee &lt;ping@lfw.org&gt;</font>'''
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002028 self.send_document('Index of Modules', contents)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002029
2030 def log_message(self, *args): pass
2031
Ka-Ping Yeefd540692001-04-12 12:54:36 +00002032 class DocServer(BaseHTTPServer.HTTPServer):
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002033 def __init__(self, port, callback):
Ka-Ping Yeedb8ed152001-03-02 05:58:17 +00002034 host = (sys.platform == 'mac') and '127.0.0.1' or 'localhost'
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00002035 self.address = ('', port)
Ka-Ping Yeedb8ed152001-03-02 05:58:17 +00002036 self.url = 'http://%s:%d/' % (host, port)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002037 self.callback = callback
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002038 self.base.__init__(self, self.address, self.handler)
2039
2040 def serve_until_quit(self):
2041 import select
Guido van Rossum8ca162f2002-04-07 06:36:23 +00002042 self.quit = False
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002043 while not self.quit:
2044 rd, wr, ex = select.select([self.socket.fileno()], [], [], 1)
2045 if rd: self.handle_request()
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002046
2047 def server_activate(self):
2048 self.base.server_activate(self)
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002049 if self.callback: self.callback(self)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002050
2051 DocServer.base = BaseHTTPServer.HTTPServer
2052 DocServer.handler = DocHandler
2053 DocHandler.MessageClass = Message
2054 try:
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00002055 try:
2056 DocServer(port, callback).serve_until_quit()
2057 except (KeyboardInterrupt, select.error):
2058 pass
2059 finally:
Ka-Ping Yee66246962001-04-12 11:59:50 +00002060 if completer: completer()
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002061
2062# ----------------------------------------------------- graphical interface
2063
2064def gui():
2065 """Graphical interface (starts web server and pops up a control window)."""
2066 class GUI:
2067 def __init__(self, window, port=7464):
2068 self.window = window
2069 self.server = None
2070 self.scanner = None
2071
Georg Brandl6634bf22008-05-20 07:13:37 +00002072 import Tkinter
2073 self.server_frm = Tkinter.Frame(window)
2074 self.title_lbl = Tkinter.Label(self.server_frm,
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002075 text='Starting server...\n ')
Georg Brandl6634bf22008-05-20 07:13:37 +00002076 self.open_btn = Tkinter.Button(self.server_frm,
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002077 text='open browser', command=self.open, state='disabled')
Georg Brandl6634bf22008-05-20 07:13:37 +00002078 self.quit_btn = Tkinter.Button(self.server_frm,
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002079 text='quit serving', command=self.quit, state='disabled')
2080
Georg Brandl6634bf22008-05-20 07:13:37 +00002081 self.search_frm = Tkinter.Frame(window)
2082 self.search_lbl = Tkinter.Label(self.search_frm, text='Search for')
2083 self.search_ent = Tkinter.Entry(self.search_frm)
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002084 self.search_ent.bind('<Return>', self.search)
Georg Brandl6634bf22008-05-20 07:13:37 +00002085 self.stop_btn = Tkinter.Button(self.search_frm,
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002086 text='stop', pady=0, command=self.stop, state='disabled')
2087 if sys.platform == 'win32':
Ka-Ping Yeec92cdf72001-03-02 05:54:35 +00002088 # Trying to hide and show this button crashes under Windows.
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002089 self.stop_btn.pack(side='right')
2090
2091 self.window.title('pydoc')
2092 self.window.protocol('WM_DELETE_WINDOW', self.quit)
2093 self.title_lbl.pack(side='top', fill='x')
2094 self.open_btn.pack(side='left', fill='x', expand=1)
2095 self.quit_btn.pack(side='right', fill='x', expand=1)
2096 self.server_frm.pack(side='top', fill='x')
2097
2098 self.search_lbl.pack(side='left')
2099 self.search_ent.pack(side='right', fill='x', expand=1)
2100 self.search_frm.pack(side='top', fill='x')
2101 self.search_ent.focus_set()
2102
Ka-Ping Yeec92cdf72001-03-02 05:54:35 +00002103 font = ('helvetica', sys.platform == 'win32' and 8 or 10)
Georg Brandl6634bf22008-05-20 07:13:37 +00002104 self.result_lst = Tkinter.Listbox(window, font=font, height=6)
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002105 self.result_lst.bind('<Button-1>', self.select)
2106 self.result_lst.bind('<Double-Button-1>', self.goto)
Georg Brandl6634bf22008-05-20 07:13:37 +00002107 self.result_scr = Tkinter.Scrollbar(window,
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002108 orient='vertical', command=self.result_lst.yview)
2109 self.result_lst.config(yscrollcommand=self.result_scr.set)
2110
Georg Brandl6634bf22008-05-20 07:13:37 +00002111 self.result_frm = Tkinter.Frame(window)
2112 self.goto_btn = Tkinter.Button(self.result_frm,
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002113 text='go to selected', command=self.goto)
Georg Brandl6634bf22008-05-20 07:13:37 +00002114 self.hide_btn = Tkinter.Button(self.result_frm,
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002115 text='hide results', command=self.hide)
2116 self.goto_btn.pack(side='left', fill='x', expand=1)
2117 self.hide_btn.pack(side='right', fill='x', expand=1)
2118
2119 self.window.update()
2120 self.minwidth = self.window.winfo_width()
2121 self.minheight = self.window.winfo_height()
2122 self.bigminheight = (self.server_frm.winfo_reqheight() +
2123 self.search_frm.winfo_reqheight() +
2124 self.result_lst.winfo_reqheight() +
2125 self.result_frm.winfo_reqheight())
2126 self.bigwidth, self.bigheight = self.minwidth, self.bigminheight
2127 self.expanded = 0
2128 self.window.wm_geometry('%dx%d' % (self.minwidth, self.minheight))
2129 self.window.wm_minsize(self.minwidth, self.minheight)
Martin v. Löwis5b26abb2002-12-28 09:23:09 +00002130 self.window.tk.willdispatch()
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002131
2132 import threading
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00002133 threading.Thread(
2134 target=serve, args=(port, self.ready, self.quit)).start()
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002135
2136 def ready(self, server):
2137 self.server = server
2138 self.title_lbl.config(
2139 text='Python documentation server at\n' + server.url)
2140 self.open_btn.config(state='normal')
2141 self.quit_btn.config(state='normal')
2142
Ka-Ping Yee239432a2001-03-02 02:45:08 +00002143 def open(self, event=None, url=None):
2144 url = url or self.server.url
2145 try:
2146 import webbrowser
2147 webbrowser.open(url)
2148 except ImportError: # pre-webbrowser.py compatibility
Ka-Ping Yeec92cdf72001-03-02 05:54:35 +00002149 if sys.platform == 'win32':
Ka-Ping Yee239432a2001-03-02 02:45:08 +00002150 os.system('start "%s"' % url)
2151 elif sys.platform == 'mac':
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +00002152 try: import ic
Ka-Ping Yee239432a2001-03-02 02:45:08 +00002153 except ImportError: pass
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +00002154 else: ic.launchurl(url)
Ka-Ping Yee239432a2001-03-02 02:45:08 +00002155 else:
2156 rc = os.system('netscape -remote "openURL(%s)" &' % url)
2157 if rc: os.system('netscape "%s" &' % url)
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002158
2159 def quit(self, event=None):
2160 if self.server:
2161 self.server.quit = 1
2162 self.window.quit()
2163
2164 def search(self, event=None):
2165 key = self.search_ent.get()
2166 self.stop_btn.pack(side='right')
2167 self.stop_btn.config(state='normal')
2168 self.search_lbl.config(text='Searching for "%s"...' % key)
2169 self.search_ent.forget()
2170 self.search_lbl.pack(side='left')
2171 self.result_lst.delete(0, 'end')
2172 self.goto_btn.config(state='disabled')
2173 self.expand()
2174
2175 import threading
2176 if self.scanner:
2177 self.scanner.quit = 1
2178 self.scanner = ModuleScanner()
2179 threading.Thread(target=self.scanner.run,
Ka-Ping Yee6dcfa382001-04-12 20:27:31 +00002180 args=(self.update, key, self.done)).start()
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002181
2182 def update(self, path, modname, desc):
2183 if modname[-9:] == '.__init__':
2184 modname = modname[:-9] + ' (package)'
2185 self.result_lst.insert('end',
2186 modname + ' - ' + (desc or '(no description)'))
2187
2188 def stop(self, event=None):
2189 if self.scanner:
2190 self.scanner.quit = 1
2191 self.scanner = None
2192
2193 def done(self):
2194 self.scanner = None
2195 self.search_lbl.config(text='Search for')
2196 self.search_lbl.pack(side='left')
2197 self.search_ent.pack(side='right', fill='x', expand=1)
2198 if sys.platform != 'win32': self.stop_btn.forget()
2199 self.stop_btn.config(state='disabled')
2200
2201 def select(self, event=None):
2202 self.goto_btn.config(state='normal')
2203
2204 def goto(self, event=None):
2205 selection = self.result_lst.curselection()
2206 if selection:
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002207 modname = split(self.result_lst.get(selection[0]))[0]
Ka-Ping Yee239432a2001-03-02 02:45:08 +00002208 self.open(url=self.server.url + modname + '.html')
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002209
2210 def collapse(self):
2211 if not self.expanded: return
2212 self.result_frm.forget()
2213 self.result_scr.forget()
2214 self.result_lst.forget()
2215 self.bigwidth = self.window.winfo_width()
2216 self.bigheight = self.window.winfo_height()
2217 self.window.wm_geometry('%dx%d' % (self.minwidth, self.minheight))
2218 self.window.wm_minsize(self.minwidth, self.minheight)
2219 self.expanded = 0
2220
2221 def expand(self):
2222 if self.expanded: return
2223 self.result_frm.pack(side='bottom', fill='x')
2224 self.result_scr.pack(side='right', fill='y')
2225 self.result_lst.pack(side='top', fill='both', expand=1)
2226 self.window.wm_geometry('%dx%d' % (self.bigwidth, self.bigheight))
2227 self.window.wm_minsize(self.minwidth, self.bigminheight)
2228 self.expanded = 1
2229
2230 def hide(self, event=None):
2231 self.stop()
2232 self.collapse()
2233
Georg Brandl6634bf22008-05-20 07:13:37 +00002234 import Tkinter
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002235 try:
Georg Brandl6634bf22008-05-20 07:13:37 +00002236 root = Tkinter.Tk()
Martin v. Löwise09bd932004-08-22 16:13:26 +00002237 # Tk will crash if pythonw.exe has an XP .manifest
2238 # file and the root has is not destroyed explicitly.
2239 # If the problem is ever fixed in Tk, the explicit
2240 # destroy can go.
2241 try:
2242 gui = GUI(root)
2243 root.mainloop()
2244 finally:
2245 root.destroy()
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002246 except KeyboardInterrupt:
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002247 pass
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002248
2249# -------------------------------------------------- command-line interface
2250
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00002251def ispath(x):
Ka-Ping Yee45daeb02002-08-11 15:11:33 +00002252 return isinstance(x, str) and find(x, os.sep) >= 0
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00002253
Ka-Ping Yee1d384632001-03-01 00:24:32 +00002254def cli():
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002255 """Command-line interface (looks at sys.argv to decide what to do)."""
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002256 import getopt
2257 class BadUsage: pass
2258
Nick Coghlan29d5bb12009-11-15 22:58:27 +00002259 # Scripts don't get the current directory in their path by default
2260 # unless they are run with the '-m' switch
2261 if '' not in sys.path:
2262 scriptdir = os.path.dirname(sys.argv[0])
2263 if scriptdir in sys.path:
2264 sys.path.remove(scriptdir)
2265 sys.path.insert(0, '.')
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002266
Ka-Ping Yee3bda8792001-03-23 13:17:50 +00002267 try:
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002268 opts, args = getopt.getopt(sys.argv[1:], 'gk:p:w')
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002269 writing = 0
2270
2271 for opt, val in opts:
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002272 if opt == '-g':
2273 gui()
2274 return
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002275 if opt == '-k':
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002276 apropos(val)
2277 return
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002278 if opt == '-p':
2279 try:
2280 port = int(val)
2281 except ValueError:
2282 raise BadUsage
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002283 def ready(server):
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00002284 print 'pydoc server ready at %s' % server.url
2285 def stopped():
2286 print 'pydoc server stopped'
2287 serve(port, ready, stopped)
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002288 return
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002289 if opt == '-w':
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002290 writing = 1
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002291
2292 if not args: raise BadUsage
2293 for arg in args:
Ka-Ping Yee45daeb02002-08-11 15:11:33 +00002294 if ispath(arg) and not os.path.exists(arg):
2295 print 'file %r does not exist' % arg
2296 break
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002297 try:
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00002298 if ispath(arg) and os.path.isfile(arg):
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002299 arg = importfile(arg)
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00002300 if writing:
2301 if ispath(arg) and os.path.isdir(arg):
2302 writedocs(arg)
2303 else:
2304 writedoc(arg)
2305 else:
Martin v. Löwisb8c084e2003-06-14 09:03:46 +00002306 help.help(arg)
Ka-Ping Yee3bda8792001-03-23 13:17:50 +00002307 except ErrorDuringImport, value:
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00002308 print value
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002309
2310 except (getopt.error, BadUsage):
Martin v. Löwisf9b08b82003-10-31 13:05:21 +00002311 cmd = os.path.basename(sys.argv[0])
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002312 print """pydoc - the Python documentation tool
2313
2314%s <name> ...
2315 Show text documentation on something. <name> may be the name of a
Martin v. Löwisb8c084e2003-06-14 09:03:46 +00002316 Python keyword, topic, function, module, or package, or a dotted
2317 reference to a class or function within a module or module in a
2318 package. If <name> contains a '%s', it is used as the path to a
2319 Python source file to document. If name is 'keywords', 'topics',
2320 or 'modules', a listing of these things is displayed.
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002321
2322%s -k <keyword>
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002323 Search for a keyword in the synopsis lines of all available modules.
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002324
2325%s -p <port>
2326 Start an HTTP server on the given port on the local machine.
2327
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002328%s -g
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +00002329 Pop up a graphical interface for finding and serving documentation.
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002330
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002331%s -w <name> ...
2332 Write out the HTML documentation for a module to a file in the current
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +00002333 directory. If <name> contains a '%s', it is treated as a filename; if
2334 it names a directory, documentation is written for all the contents.
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002335""" % (cmd, os.sep, cmd, cmd, cmd, cmd, os.sep)
Ka-Ping Yee1d384632001-03-01 00:24:32 +00002336
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002337if __name__ == '__main__': cli()