blob: 7a55023319ec447435bba3cc536d8418eea40ea5 [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 Murrayef087da2009-06-23 18:02:46 +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 Melottie511fc72010-02-16 23:26:09 +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 Murrayef087da2009-06-23 18:02:46 +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
359 not file.startswith(os.path.join(basedir, 'site-packages'))))):
360 if docloc.startswith("http://"):
Georg Brandl0751d1a2008-01-21 17:13:03 +0000361 docloc = "%s/%s" % (docloc.rstrip("/"), object.__name__)
Skip Montanaro4997a692003-09-10 16:47:51 +0000362 else:
Georg Brandl0751d1a2008-01-21 17:13:03 +0000363 docloc = os.path.join(docloc, object.__name__ + ".html")
Skip Montanaro4997a692003-09-10 16:47:51 +0000364 else:
365 docloc = None
366 return docloc
367
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000368# -------------------------------------------- HTML documentation generator
369
370class HTMLRepr(Repr):
371 """Class for safely making an HTML representation of a Python object."""
372 def __init__(self):
373 Repr.__init__(self)
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000374 self.maxlist = self.maxtuple = 20
375 self.maxdict = 10
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000376 self.maxstring = self.maxother = 100
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000377
378 def escape(self, text):
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +0000379 return replace(text, '&', '&amp;', '<', '&lt;', '>', '&gt;')
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000380
381 def repr(self, object):
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000382 return Repr.repr(self, object)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000383
384 def repr1(self, x, level):
Skip Montanaro0fe8fce2003-06-27 15:45:41 +0000385 if hasattr(type(x), '__name__'):
386 methodname = 'repr_' + join(split(type(x).__name__), '_')
387 if hasattr(self, methodname):
388 return getattr(self, methodname)(x, level)
389 return self.escape(cram(stripid(repr(x)), self.maxother))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000390
391 def repr_string(self, x, level):
Ka-Ping Yeea2fe1032001-03-02 01:19:14 +0000392 test = cram(x, self.maxstring)
393 testrepr = repr(test)
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +0000394 if '\\' in test and '\\' not in replace(testrepr, r'\\', ''):
Ka-Ping Yeea2fe1032001-03-02 01:19:14 +0000395 # Backslashes are only literal in the string and are never
396 # needed to make any special characters, so show a raw string.
397 return 'r' + testrepr[0] + self.escape(test) + testrepr[0]
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +0000398 return re.sub(r'((\\[\\abfnrtv\'"]|\\[0-9]..|\\x..|\\u....)+)',
Ka-Ping Yeea2fe1032001-03-02 01:19:14 +0000399 r'<font color="#c040c0">\1</font>',
400 self.escape(testrepr))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000401
Skip Montanarodf708782002-03-07 22:58:02 +0000402 repr_str = repr_string
403
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000404 def repr_instance(self, x, level):
405 try:
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000406 return self.escape(cram(stripid(repr(x)), self.maxstring))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000407 except:
408 return self.escape('<%s instance>' % x.__class__.__name__)
409
410 repr_unicode = repr_string
411
412class HTMLDoc(Doc):
413 """Formatter class for HTML documentation."""
414
415 # ------------------------------------------- HTML formatting utilities
416
417 _repr_instance = HTMLRepr()
418 repr = _repr_instance.repr
419 escape = _repr_instance.escape
420
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000421 def page(self, title, contents):
422 """Format an HTML page."""
423 return '''
Georg Brandle0197062009-04-10 08:20:23 +0000424<!DOCTYPE html PUBLIC "-//W3C//DTD HTML 4.0 Transitional//EN">
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000425<html><head><title>Python: %s</title>
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000426</head><body bgcolor="#f0f0f8">
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000427%s
428</body></html>''' % (title, contents)
429
430 def heading(self, title, fgcol, bgcol, extras=''):
431 """Format a page heading."""
432 return '''
Tim Peters59ed4482001-10-31 04:20:26 +0000433<table width="100%%" cellspacing=0 cellpadding=2 border=0 summary="heading">
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000434<tr bgcolor="%s">
Tim Peters2306d242001-09-25 03:18:32 +0000435<td valign=bottom>&nbsp;<br>
436<font color="%s" face="helvetica, arial">&nbsp;<br>%s</font></td
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000437><td align=right valign=bottom
Ka-Ping Yee987ec902001-03-23 13:35:45 +0000438><font color="%s" face="helvetica, arial">%s</font></td></tr></table>
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000439 ''' % (bgcol, fgcol, title, fgcol, extras or '&nbsp;')
440
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000441 def section(self, title, fgcol, bgcol, contents, width=6,
442 prelude='', marginalia=None, gap='&nbsp;'):
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000443 """Format a section with a heading."""
444 if marginalia is None:
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +0000445 marginalia = '<tt>' + '&nbsp;' * width + '</tt>'
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000446 result = '''<p>
Tim Peters59ed4482001-10-31 04:20:26 +0000447<table width="100%%" cellspacing=0 cellpadding=2 border=0 summary="section">
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000448<tr bgcolor="%s">
Tim Peters2306d242001-09-25 03:18:32 +0000449<td colspan=3 valign=bottom>&nbsp;<br>
450<font color="%s" face="helvetica, arial">%s</font></td></tr>
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000451 ''' % (bgcol, fgcol, title)
452 if prelude:
453 result = result + '''
Ka-Ping Yee987ec902001-03-23 13:35:45 +0000454<tr bgcolor="%s"><td rowspan=2>%s</td>
455<td colspan=2>%s</td></tr>
456<tr><td>%s</td>''' % (bgcol, marginalia, prelude, gap)
457 else:
458 result = result + '''
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000459<tr><td bgcolor="%s">%s</td><td>%s</td>''' % (bgcol, marginalia, gap)
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000460
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +0000461 return result + '\n<td width="100%%">%s</td></tr></table>' % contents
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000462
463 def bigsection(self, title, *args):
464 """Format a section with a big heading."""
465 title = '<big><strong>%s</strong></big>' % title
Guido van Rossum68468eb2003-02-27 20:14:51 +0000466 return self.section(title, *args)
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000467
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000468 def preformat(self, text):
469 """Format literal preformatted text."""
470 text = self.escape(expandtabs(text))
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +0000471 return replace(text, '\n\n', '\n \n', '\n\n', '\n \n',
472 ' ', '&nbsp;', '\n', '<br>\n')
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000473
474 def multicolumn(self, list, format, cols=4):
475 """Format a list of items into a multi-column list."""
476 result = ''
477 rows = (len(list)+cols-1)/cols
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000478 for col in range(cols):
479 result = result + '<td width="%d%%" valign=top>' % (100/cols)
480 for i in range(rows*col, rows*col+rows):
481 if i < len(list):
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000482 result = result + format(list[i]) + '<br>\n'
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000483 result = result + '</td>'
Tim Peters59ed4482001-10-31 04:20:26 +0000484 return '<table width="100%%" summary="list"><tr>%s</tr></table>' % result
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000485
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000486 def grey(self, text): return '<font color="#909090">%s</font>' % text
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000487
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000488 def namelink(self, name, *dicts):
489 """Make a link for an identifier, given name-to-URL mappings."""
490 for dict in dicts:
Raymond Hettinger54f02222002-06-01 14:18:47 +0000491 if name in dict:
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000492 return '<a href="%s">%s</a>' % (dict[name], name)
493 return name
494
Ka-Ping Yeeb7a48302001-04-12 20:39:14 +0000495 def classlink(self, object, modname):
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000496 """Make a link for a class."""
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000497 name, module = object.__name__, sys.modules.get(object.__module__)
498 if hasattr(module, name) and getattr(module, name) is object:
Ka-Ping Yeeb7a48302001-04-12 20:39:14 +0000499 return '<a href="%s.html#%s">%s</a>' % (
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000500 module.__name__, name, classname(object, modname))
501 return classname(object, modname)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000502
503 def modulelink(self, object):
504 """Make a link for a module."""
505 return '<a href="%s.html">%s</a>' % (object.__name__, object.__name__)
506
Brett Cannonaca98b02008-08-03 00:58:51 +0000507 def modpkglink(self, data):
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000508 """Make a link for a module or package to display in an index."""
Brett Cannonaca98b02008-08-03 00:58:51 +0000509 name, path, ispackage, shadowed = data
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000510 if shadowed:
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000511 return self.grey(name)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000512 if path:
513 url = '%s.%s.html' % (path, name)
514 else:
515 url = '%s.html' % name
516 if ispackage:
517 text = '<strong>%s</strong>&nbsp;(package)' % name
518 else:
519 text = name
520 return '<a href="%s">%s</a>' % (url, text)
521
522 def markup(self, text, escape=None, funcs={}, classes={}, methods={}):
523 """Mark up some plain text, given a context of symbols to look for.
524 Each context dictionary maps object names to anchor names."""
525 escape = escape or self.escape
526 results = []
527 here = 0
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000528 pattern = re.compile(r'\b((http|ftp)://\S+[\w/]|'
529 r'RFC[- ]?(\d+)|'
Ka-Ping Yeef78a81b2001-03-27 08:13:42 +0000530 r'PEP[- ]?(\d+)|'
Neil Schemenauerd69711c2002-03-24 23:02:07 +0000531 r'(self\.)?(\w+))')
Guido van Rossum8ca162f2002-04-07 06:36:23 +0000532 while True:
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000533 match = pattern.search(text, here)
534 if not match: break
535 start, end = match.span()
536 results.append(escape(text[here:start]))
537
Ka-Ping Yeef78a81b2001-03-27 08:13:42 +0000538 all, scheme, rfc, pep, selfdot, name = match.groups()
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000539 if scheme:
Neil Schemenauercddc1a02002-03-24 23:11:21 +0000540 url = escape(all).replace('"', '&quot;')
541 results.append('<a href="%s">%s</a>' % (url, url))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000542 elif rfc:
Ka-Ping Yeef78a81b2001-03-27 08:13:42 +0000543 url = 'http://www.rfc-editor.org/rfc/rfc%d.txt' % int(rfc)
544 results.append('<a href="%s">%s</a>' % (url, escape(all)))
545 elif pep:
Andrew M. Kuchling727a5902008-02-05 16:06:57 +0000546 url = 'http://www.python.org/dev/peps/pep-%04d/' % int(pep)
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000547 results.append('<a href="%s">%s</a>' % (url, escape(all)))
548 elif text[end:end+1] == '(':
549 results.append(self.namelink(name, methods, funcs, classes))
550 elif selfdot:
551 results.append('self.<strong>%s</strong>' % name)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000552 else:
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000553 results.append(self.namelink(name, classes))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000554 here = end
555 results.append(escape(text[here:]))
556 return join(results, '')
557
558 # ---------------------------------------------- type-specific routines
559
Ka-Ping Yeeb7a48302001-04-12 20:39:14 +0000560 def formattree(self, tree, modname, parent=None):
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000561 """Produce HTML for a class tree as given by inspect.getclasstree()."""
562 result = ''
563 for entry in tree:
564 if type(entry) is type(()):
565 c, bases = entry
Tim Peters2306d242001-09-25 03:18:32 +0000566 result = result + '<dt><font face="helvetica, arial">'
Ka-Ping Yeeb7a48302001-04-12 20:39:14 +0000567 result = result + self.classlink(c, modname)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000568 if bases and bases != (parent,):
569 parents = []
570 for base in bases:
Ka-Ping Yeeb7a48302001-04-12 20:39:14 +0000571 parents.append(self.classlink(base, modname))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000572 result = result + '(' + join(parents, ', ') + ')'
Tim Peters2306d242001-09-25 03:18:32 +0000573 result = result + '\n</font></dt>'
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000574 elif type(entry) is type([]):
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000575 result = result + '<dd>\n%s</dd>\n' % self.formattree(
Ka-Ping Yeeb7a48302001-04-12 20:39:14 +0000576 entry, modname, c)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000577 return '<dl>\n%s</dl>\n' % result
578
Tim Peters8dd7ade2001-10-18 19:56:17 +0000579 def docmodule(self, object, name=None, mod=None, *ignored):
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000580 """Produce HTML documentation for a module object."""
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000581 name = object.__name__ # ignore the passed-in name
Skip Montanaroa5616d22004-06-11 04:46:12 +0000582 try:
583 all = object.__all__
584 except AttributeError:
585 all = None
Ka-Ping Yee66efbc72001-03-01 13:55:20 +0000586 parts = split(name, '.')
587 links = []
588 for i in range(len(parts)-1):
589 links.append(
590 '<a href="%s.html"><font color="#ffffff">%s</font></a>' %
591 (join(parts[:i+1], '.'), parts[i]))
592 linkedname = join(links + parts[-1:], '.')
593 head = '<big><big><strong>%s</strong></big></big>' % linkedname
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000594 try:
Ka-Ping Yee239432a2001-03-02 02:45:08 +0000595 path = inspect.getabsfile(object)
Ka-Ping Yee6191a232001-04-13 15:00:27 +0000596 url = path
597 if sys.platform == 'win32':
598 import nturl2path
599 url = nturl2path.pathname2url(path)
600 filelink = '<a href="file:%s">%s</a>' % (url, path)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000601 except TypeError:
602 filelink = '(built-in)'
Ka-Ping Yee6f3f9a42001-02-27 22:42:36 +0000603 info = []
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000604 if hasattr(object, '__version__'):
Ka-Ping Yee6f3f9a42001-02-27 22:42:36 +0000605 version = str(object.__version__)
Ka-Ping Yee40c49912001-02-27 22:46:01 +0000606 if version[:11] == '$' + 'Revision: ' and version[-1:] == '$':
607 version = strip(version[11:-1])
Ka-Ping Yee1d384632001-03-01 00:24:32 +0000608 info.append('version %s' % self.escape(version))
Ka-Ping Yee6f3f9a42001-02-27 22:42:36 +0000609 if hasattr(object, '__date__'):
610 info.append(self.escape(str(object.__date__)))
611 if info:
612 head = head + ' (%s)' % join(info, ', ')
Skip Montanaro4997a692003-09-10 16:47:51 +0000613 docloc = self.getdocloc(object)
614 if docloc is not None:
615 docloc = '<br><a href="%(docloc)s">Module Docs</a>' % locals()
616 else:
617 docloc = ''
Ka-Ping Yee66efbc72001-03-01 13:55:20 +0000618 result = self.heading(
Skip Montanaro4997a692003-09-10 16:47:51 +0000619 head, '#ffffff', '#7799ee',
620 '<a href=".">index</a><br>' + filelink + docloc)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000621
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000622 modules = inspect.getmembers(object, inspect.ismodule)
623
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000624 classes, cdict = [], {}
625 for key, value in inspect.getmembers(object, inspect.isclass):
Johannes Gijsbers4c11f602004-08-30 14:13:04 +0000626 # if __all__ exists, believe it. Otherwise use old heuristic.
627 if (all is not None or
628 (inspect.getmodule(value) or object) is object):
Skip Montanaroa5616d22004-06-11 04:46:12 +0000629 if visiblename(key, all):
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000630 classes.append((key, value))
631 cdict[key] = cdict[value] = '#' + key
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000632 for key, value in classes:
633 for base in value.__bases__:
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000634 key, modname = base.__name__, base.__module__
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000635 module = sys.modules.get(modname)
636 if modname != name and module and hasattr(module, key):
637 if getattr(module, key) is base:
Raymond Hettinger54f02222002-06-01 14:18:47 +0000638 if not key in cdict:
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000639 cdict[key] = cdict[base] = modname + '.html#' + key
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000640 funcs, fdict = [], {}
641 for key, value in inspect.getmembers(object, inspect.isroutine):
Johannes Gijsbers4c11f602004-08-30 14:13:04 +0000642 # if __all__ exists, believe it. Otherwise use old heuristic.
643 if (all is not None or
644 inspect.isbuiltin(value) or inspect.getmodule(value) is object):
Skip Montanaroa5616d22004-06-11 04:46:12 +0000645 if visiblename(key, all):
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000646 funcs.append((key, value))
647 fdict[key] = '#-' + key
648 if inspect.isfunction(value): fdict[value] = fdict[key]
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000649 data = []
650 for key, value in inspect.getmembers(object, isdata):
Skip Montanaroa5616d22004-06-11 04:46:12 +0000651 if visiblename(key, all):
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000652 data.append((key, value))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000653
654 doc = self.markup(getdoc(object), self.preformat, fdict, cdict)
655 doc = doc and '<tt>%s</tt>' % doc
Tim Peters2306d242001-09-25 03:18:32 +0000656 result = result + '<p>%s</p>\n' % doc
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000657
658 if hasattr(object, '__path__'):
659 modpkgs = []
Phillip J. Ebyceb30872006-04-18 00:59:55 +0000660 for importer, modname, ispkg in pkgutil.iter_modules(object.__path__):
661 modpkgs.append((modname, name, ispkg, 0))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000662 modpkgs.sort()
663 contents = self.multicolumn(modpkgs, self.modpkglink)
664 result = result + self.bigsection(
665 'Package Contents', '#ffffff', '#aa55cc', contents)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000666 elif modules:
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000667 contents = self.multicolumn(
Brett Cannonaca98b02008-08-03 00:58:51 +0000668 modules, lambda key_value, s=self: s.modulelink(key_value[1]))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000669 result = result + self.bigsection(
Georg Brandlb169eaa2008-02-17 21:18:55 +0000670 'Modules', '#ffffff', '#aa55cc', contents)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000671
672 if classes:
Brett Cannonaca98b02008-08-03 00:58:51 +0000673 classlist = map(lambda key_value: key_value[1], classes)
Ka-Ping Yeeb7a48302001-04-12 20:39:14 +0000674 contents = [
675 self.formattree(inspect.getclasstree(classlist, 1), name)]
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000676 for key, value in classes:
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +0000677 contents.append(self.document(value, key, name, fdict, cdict))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000678 result = result + self.bigsection(
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000679 'Classes', '#ffffff', '#ee77aa', join(contents))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000680 if funcs:
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000681 contents = []
682 for key, value in funcs:
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +0000683 contents.append(self.document(value, key, name, fdict, cdict))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000684 result = result + self.bigsection(
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000685 'Functions', '#ffffff', '#eeaa77', join(contents))
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000686 if data:
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000687 contents = []
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000688 for key, value in data:
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000689 contents.append(self.document(value, key))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000690 result = result + self.bigsection(
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000691 'Data', '#ffffff', '#55aa55', join(contents, '<br>\n'))
Ka-Ping Yee6f3f9a42001-02-27 22:42:36 +0000692 if hasattr(object, '__author__'):
693 contents = self.markup(str(object.__author__), self.preformat)
694 result = result + self.bigsection(
695 'Author', '#ffffff', '#7799ee', contents)
Ka-Ping Yee6f3f9a42001-02-27 22:42:36 +0000696 if hasattr(object, '__credits__'):
697 contents = self.markup(str(object.__credits__), self.preformat)
698 result = result + self.bigsection(
699 'Credits', '#ffffff', '#7799ee', contents)
700
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000701 return result
702
Tim Peters8dd7ade2001-10-18 19:56:17 +0000703 def docclass(self, object, name=None, mod=None, funcs={}, classes={},
704 *ignored):
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000705 """Produce HTML documentation for a class object."""
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000706 realname = object.__name__
707 name = name or realname
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000708 bases = object.__bases__
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000709
Tim Petersb47879b2001-09-24 04:47:19 +0000710 contents = []
711 push = contents.append
712
Tim Petersfa26f7c2001-09-24 08:05:11 +0000713 # Cute little class to pump out a horizontal rule between sections.
714 class HorizontalRule:
715 def __init__(self):
716 self.needone = 0
717 def maybe(self):
718 if self.needone:
719 push('<hr>\n')
720 self.needone = 1
721 hr = HorizontalRule()
722
Tim Petersc86f6ca2001-09-26 21:31:51 +0000723 # List the mro, if non-trivial.
Raymond Hettinger756b3f32004-01-29 06:37:52 +0000724 mro = deque(inspect.getmro(object))
Tim Petersc86f6ca2001-09-26 21:31:51 +0000725 if len(mro) > 2:
726 hr.maybe()
727 push('<dl><dt>Method resolution order:</dt>\n')
728 for base in mro:
729 push('<dd>%s</dd>\n' % self.classlink(base,
730 object.__module__))
731 push('</dl>\n')
732
Tim Petersb47879b2001-09-24 04:47:19 +0000733 def spill(msg, attrs, predicate):
Tim Petersfa26f7c2001-09-24 08:05:11 +0000734 ok, attrs = _split_list(attrs, predicate)
Tim Petersb47879b2001-09-24 04:47:19 +0000735 if ok:
Tim Petersfa26f7c2001-09-24 08:05:11 +0000736 hr.maybe()
Tim Petersb47879b2001-09-24 04:47:19 +0000737 push(msg)
738 for name, kind, homecls, value in ok:
739 push(self.document(getattr(object, name), name, mod,
740 funcs, classes, mdict, object))
741 push('\n')
742 return attrs
743
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +0000744 def spilldescriptors(msg, attrs, predicate):
Tim Petersfa26f7c2001-09-24 08:05:11 +0000745 ok, attrs = _split_list(attrs, predicate)
Tim Petersb47879b2001-09-24 04:47:19 +0000746 if ok:
Tim Petersfa26f7c2001-09-24 08:05:11 +0000747 hr.maybe()
Tim Petersb47879b2001-09-24 04:47:19 +0000748 push(msg)
749 for name, kind, homecls, value in ok:
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +0000750 push(self._docdescriptor(name, value, mod))
Tim Petersb47879b2001-09-24 04:47:19 +0000751 return attrs
752
Tim Petersfa26f7c2001-09-24 08:05:11 +0000753 def spilldata(msg, attrs, predicate):
754 ok, attrs = _split_list(attrs, predicate)
Tim Petersb47879b2001-09-24 04:47:19 +0000755 if ok:
Tim Petersfa26f7c2001-09-24 08:05:11 +0000756 hr.maybe()
Tim Petersb47879b2001-09-24 04:47:19 +0000757 push(msg)
758 for name, kind, homecls, value in ok:
759 base = self.docother(getattr(object, name), name, mod)
Brett Cannonaca98b02008-08-03 00:58:51 +0000760 if (hasattr(value, '__call__') or
761 inspect.isdatadescriptor(value)):
Guido van Rossum5e355b22002-05-21 20:56:15 +0000762 doc = getattr(value, "__doc__", None)
763 else:
764 doc = None
Tim Petersb47879b2001-09-24 04:47:19 +0000765 if doc is None:
766 push('<dl><dt>%s</dl>\n' % base)
767 else:
768 doc = self.markup(getdoc(value), self.preformat,
769 funcs, classes, mdict)
Tim Peters2306d242001-09-25 03:18:32 +0000770 doc = '<dd><tt>%s</tt>' % doc
Tim Petersb47879b2001-09-24 04:47:19 +0000771 push('<dl><dt>%s%s</dl>\n' % (base, doc))
772 push('\n')
773 return attrs
774
Brett Cannonaca98b02008-08-03 00:58:51 +0000775 attrs = filter(lambda data: visiblename(data[0]),
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +0000776 classify_class_attrs(object))
Tim Petersb47879b2001-09-24 04:47:19 +0000777 mdict = {}
778 for key, kind, homecls, value in attrs:
779 mdict[key] = anchor = '#' + name + '-' + key
780 value = getattr(object, key)
781 try:
782 # The value may not be hashable (e.g., a data attr with
783 # a dict or list value).
784 mdict[value] = anchor
785 except TypeError:
786 pass
787
Tim Petersfa26f7c2001-09-24 08:05:11 +0000788 while attrs:
Tim Peters351e3622001-09-27 03:29:51 +0000789 if mro:
Raymond Hettinger756b3f32004-01-29 06:37:52 +0000790 thisclass = mro.popleft()
Tim Peters351e3622001-09-27 03:29:51 +0000791 else:
792 thisclass = attrs[0][2]
Tim Petersfa26f7c2001-09-24 08:05:11 +0000793 attrs, inherited = _split_list(attrs, lambda t: t[2] is thisclass)
794
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000795 if thisclass is __builtin__.object:
796 attrs = inherited
797 continue
798 elif thisclass is object:
799 tag = 'defined here'
Tim Petersb47879b2001-09-24 04:47:19 +0000800 else:
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000801 tag = 'inherited from %s' % self.classlink(thisclass,
802 object.__module__)
Tim Petersb47879b2001-09-24 04:47:19 +0000803 tag += ':<br>\n'
804
805 # Sort attrs by name.
Phillip J. Ebyceb30872006-04-18 00:59:55 +0000806 try:
807 attrs.sort(key=lambda t: t[0])
808 except TypeError:
809 attrs.sort(lambda t1, t2: cmp(t1[0], t2[0])) # 2.3 compat
Tim Petersb47879b2001-09-24 04:47:19 +0000810
811 # Pump out the attrs, segregated by kind.
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000812 attrs = spill('Methods %s' % tag, attrs,
Tim Petersb47879b2001-09-24 04:47:19 +0000813 lambda t: t[1] == 'method')
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000814 attrs = spill('Class methods %s' % tag, attrs,
Tim Petersb47879b2001-09-24 04:47:19 +0000815 lambda t: t[1] == 'class method')
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000816 attrs = spill('Static methods %s' % tag, attrs,
Tim Petersb47879b2001-09-24 04:47:19 +0000817 lambda t: t[1] == 'static method')
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +0000818 attrs = spilldescriptors('Data descriptors %s' % tag, attrs,
819 lambda t: t[1] == 'data descriptor')
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000820 attrs = spilldata('Data and other attributes %s' % tag, attrs,
Tim Petersfa26f7c2001-09-24 08:05:11 +0000821 lambda t: t[1] == 'data')
Tim Petersb47879b2001-09-24 04:47:19 +0000822 assert attrs == []
Tim Peters351e3622001-09-27 03:29:51 +0000823 attrs = inherited
Tim Petersb47879b2001-09-24 04:47:19 +0000824
825 contents = ''.join(contents)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000826
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000827 if name == realname:
828 title = '<a name="%s">class <strong>%s</strong></a>' % (
829 name, realname)
830 else:
831 title = '<strong>%s</strong> = <a name="%s">class %s</a>' % (
832 name, name, realname)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000833 if bases:
834 parents = []
835 for base in bases:
Ka-Ping Yeeb7a48302001-04-12 20:39:14 +0000836 parents.append(self.classlink(base, object.__module__))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000837 title = title + '(%s)' % join(parents, ', ')
Tim Peters2306d242001-09-25 03:18:32 +0000838 doc = self.markup(getdoc(object), self.preformat, funcs, classes, mdict)
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000839 doc = doc and '<tt>%s<br>&nbsp;</tt>' % doc
Tim Petersc86f6ca2001-09-26 21:31:51 +0000840
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +0000841 return self.section(title, '#000000', '#ffc8d8', contents, 3, doc)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000842
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000843 def formatvalue(self, object):
844 """Format an argument default value as text."""
Tim Peters2306d242001-09-25 03:18:32 +0000845 return self.grey('=' + self.repr(object))
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000846
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +0000847 def docroutine(self, object, name=None, mod=None,
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000848 funcs={}, classes={}, methods={}, cl=None):
Ka-Ping Yee66efbc72001-03-01 13:55:20 +0000849 """Produce HTML documentation for a function or method object."""
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000850 realname = object.__name__
851 name = name or realname
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000852 anchor = (cl and cl.__name__ or '') + '-' + name
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000853 note = ''
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000854 skipdocs = 0
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000855 if inspect.ismethod(object):
Ka-Ping Yee6dcfa382001-04-12 20:27:31 +0000856 imclass = object.im_class
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000857 if cl:
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +0000858 if imclass is not cl:
Ka-Ping Yeeb7a48302001-04-12 20:39:14 +0000859 note = ' from ' + self.classlink(imclass, mod)
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000860 else:
Georg Brandl7e2b6bb2007-03-13 22:16:30 +0000861 if object.im_self is not None:
Ka-Ping Yeeb7a48302001-04-12 20:39:14 +0000862 note = ' method of %s instance' % self.classlink(
863 object.im_self.__class__, mod)
864 else:
865 note = ' unbound %s method' % self.classlink(imclass,mod)
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000866 object = object.im_func
867
868 if name == realname:
869 title = '<a name="%s"><strong>%s</strong></a>' % (anchor, realname)
870 else:
Raymond Hettinger54f02222002-06-01 14:18:47 +0000871 if (cl and realname in cl.__dict__ and
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000872 cl.__dict__[realname] is object):
Ka-Ping Yeee280c062001-03-23 14:05:53 +0000873 reallink = '<a href="#%s">%s</a>' % (
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000874 cl.__name__ + '-' + realname, realname)
875 skipdocs = 1
876 else:
877 reallink = realname
878 title = '<a name="%s"><strong>%s</strong></a> = %s' % (
879 anchor, name, reallink)
Tim Peters4bcfa312001-09-20 06:08:24 +0000880 if inspect.isfunction(object):
Ka-Ping Yee66efbc72001-03-01 13:55:20 +0000881 args, varargs, varkw, defaults = inspect.getargspec(object)
882 argspec = inspect.formatargspec(
883 args, varargs, varkw, defaults, formatvalue=self.formatvalue)
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000884 if realname == '<lambda>':
Tim Peters59ed4482001-10-31 04:20:26 +0000885 title = '<strong>%s</strong> <em>lambda</em> ' % name
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000886 argspec = argspec[1:-1] # remove parentheses
Tim Peters4bcfa312001-09-20 06:08:24 +0000887 else:
888 argspec = '(...)'
Ka-Ping Yee66efbc72001-03-01 13:55:20 +0000889
Tim Peters2306d242001-09-25 03:18:32 +0000890 decl = title + argspec + (note and self.grey(
891 '<font face="helvetica, arial">%s</font>' % note))
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000892
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000893 if skipdocs:
Tim Peters2306d242001-09-25 03:18:32 +0000894 return '<dl><dt>%s</dt></dl>\n' % decl
Ka-Ping Yee3bda8792001-03-23 13:17:50 +0000895 else:
896 doc = self.markup(
897 getdoc(object), self.preformat, funcs, classes, methods)
Tim Peters2306d242001-09-25 03:18:32 +0000898 doc = doc and '<dd><tt>%s</tt></dd>' % doc
899 return '<dl><dt>%s</dt>%s</dl>\n' % (decl, doc)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000900
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +0000901 def _docdescriptor(self, name, value, mod):
Johannes Gijsbers8de645a2004-11-07 19:16:05 +0000902 results = []
903 push = results.append
904
905 if name:
906 push('<dl><dt><strong>%s</strong></dt>\n' % name)
907 if value.__doc__ is not None:
Ka-Ping Yeebba6acc2005-02-19 22:58:26 +0000908 doc = self.markup(getdoc(value), self.preformat)
Johannes Gijsbers8de645a2004-11-07 19:16:05 +0000909 push('<dd><tt>%s</tt></dd>\n' % doc)
Johannes Gijsbers8de645a2004-11-07 19:16:05 +0000910 push('</dl>\n')
911
912 return ''.join(results)
913
914 def docproperty(self, object, name=None, mod=None, cl=None):
915 """Produce html documentation for a property."""
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +0000916 return self._docdescriptor(name, object, mod)
Johannes Gijsbers8de645a2004-11-07 19:16:05 +0000917
Tim Peters8dd7ade2001-10-18 19:56:17 +0000918 def docother(self, object, name=None, mod=None, *ignored):
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000919 """Produce HTML documentation for a data object."""
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +0000920 lhs = name and '<strong>%s</strong> = ' % name or ''
921 return lhs + self.repr(object)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000922
Barry Warsaw00decd72006-07-27 23:43:15 +0000923 def docdata(self, object, name=None, mod=None, cl=None):
924 """Produce html documentation for a data descriptor."""
925 return self._docdescriptor(name, object, mod)
926
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000927 def index(self, dir, shadowed=None):
928 """Generate an HTML index for a directory of modules."""
929 modpkgs = []
930 if shadowed is None: shadowed = {}
Phillip J. Ebyceb30872006-04-18 00:59:55 +0000931 for importer, name, ispkg in pkgutil.iter_modules([dir]):
932 modpkgs.append((name, '', ispkg, name in shadowed))
933 shadowed[name] = 1
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000934
935 modpkgs.sort()
936 contents = self.multicolumn(modpkgs, self.modpkglink)
937 return self.bigsection(dir, '#ffffff', '#ee77aa', contents)
938
939# -------------------------------------------- text documentation generator
940
941class TextRepr(Repr):
942 """Class for safely making a text representation of a Python object."""
943 def __init__(self):
944 Repr.__init__(self)
Ka-Ping Yeedec96e92001-04-13 09:55:49 +0000945 self.maxlist = self.maxtuple = 20
946 self.maxdict = 10
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000947 self.maxstring = self.maxother = 100
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000948
949 def repr1(self, x, level):
Skip Montanaro0fe8fce2003-06-27 15:45:41 +0000950 if hasattr(type(x), '__name__'):
951 methodname = 'repr_' + join(split(type(x).__name__), '_')
952 if hasattr(self, methodname):
953 return getattr(self, methodname)(x, level)
954 return cram(stripid(repr(x)), self.maxother)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000955
Ka-Ping Yeea2fe1032001-03-02 01:19:14 +0000956 def repr_string(self, x, level):
957 test = cram(x, self.maxstring)
958 testrepr = repr(test)
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +0000959 if '\\' in test and '\\' not in replace(testrepr, r'\\', ''):
Ka-Ping Yeea2fe1032001-03-02 01:19:14 +0000960 # Backslashes are only literal in the string and are never
961 # needed to make any special characters, so show a raw string.
962 return 'r' + testrepr[0] + test + testrepr[0]
963 return testrepr
964
Skip Montanarodf708782002-03-07 22:58:02 +0000965 repr_str = repr_string
966
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000967 def repr_instance(self, x, level):
968 try:
Ka-Ping Yee1d384632001-03-01 00:24:32 +0000969 return cram(stripid(repr(x)), self.maxstring)
Ka-Ping Yeedd175342001-02-27 14:43:46 +0000970 except:
971 return '<%s instance>' % x.__class__.__name__
972
973class TextDoc(Doc):
974 """Formatter class for text documentation."""
975
976 # ------------------------------------------- text formatting utilities
977
978 _repr_instance = TextRepr()
979 repr = _repr_instance.repr
980
981 def bold(self, text):
982 """Format a string in bold by overstriking."""
983 return join(map(lambda ch: ch + '\b' + ch, text), '')
984
985 def indent(self, text, prefix=' '):
986 """Indent text by prepending a given prefix to each line."""
987 if not text: return ''
988 lines = split(text, '\n')
989 lines = map(lambda line, prefix=prefix: prefix + line, lines)
990 if lines: lines[-1] = rstrip(lines[-1])
991 return join(lines, '\n')
992
993 def section(self, title, contents):
994 """Format a section with a given heading."""
995 return self.bold(title) + '\n' + rstrip(self.indent(contents)) + '\n\n'
996
997 # ---------------------------------------------- type-specific routines
998
Ka-Ping Yee37f7b382001-03-23 00:12:53 +0000999 def formattree(self, tree, modname, parent=None, prefix=''):
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001000 """Render in text a class tree as returned by inspect.getclasstree()."""
1001 result = ''
1002 for entry in tree:
1003 if type(entry) is type(()):
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001004 c, bases = entry
1005 result = result + prefix + classname(c, modname)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001006 if bases and bases != (parent,):
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001007 parents = map(lambda c, m=modname: classname(c, m), bases)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001008 result = result + '(%s)' % join(parents, ', ')
1009 result = result + '\n'
1010 elif type(entry) is type([]):
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001011 result = result + self.formattree(
1012 entry, modname, c, prefix + ' ')
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001013 return result
1014
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001015 def docmodule(self, object, name=None, mod=None):
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001016 """Produce text documentation for a given module object."""
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001017 name = object.__name__ # ignore the passed-in name
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +00001018 synop, desc = splitdoc(getdoc(object))
1019 result = self.section('NAME', name + (synop and ' - ' + synop))
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001020
1021 try:
Skip Montanaroa5616d22004-06-11 04:46:12 +00001022 all = object.__all__
1023 except AttributeError:
1024 all = None
1025
1026 try:
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001027 file = inspect.getabsfile(object)
1028 except TypeError:
1029 file = '(built-in)'
Ka-Ping Yee239432a2001-03-02 02:45:08 +00001030 result = result + self.section('FILE', file)
Skip Montanaro4997a692003-09-10 16:47:51 +00001031
1032 docloc = self.getdocloc(object)
1033 if docloc is not None:
1034 result = result + self.section('MODULE DOCS', docloc)
1035
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +00001036 if desc:
1037 result = result + self.section('DESCRIPTION', desc)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001038
1039 classes = []
1040 for key, value in inspect.getmembers(object, inspect.isclass):
Johannes Gijsbers4c11f602004-08-30 14:13:04 +00001041 # if __all__ exists, believe it. Otherwise use old heuristic.
1042 if (all is not None
1043 or (inspect.getmodule(value) or object) is object):
Skip Montanaroa5616d22004-06-11 04:46:12 +00001044 if visiblename(key, all):
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +00001045 classes.append((key, value))
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001046 funcs = []
1047 for key, value in inspect.getmembers(object, inspect.isroutine):
Johannes Gijsbers4c11f602004-08-30 14:13:04 +00001048 # if __all__ exists, believe it. Otherwise use old heuristic.
1049 if (all is not None or
1050 inspect.isbuiltin(value) or inspect.getmodule(value) is object):
Skip Montanaroa5616d22004-06-11 04:46:12 +00001051 if visiblename(key, all):
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +00001052 funcs.append((key, value))
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001053 data = []
1054 for key, value in inspect.getmembers(object, isdata):
Skip Montanaroa5616d22004-06-11 04:46:12 +00001055 if visiblename(key, all):
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001056 data.append((key, value))
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001057
Georg Brandlf2dae0e2008-01-21 21:05:49 +00001058 modpkgs = []
1059 modpkgs_names = set()
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001060 if hasattr(object, '__path__'):
Phillip J. Ebyceb30872006-04-18 00:59:55 +00001061 for importer, modname, ispkg in pkgutil.iter_modules(object.__path__):
Georg Brandlf2dae0e2008-01-21 21:05:49 +00001062 modpkgs_names.add(modname)
Phillip J. Ebyceb30872006-04-18 00:59:55 +00001063 if ispkg:
1064 modpkgs.append(modname + ' (package)')
1065 else:
1066 modpkgs.append(modname)
1067
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001068 modpkgs.sort()
1069 result = result + self.section(
1070 'PACKAGE CONTENTS', join(modpkgs, '\n'))
1071
Georg Brandlf2dae0e2008-01-21 21:05:49 +00001072 # Detect submodules as sometimes created by C extensions
1073 submodules = []
1074 for key, value in inspect.getmembers(object, inspect.ismodule):
1075 if value.__name__.startswith(name + '.') and key not in modpkgs_names:
1076 submodules.append(key)
1077 if submodules:
1078 submodules.sort()
1079 result = result + self.section(
1080 'SUBMODULES', join(submodules, '\n'))
1081
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001082 if classes:
Brett Cannonaca98b02008-08-03 00:58:51 +00001083 classlist = map(lambda key_value: key_value[1], classes)
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001084 contents = [self.formattree(
1085 inspect.getclasstree(classlist, 1), name)]
1086 for key, value in classes:
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001087 contents.append(self.document(value, key, name))
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001088 result = result + self.section('CLASSES', join(contents, '\n'))
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001089
1090 if funcs:
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001091 contents = []
1092 for key, value in funcs:
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001093 contents.append(self.document(value, key, name))
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001094 result = result + self.section('FUNCTIONS', join(contents, '\n'))
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001095
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001096 if data:
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001097 contents = []
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001098 for key, value in data:
Georg Brandl8b813db2005-10-01 16:32:31 +00001099 contents.append(self.docother(value, key, name, maxlen=70))
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001100 result = result + self.section('DATA', join(contents, '\n'))
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001101
1102 if hasattr(object, '__version__'):
1103 version = str(object.__version__)
Ka-Ping Yee1d384632001-03-01 00:24:32 +00001104 if version[:11] == '$' + 'Revision: ' and version[-1:] == '$':
1105 version = strip(version[11:-1])
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001106 result = result + self.section('VERSION', version)
Ka-Ping Yee6f3f9a42001-02-27 22:42:36 +00001107 if hasattr(object, '__date__'):
1108 result = result + self.section('DATE', str(object.__date__))
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001109 if hasattr(object, '__author__'):
Ka-Ping Yee6f3f9a42001-02-27 22:42:36 +00001110 result = result + self.section('AUTHOR', str(object.__author__))
1111 if hasattr(object, '__credits__'):
1112 result = result + self.section('CREDITS', str(object.__credits__))
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001113 return result
1114
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001115 def docclass(self, object, name=None, mod=None):
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001116 """Produce text documentation for a given class object."""
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001117 realname = object.__name__
1118 name = name or realname
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001119 bases = object.__bases__
1120
Tim Petersc86f6ca2001-09-26 21:31:51 +00001121 def makename(c, m=object.__module__):
1122 return classname(c, m)
1123
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001124 if name == realname:
1125 title = 'class ' + self.bold(realname)
1126 else:
1127 title = self.bold(name) + ' = class ' + realname
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001128 if bases:
Ka-Ping Yee3bda8792001-03-23 13:17:50 +00001129 parents = map(makename, bases)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001130 title = title + '(%s)' % join(parents, ', ')
1131
1132 doc = getdoc(object)
Tim Peters28355492001-09-23 21:29:55 +00001133 contents = doc and [doc + '\n'] or []
1134 push = contents.append
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001135
Tim Petersc86f6ca2001-09-26 21:31:51 +00001136 # List the mro, if non-trivial.
Raymond Hettinger756b3f32004-01-29 06:37:52 +00001137 mro = deque(inspect.getmro(object))
Tim Petersc86f6ca2001-09-26 21:31:51 +00001138 if len(mro) > 2:
1139 push("Method resolution order:")
1140 for base in mro:
1141 push(' ' + makename(base))
1142 push('')
1143
Tim Petersf4aad8e2001-09-24 22:40:47 +00001144 # Cute little class to pump out a horizontal rule between sections.
1145 class HorizontalRule:
1146 def __init__(self):
1147 self.needone = 0
1148 def maybe(self):
1149 if self.needone:
1150 push('-' * 70)
1151 self.needone = 1
1152 hr = HorizontalRule()
1153
Tim Peters28355492001-09-23 21:29:55 +00001154 def spill(msg, attrs, predicate):
Tim Petersfa26f7c2001-09-24 08:05:11 +00001155 ok, attrs = _split_list(attrs, predicate)
Tim Peters28355492001-09-23 21:29:55 +00001156 if ok:
Tim Petersf4aad8e2001-09-24 22:40:47 +00001157 hr.maybe()
Tim Peters28355492001-09-23 21:29:55 +00001158 push(msg)
1159 for name, kind, homecls, value in ok:
1160 push(self.document(getattr(object, name),
1161 name, mod, object))
1162 return attrs
1163
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +00001164 def spilldescriptors(msg, attrs, predicate):
Tim Petersfa26f7c2001-09-24 08:05:11 +00001165 ok, attrs = _split_list(attrs, predicate)
Tim Peters28355492001-09-23 21:29:55 +00001166 if ok:
Tim Petersf4aad8e2001-09-24 22:40:47 +00001167 hr.maybe()
Tim Peters28355492001-09-23 21:29:55 +00001168 push(msg)
1169 for name, kind, homecls, value in ok:
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +00001170 push(self._docdescriptor(name, value, mod))
Tim Peters28355492001-09-23 21:29:55 +00001171 return attrs
Tim Petersb47879b2001-09-24 04:47:19 +00001172
Tim Petersfa26f7c2001-09-24 08:05:11 +00001173 def spilldata(msg, attrs, predicate):
1174 ok, attrs = _split_list(attrs, predicate)
Tim Peters28355492001-09-23 21:29:55 +00001175 if ok:
Tim Petersf4aad8e2001-09-24 22:40:47 +00001176 hr.maybe()
Tim Peters28355492001-09-23 21:29:55 +00001177 push(msg)
1178 for name, kind, homecls, value in ok:
Brett Cannonaca98b02008-08-03 00:58:51 +00001179 if (hasattr(value, '__call__') or
1180 inspect.isdatadescriptor(value)):
Ka-Ping Yeebba6acc2005-02-19 22:58:26 +00001181 doc = getdoc(value)
Guido van Rossum5e355b22002-05-21 20:56:15 +00001182 else:
1183 doc = None
Tim Peters28355492001-09-23 21:29:55 +00001184 push(self.docother(getattr(object, name),
Georg Brandl8b813db2005-10-01 16:32:31 +00001185 name, mod, maxlen=70, doc=doc) + '\n')
Tim Peters28355492001-09-23 21:29:55 +00001186 return attrs
1187
Brett Cannonaca98b02008-08-03 00:58:51 +00001188 attrs = filter(lambda data: visiblename(data[0]),
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +00001189 classify_class_attrs(object))
Tim Petersfa26f7c2001-09-24 08:05:11 +00001190 while attrs:
Tim Peters351e3622001-09-27 03:29:51 +00001191 if mro:
Raymond Hettinger756b3f32004-01-29 06:37:52 +00001192 thisclass = mro.popleft()
Tim Peters351e3622001-09-27 03:29:51 +00001193 else:
1194 thisclass = attrs[0][2]
Tim Petersfa26f7c2001-09-24 08:05:11 +00001195 attrs, inherited = _split_list(attrs, lambda t: t[2] is thisclass)
1196
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +00001197 if thisclass is __builtin__.object:
1198 attrs = inherited
1199 continue
1200 elif thisclass is object:
Tim Peters28355492001-09-23 21:29:55 +00001201 tag = "defined here"
1202 else:
Tim Petersfa26f7c2001-09-24 08:05:11 +00001203 tag = "inherited from %s" % classname(thisclass,
1204 object.__module__)
Tim Peters28355492001-09-23 21:29:55 +00001205
1206 # Sort attrs by name.
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +00001207 attrs.sort()
Tim Peters28355492001-09-23 21:29:55 +00001208
1209 # Pump out the attrs, segregated by kind.
Tim Petersf4aad8e2001-09-24 22:40:47 +00001210 attrs = spill("Methods %s:\n" % tag, attrs,
Tim Peters28355492001-09-23 21:29:55 +00001211 lambda t: t[1] == 'method')
Tim Petersf4aad8e2001-09-24 22:40:47 +00001212 attrs = spill("Class methods %s:\n" % tag, attrs,
Tim Peters28355492001-09-23 21:29:55 +00001213 lambda t: t[1] == 'class method')
Tim Petersf4aad8e2001-09-24 22:40:47 +00001214 attrs = spill("Static methods %s:\n" % tag, attrs,
Tim Peters28355492001-09-23 21:29:55 +00001215 lambda t: t[1] == 'static method')
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +00001216 attrs = spilldescriptors("Data descriptors %s:\n" % tag, attrs,
1217 lambda t: t[1] == 'data descriptor')
Ka-Ping Yeed9e213e2003-03-28 16:35:51 +00001218 attrs = spilldata("Data and other attributes %s:\n" % tag, attrs,
1219 lambda t: t[1] == 'data')
Tim Peters28355492001-09-23 21:29:55 +00001220 assert attrs == []
Tim Peters351e3622001-09-27 03:29:51 +00001221 attrs = inherited
Tim Peters28355492001-09-23 21:29:55 +00001222
1223 contents = '\n'.join(contents)
1224 if not contents:
1225 return title + '\n'
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001226 return title + '\n' + self.indent(rstrip(contents), ' | ') + '\n'
1227
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001228 def formatvalue(self, object):
1229 """Format an argument default value as text."""
1230 return '=' + self.repr(object)
1231
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001232 def docroutine(self, object, name=None, mod=None, cl=None):
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001233 """Produce text documentation for a function or method object."""
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001234 realname = object.__name__
1235 name = name or realname
1236 note = ''
Ka-Ping Yee3bda8792001-03-23 13:17:50 +00001237 skipdocs = 0
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001238 if inspect.ismethod(object):
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001239 imclass = object.im_class
Ka-Ping Yee3bda8792001-03-23 13:17:50 +00001240 if cl:
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001241 if imclass is not cl:
1242 note = ' from ' + classname(imclass, mod)
Ka-Ping Yee3bda8792001-03-23 13:17:50 +00001243 else:
Georg Brandl7e2b6bb2007-03-13 22:16:30 +00001244 if object.im_self is not None:
Ka-Ping Yeeb7a48302001-04-12 20:39:14 +00001245 note = ' method of %s instance' % classname(
1246 object.im_self.__class__, mod)
1247 else:
1248 note = ' unbound %s method' % classname(imclass,mod)
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001249 object = object.im_func
1250
1251 if name == realname:
1252 title = self.bold(realname)
1253 else:
Raymond Hettinger54f02222002-06-01 14:18:47 +00001254 if (cl and realname in cl.__dict__ and
Ka-Ping Yee3bda8792001-03-23 13:17:50 +00001255 cl.__dict__[realname] is object):
1256 skipdocs = 1
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001257 title = self.bold(name) + ' = ' + realname
Tim Peters4bcfa312001-09-20 06:08:24 +00001258 if inspect.isfunction(object):
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001259 args, varargs, varkw, defaults = inspect.getargspec(object)
1260 argspec = inspect.formatargspec(
1261 args, varargs, varkw, defaults, formatvalue=self.formatvalue)
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001262 if realname == '<lambda>':
Georg Brandl501dd0d2006-02-17 09:45:40 +00001263 title = self.bold(name) + ' lambda '
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001264 argspec = argspec[1:-1] # remove parentheses
Tim Peters4bcfa312001-09-20 06:08:24 +00001265 else:
1266 argspec = '(...)'
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001267 decl = title + argspec + note
1268
Ka-Ping Yee3bda8792001-03-23 13:17:50 +00001269 if skipdocs:
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001270 return decl + '\n'
Ka-Ping Yee3bda8792001-03-23 13:17:50 +00001271 else:
1272 doc = getdoc(object) or ''
1273 return decl + '\n' + (doc and rstrip(self.indent(doc)) + '\n')
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001274
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +00001275 def _docdescriptor(self, name, value, mod):
Johannes Gijsbers8de645a2004-11-07 19:16:05 +00001276 results = []
1277 push = results.append
1278
1279 if name:
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +00001280 push(self.bold(name))
1281 push('\n')
Johannes Gijsbers8de645a2004-11-07 19:16:05 +00001282 doc = getdoc(value) or ''
1283 if doc:
1284 push(self.indent(doc))
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +00001285 push('\n')
1286 return ''.join(results)
Johannes Gijsbers8de645a2004-11-07 19:16:05 +00001287
1288 def docproperty(self, object, name=None, mod=None, cl=None):
1289 """Produce text documentation for a property."""
Johannes Gijsbers9ddb3002005-01-08 20:16:43 +00001290 return self._docdescriptor(name, object, mod)
Johannes Gijsbers8de645a2004-11-07 19:16:05 +00001291
Barry Warsaw00decd72006-07-27 23:43:15 +00001292 def docdata(self, object, name=None, mod=None, cl=None):
1293 """Produce text documentation for a data descriptor."""
1294 return self._docdescriptor(name, object, mod)
1295
Georg Brandl8b813db2005-10-01 16:32:31 +00001296 def docother(self, object, name=None, mod=None, parent=None, maxlen=None, doc=None):
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001297 """Produce text documentation for a data object."""
1298 repr = self.repr(object)
1299 if maxlen:
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001300 line = (name and name + ' = ' or '') + repr
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001301 chop = maxlen - len(line)
1302 if chop < 0: repr = repr[:chop] + '...'
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001303 line = (name and self.bold(name) + ' = ' or '') + repr
Tim Peters28355492001-09-23 21:29:55 +00001304 if doc is not None:
1305 line += '\n' + self.indent(str(doc))
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001306 return line
1307
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001308# --------------------------------------------------------- user interfaces
1309
1310def pager(text):
1311 """The first time this is called, determine what kind of pager to use."""
1312 global pager
1313 pager = getpager()
1314 pager(text)
1315
1316def getpager():
1317 """Decide what method to use for paging through text."""
1318 if type(sys.stdout) is not types.FileType:
1319 return plainpager
1320 if not sys.stdin.isatty() or not sys.stdout.isatty():
1321 return plainpager
Raymond Hettinger54f02222002-06-01 14:18:47 +00001322 if 'PAGER' in os.environ:
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +00001323 if sys.platform == 'win32': # pipes completely broken in Windows
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001324 return lambda text: tempfilepager(plain(text), os.environ['PAGER'])
Raymond Hettingerdbecd932005-02-06 06:57:08 +00001325 elif os.environ.get('TERM') in ('dumb', 'emacs'):
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001326 return lambda text: pipepager(plain(text), os.environ['PAGER'])
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +00001327 else:
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001328 return lambda text: pipepager(text, os.environ['PAGER'])
Ka-Ping Yeea487e4e2005-11-05 04:49:18 +00001329 if os.environ.get('TERM') in ('dumb', 'emacs'):
1330 return plainpager
Andrew MacIntyre54e0eab2002-03-03 03:12:30 +00001331 if sys.platform == 'win32' or sys.platform.startswith('os2'):
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001332 return lambda text: tempfilepager(plain(text), 'more <')
Skip Montanarod404bee2002-09-26 21:44:57 +00001333 if hasattr(os, 'system') and os.system('(less) 2>/dev/null') == 0:
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001334 return lambda text: pipepager(text, 'less')
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001335
1336 import tempfile
Guido van Rossum3b0a3292002-08-09 16:38:32 +00001337 (fd, filename) = tempfile.mkstemp()
1338 os.close(fd)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001339 try:
Georg Brandlb32dea52008-07-16 21:19:28 +00001340 if hasattr(os, 'system') and os.system('more "%s"' % filename) == 0:
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001341 return lambda text: pipepager(text, 'more')
1342 else:
1343 return ttypager
1344 finally:
1345 os.unlink(filename)
1346
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001347def plain(text):
1348 """Remove boldface formatting from text."""
1349 return re.sub('.\b', '', text)
1350
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001351def pipepager(text, cmd):
1352 """Page through text by feeding it to another program."""
1353 pipe = os.popen(cmd, 'w')
1354 try:
1355 pipe.write(text)
1356 pipe.close()
1357 except IOError:
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001358 pass # Ignore broken pipes caused by quitting the pager program.
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001359
1360def tempfilepager(text, cmd):
1361 """Page through text by invoking a program on a temporary file."""
1362 import tempfile
Tim Peters550e4e52003-02-07 01:53:46 +00001363 filename = tempfile.mktemp()
1364 file = open(filename, 'w')
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001365 file.write(text)
1366 file.close()
1367 try:
Georg Brandlb32dea52008-07-16 21:19:28 +00001368 os.system(cmd + ' "' + filename + '"')
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001369 finally:
1370 os.unlink(filename)
1371
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001372def ttypager(text):
1373 """Page through text on a text terminal."""
1374 lines = split(plain(text), '\n')
1375 try:
1376 import tty
1377 fd = sys.stdin.fileno()
1378 old = tty.tcgetattr(fd)
1379 tty.setcbreak(fd)
1380 getchar = lambda: sys.stdin.read(1)
Ka-Ping Yee457aab22001-02-27 23:36:29 +00001381 except (ImportError, AttributeError):
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001382 tty = None
1383 getchar = lambda: sys.stdin.readline()[:-1][:1]
1384
1385 try:
1386 r = inc = os.environ.get('LINES', 25) - 1
1387 sys.stdout.write(join(lines[:inc], '\n') + '\n')
1388 while lines[r:]:
1389 sys.stdout.write('-- more --')
1390 sys.stdout.flush()
1391 c = getchar()
1392
Raymond Hettingerdbecd932005-02-06 06:57:08 +00001393 if c in ('q', 'Q'):
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001394 sys.stdout.write('\r \r')
1395 break
Raymond Hettingerdbecd932005-02-06 06:57:08 +00001396 elif c in ('\r', '\n'):
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001397 sys.stdout.write('\r \r' + lines[r] + '\n')
1398 r = r + 1
1399 continue
Raymond Hettingerdbecd932005-02-06 06:57:08 +00001400 if c in ('b', 'B', '\x1b'):
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001401 r = r - inc - inc
1402 if r < 0: r = 0
1403 sys.stdout.write('\n' + join(lines[r:r+inc], '\n') + '\n')
1404 r = r + inc
1405
1406 finally:
1407 if tty:
1408 tty.tcsetattr(fd, tty.TCSAFLUSH, old)
1409
1410def plainpager(text):
1411 """Simply print unformatted text. This is the ultimate fallback."""
1412 sys.stdout.write(plain(text))
1413
1414def describe(thing):
Ka-Ping Yee3bda8792001-03-23 13:17:50 +00001415 """Produce a short description of the given thing."""
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001416 if inspect.ismodule(thing):
1417 if thing.__name__ in sys.builtin_module_names:
1418 return 'built-in module ' + thing.__name__
1419 if hasattr(thing, '__path__'):
1420 return 'package ' + thing.__name__
1421 else:
1422 return 'module ' + thing.__name__
1423 if inspect.isbuiltin(thing):
1424 return 'built-in function ' + thing.__name__
Barry Warsaw00decd72006-07-27 23:43:15 +00001425 if inspect.isgetsetdescriptor(thing):
1426 return 'getset descriptor %s.%s.%s' % (
1427 thing.__objclass__.__module__, thing.__objclass__.__name__,
1428 thing.__name__)
1429 if inspect.ismemberdescriptor(thing):
1430 return 'member descriptor %s.%s.%s' % (
1431 thing.__objclass__.__module__, thing.__objclass__.__name__,
1432 thing.__name__)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001433 if inspect.isclass(thing):
1434 return 'class ' + thing.__name__
1435 if inspect.isfunction(thing):
1436 return 'function ' + thing.__name__
1437 if inspect.ismethod(thing):
1438 return 'method ' + thing.__name__
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001439 if type(thing) is types.InstanceType:
1440 return 'instance of ' + thing.__class__.__name__
1441 return type(thing).__name__
1442
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001443def locate(path, forceload=0):
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001444 """Locate an object by name or dotted path, importing as necessary."""
Guido van Rossum97dede02003-02-16 01:12:32 +00001445 parts = [part for part in split(path, '.') if part]
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001446 module, n = None, 0
1447 while n < len(parts):
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001448 nextmodule = safeimport(join(parts[:n+1], '.'), forceload)
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001449 if nextmodule: module, n = nextmodule, n + 1
1450 else: break
1451 if module:
1452 object = module
1453 for part in parts[n:]:
1454 try: object = getattr(object, part)
1455 except AttributeError: return None
1456 return object
1457 else:
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001458 if hasattr(__builtin__, path):
1459 return getattr(__builtin__, path)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001460
1461# --------------------------------------- interactive interpreter interface
1462
1463text = TextDoc()
1464html = HTMLDoc()
1465
Ka-Ping Yee8ef1cf32007-01-14 04:25:15 +00001466class _OldStyleClass: pass
1467_OLD_INSTANCE_TYPE = type(_OldStyleClass())
1468
Ka-Ping Yee45daeb02002-08-11 15:11:33 +00001469def resolve(thing, forceload=0):
1470 """Given an object or a path to an object, get the object and its name."""
1471 if isinstance(thing, str):
1472 object = locate(thing, forceload)
1473 if not object:
1474 raise ImportError, 'no Python documentation found for %r' % thing
1475 return object, thing
1476 else:
1477 return thing, getattr(thing, '__name__', None)
1478
Georg Brandl8441f152007-03-13 20:02:57 +00001479def render_doc(thing, title='Python Library Documentation: %s', forceload=0):
1480 """Render text documentation, given an object or a path to an object."""
1481 object, name = resolve(thing, forceload)
1482 desc = describe(object)
1483 module = inspect.getmodule(object)
1484 if name and '.' in name:
1485 desc += ' in ' + name[:name.rfind('.')]
1486 elif module and module is not object:
1487 desc += ' in module ' + module.__name__
1488 if type(object) is _OLD_INSTANCE_TYPE:
1489 # If the passed object is an instance of an old-style class,
1490 # document its available methods instead of its value.
1491 object = object.__class__
1492 elif not (inspect.ismodule(object) or
1493 inspect.isclass(object) or
1494 inspect.isroutine(object) or
1495 inspect.isgetsetdescriptor(object) or
1496 inspect.ismemberdescriptor(object) or
1497 isinstance(object, property)):
1498 # If the passed object is a piece of data or an instance,
1499 # document its available methods instead of its value.
1500 object = type(object)
1501 desc += ' object'
1502 return title % desc + '\n\n' + text.document(object, name)
1503
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001504def doc(thing, title='Python Library Documentation: %s', forceload=0):
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001505 """Display text documentation, given an object or a path to an object."""
Ka-Ping Yee45daeb02002-08-11 15:11:33 +00001506 try:
Georg Brandl8441f152007-03-13 20:02:57 +00001507 pager(render_doc(thing, title, forceload))
Ka-Ping Yee45daeb02002-08-11 15:11:33 +00001508 except (ImportError, ErrorDuringImport), value:
1509 print value
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001510
Ka-Ping Yee45daeb02002-08-11 15:11:33 +00001511def writedoc(thing, forceload=0):
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001512 """Write HTML documentation to a file in the current directory."""
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001513 try:
Ka-Ping Yee45daeb02002-08-11 15:11:33 +00001514 object, name = resolve(thing, forceload)
1515 page = html.page(describe(object), html.document(object, name))
1516 file = open(name + '.html', 'w')
1517 file.write(page)
1518 file.close()
1519 print 'wrote', name + '.html'
1520 except (ImportError, ErrorDuringImport), value:
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001521 print value
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001522
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001523def writedocs(dir, pkgpath='', done=None):
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001524 """Write out HTML documentation for all modules in a directory tree."""
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001525 if done is None: done = {}
Phillip J. Ebyceb30872006-04-18 00:59:55 +00001526 for importer, modname, ispkg in pkgutil.walk_packages([dir], pkgpath):
1527 writedoc(modname)
1528 return
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001529
1530class Helper:
Georg Brandl681001e2008-06-01 20:33:55 +00001531
1532 # These dictionaries map a topic name to either an alias, or a tuple
1533 # (label, seealso-items). The "label" is the label of the corresponding
1534 # section in the .rst file under Doc/ and an index into the dictionary
Georg Brandl43819252009-04-26 09:56:44 +00001535 # in pydoc_data/topics.py.
Georg Brandl681001e2008-06-01 20:33:55 +00001536 #
1537 # CAUTION: if you change one of these dictionaries, be sure to adapt the
1538 # list of needed labels in Doc/tools/sphinxext/pyspecific.py and
Georg Brandl43819252009-04-26 09:56:44 +00001539 # regenerate the pydoc_data/topics.py file by running
Georg Brandl681001e2008-06-01 20:33:55 +00001540 # make pydoc-topics
1541 # in Doc/ and copying the output file into the Lib/ directory.
1542
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001543 keywords = {
1544 'and': 'BOOLEAN',
Georg Brandlb6a87542007-03-13 10:06:48 +00001545 'as': 'with',
Georg Brandl681001e2008-06-01 20:33:55 +00001546 'assert': ('assert', ''),
1547 'break': ('break', 'while for'),
1548 'class': ('class', 'CLASSES SPECIALMETHODS'),
1549 'continue': ('continue', 'while for'),
1550 'def': ('function', ''),
1551 'del': ('del', 'BASICMETHODS'),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001552 'elif': 'if',
Georg Brandl681001e2008-06-01 20:33:55 +00001553 'else': ('else', 'while for'),
Georg Brandl5c669db2008-08-30 19:03:43 +00001554 'except': 'try',
Georg Brandl681001e2008-06-01 20:33:55 +00001555 'exec': ('exec', ''),
Georg Brandl5c669db2008-08-30 19:03:43 +00001556 'finally': 'try',
Georg Brandl681001e2008-06-01 20:33:55 +00001557 'for': ('for', 'break continue while'),
Georg Brandl5c669db2008-08-30 19:03:43 +00001558 'from': 'import',
Georg Brandl681001e2008-06-01 20:33:55 +00001559 'global': ('global', 'NAMESPACES'),
1560 'if': ('if', 'TRUTHVALUE'),
1561 'import': ('import', 'MODULES'),
1562 'in': ('in', 'SEQUENCEMETHODS2'),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001563 'is': 'COMPARISON',
Georg Brandl681001e2008-06-01 20:33:55 +00001564 'lambda': ('lambda', 'FUNCTIONS'),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001565 'not': 'BOOLEAN',
1566 'or': 'BOOLEAN',
Georg Brandl681001e2008-06-01 20:33:55 +00001567 'pass': ('pass', ''),
1568 'print': ('print', ''),
1569 'raise': ('raise', 'EXCEPTIONS'),
1570 'return': ('return', 'FUNCTIONS'),
1571 'try': ('try', 'EXCEPTIONS'),
1572 'while': ('while', 'break continue if TRUTHVALUE'),
1573 'with': ('with', 'CONTEXTMANAGERS EXCEPTIONS yield'),
1574 'yield': ('yield', ''),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001575 }
Georg Brandlc3a98032008-12-27 18:20:04 +00001576 # Either add symbols to this dictionary or to the symbols dictionary
1577 # directly: Whichever is easier. They are merged later.
1578 _symbols_inverse = {
1579 'STRINGS' : ("'", "'''", "r'", "u'", '"""', '"', 'r"', 'u"'),
1580 'OPERATORS' : ('+', '-', '*', '**', '/', '//', '%', '<<', '>>', '&',
1581 '|', '^', '~', '<', '>', '<=', '>=', '==', '!=', '<>'),
1582 'COMPARISON' : ('<', '>', '<=', '>=', '==', '!=', '<>'),
1583 'UNARY' : ('-', '~'),
1584 'AUGMENTEDASSIGNMENT' : ('+=', '-=', '*=', '/=', '%=', '&=', '|=',
1585 '^=', '<<=', '>>=', '**=', '//='),
1586 'BITWISE' : ('<<', '>>', '&', '|', '^', '~'),
1587 'COMPLEX' : ('j', 'J')
1588 }
1589 symbols = {
1590 '%': 'OPERATORS FORMATTING',
1591 '**': 'POWER',
1592 ',': 'TUPLES LISTS FUNCTIONS',
1593 '.': 'ATTRIBUTES FLOAT MODULES OBJECTS',
1594 '...': 'ELLIPSIS',
1595 ':': 'SLICINGS DICTIONARYLITERALS',
1596 '@': 'def class',
1597 '\\': 'STRINGS',
1598 '_': 'PRIVATENAMES',
1599 '__': 'PRIVATENAMES SPECIALMETHODS',
1600 '`': 'BACKQUOTES',
1601 '(': 'TUPLES FUNCTIONS CALLS',
1602 ')': 'TUPLES FUNCTIONS CALLS',
1603 '[': 'LISTS SUBSCRIPTS SLICINGS',
1604 ']': 'LISTS SUBSCRIPTS SLICINGS'
1605 }
1606 for topic, symbols_ in _symbols_inverse.iteritems():
1607 for symbol in symbols_:
1608 topics = symbols.get(symbol, topic)
1609 if topic not in topics:
1610 topics = topics + ' ' + topic
1611 symbols[symbol] = topics
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001612
1613 topics = {
Georg Brandl681001e2008-06-01 20:33:55 +00001614 'TYPES': ('types', 'STRINGS UNICODE NUMBERS SEQUENCES MAPPINGS '
1615 'FUNCTIONS CLASSES MODULES FILES inspect'),
1616 'STRINGS': ('strings', 'str UNICODE SEQUENCES STRINGMETHODS FORMATTING '
1617 'TYPES'),
1618 'STRINGMETHODS': ('string-methods', 'STRINGS FORMATTING'),
1619 'FORMATTING': ('formatstrings', 'OPERATORS'),
1620 'UNICODE': ('strings', 'encodings unicode SEQUENCES STRINGMETHODS '
1621 'FORMATTING TYPES'),
1622 'NUMBERS': ('numbers', 'INTEGER FLOAT COMPLEX TYPES'),
1623 'INTEGER': ('integers', 'int range'),
1624 'FLOAT': ('floating', 'float math'),
1625 'COMPLEX': ('imaginary', 'complex cmath'),
1626 'SEQUENCES': ('typesseq', 'STRINGMETHODS FORMATTING xrange LISTS'),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001627 'MAPPINGS': 'DICTIONARIES',
Georg Brandl681001e2008-06-01 20:33:55 +00001628 'FUNCTIONS': ('typesfunctions', 'def TYPES'),
1629 'METHODS': ('typesmethods', 'class def CLASSES TYPES'),
1630 'CODEOBJECTS': ('bltin-code-objects', 'compile FUNCTIONS TYPES'),
1631 'TYPEOBJECTS': ('bltin-type-objects', 'types TYPES'),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001632 'FRAMEOBJECTS': 'TYPES',
1633 'TRACEBACKS': 'TYPES',
Georg Brandl681001e2008-06-01 20:33:55 +00001634 'NONE': ('bltin-null-object', ''),
1635 'ELLIPSIS': ('bltin-ellipsis-object', 'SLICINGS'),
1636 'FILES': ('bltin-file-objects', ''),
1637 'SPECIALATTRIBUTES': ('specialattrs', ''),
1638 'CLASSES': ('types', 'class SPECIALMETHODS PRIVATENAMES'),
1639 'MODULES': ('typesmodules', 'import'),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001640 'PACKAGES': 'import',
Georg Brandl681001e2008-06-01 20:33:55 +00001641 'EXPRESSIONS': ('operator-summary', 'lambda or and not in is BOOLEAN '
1642 'COMPARISON BITWISE SHIFTING BINARY FORMATTING POWER '
1643 'UNARY ATTRIBUTES SUBSCRIPTS SLICINGS CALLS TUPLES '
1644 'LISTS DICTIONARIES BACKQUOTES'),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001645 'OPERATORS': 'EXPRESSIONS',
1646 'PRECEDENCE': 'EXPRESSIONS',
Georg Brandl681001e2008-06-01 20:33:55 +00001647 'OBJECTS': ('objects', 'TYPES'),
1648 'SPECIALMETHODS': ('specialnames', 'BASICMETHODS ATTRIBUTEMETHODS '
1649 'CALLABLEMETHODS SEQUENCEMETHODS1 MAPPINGMETHODS '
1650 'SEQUENCEMETHODS2 NUMBERMETHODS CLASSES'),
1651 'BASICMETHODS': ('customization', 'cmp hash repr str SPECIALMETHODS'),
1652 'ATTRIBUTEMETHODS': ('attribute-access', 'ATTRIBUTES SPECIALMETHODS'),
1653 'CALLABLEMETHODS': ('callable-types', 'CALLS SPECIALMETHODS'),
1654 'SEQUENCEMETHODS1': ('sequence-types', 'SEQUENCES SEQUENCEMETHODS2 '
1655 'SPECIALMETHODS'),
1656 'SEQUENCEMETHODS2': ('sequence-methods', 'SEQUENCES SEQUENCEMETHODS1 '
1657 'SPECIALMETHODS'),
1658 'MAPPINGMETHODS': ('sequence-types', 'MAPPINGS SPECIALMETHODS'),
1659 'NUMBERMETHODS': ('numeric-types', 'NUMBERS AUGMENTEDASSIGNMENT '
1660 'SPECIALMETHODS'),
1661 'EXECUTION': ('execmodel', 'NAMESPACES DYNAMICFEATURES EXCEPTIONS'),
1662 'NAMESPACES': ('naming', 'global ASSIGNMENT DELETION DYNAMICFEATURES'),
1663 'DYNAMICFEATURES': ('dynamic-features', ''),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001664 'SCOPING': 'NAMESPACES',
1665 'FRAMES': 'NAMESPACES',
Georg Brandl681001e2008-06-01 20:33:55 +00001666 'EXCEPTIONS': ('exceptions', 'try except finally raise'),
1667 'COERCIONS': ('coercion-rules','CONVERSIONS'),
1668 'CONVERSIONS': ('conversions', 'COERCIONS'),
1669 'IDENTIFIERS': ('identifiers', 'keywords SPECIALIDENTIFIERS'),
1670 'SPECIALIDENTIFIERS': ('id-classes', ''),
1671 'PRIVATENAMES': ('atom-identifiers', ''),
1672 'LITERALS': ('atom-literals', 'STRINGS BACKQUOTES NUMBERS '
1673 'TUPLELITERALS LISTLITERALS DICTIONARYLITERALS'),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001674 'TUPLES': 'SEQUENCES',
Georg Brandl681001e2008-06-01 20:33:55 +00001675 'TUPLELITERALS': ('exprlists', 'TUPLES LITERALS'),
1676 'LISTS': ('typesseq-mutable', 'LISTLITERALS'),
1677 'LISTLITERALS': ('lists', 'LISTS LITERALS'),
1678 'DICTIONARIES': ('typesmapping', 'DICTIONARYLITERALS'),
1679 'DICTIONARYLITERALS': ('dict', 'DICTIONARIES LITERALS'),
1680 'BACKQUOTES': ('string-conversions', 'repr str STRINGS LITERALS'),
1681 'ATTRIBUTES': ('attribute-references', 'getattr hasattr setattr '
1682 'ATTRIBUTEMETHODS'),
1683 'SUBSCRIPTS': ('subscriptions', 'SEQUENCEMETHODS1'),
1684 'SLICINGS': ('slicings', 'SEQUENCEMETHODS2'),
1685 'CALLS': ('calls', 'EXPRESSIONS'),
1686 'POWER': ('power', 'EXPRESSIONS'),
1687 'UNARY': ('unary', 'EXPRESSIONS'),
1688 'BINARY': ('binary', 'EXPRESSIONS'),
1689 'SHIFTING': ('shifting', 'EXPRESSIONS'),
1690 'BITWISE': ('bitwise', 'EXPRESSIONS'),
1691 'COMPARISON': ('comparisons', 'EXPRESSIONS BASICMETHODS'),
1692 'BOOLEAN': ('booleans', 'EXPRESSIONS TRUTHVALUE'),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001693 'ASSERTION': 'assert',
Georg Brandl681001e2008-06-01 20:33:55 +00001694 'ASSIGNMENT': ('assignment', 'AUGMENTEDASSIGNMENT'),
1695 'AUGMENTEDASSIGNMENT': ('augassign', 'NUMBERMETHODS'),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001696 'DELETION': 'del',
1697 'PRINTING': 'print',
1698 'RETURNING': 'return',
1699 'IMPORTING': 'import',
1700 'CONDITIONAL': 'if',
Georg Brandl681001e2008-06-01 20:33:55 +00001701 'LOOPING': ('compound', 'for while break continue'),
1702 'TRUTHVALUE': ('truth', 'if while and or not BASICMETHODS'),
1703 'DEBUGGING': ('debugger', 'pdb'),
1704 'CONTEXTMANAGERS': ('context-managers', 'with'),
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001705 }
1706
1707 def __init__(self, input, output):
1708 self.input = input
1709 self.output = output
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001710
Ka-Ping Yee79c009d2001-04-13 10:53:25 +00001711 def __repr__(self):
Ka-Ping Yee9bc576b2001-04-13 13:57:31 +00001712 if inspect.stack()[1][3] == '?':
Ka-Ping Yee79c009d2001-04-13 10:53:25 +00001713 self()
1714 return ''
Ka-Ping Yee9bc576b2001-04-13 13:57:31 +00001715 return '<pydoc.Helper instance>'
Ka-Ping Yee79c009d2001-04-13 10:53:25 +00001716
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001717 def __call__(self, request=None):
1718 if request is not None:
1719 self.help(request)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001720 else:
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001721 self.intro()
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001722 self.interact()
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001723 self.output.write('''
Fred Drakee61967f2001-05-10 18:41:02 +00001724You are now leaving help and returning to the Python interpreter.
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001725If you want to ask for help on a particular object directly from the
1726interpreter, you can type "help(object)". Executing "help('string')"
1727has the same effect as typing a particular string at the help> prompt.
1728''')
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001729
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001730 def interact(self):
1731 self.output.write('\n')
Guido van Rossum8ca162f2002-04-07 06:36:23 +00001732 while True:
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001733 try:
Johannes Gijsberse7691d32004-08-17 13:21:53 +00001734 request = self.getline('help> ')
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001735 if not request: break
Johannes Gijsberse7691d32004-08-17 13:21:53 +00001736 except (KeyboardInterrupt, EOFError):
1737 break
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001738 request = strip(replace(request, '"', '', "'", ''))
Raymond Hettingerdbecd932005-02-06 06:57:08 +00001739 if lower(request) in ('q', 'quit'): break
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001740 self.help(request)
1741
Johannes Gijsberse7691d32004-08-17 13:21:53 +00001742 def getline(self, prompt):
1743 """Read one line, using raw_input when available."""
1744 if self.input is sys.stdin:
1745 return raw_input(prompt)
1746 else:
1747 self.output.write(prompt)
1748 self.output.flush()
1749 return self.input.readline()
1750
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001751 def help(self, request):
1752 if type(request) is type(''):
R. David Murrayd67ea7d2009-05-27 20:07:21 +00001753 request = request.strip()
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001754 if request == 'help': self.intro()
1755 elif request == 'keywords': self.listkeywords()
Georg Brandlc3a98032008-12-27 18:20:04 +00001756 elif request == 'symbols': self.listsymbols()
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001757 elif request == 'topics': self.listtopics()
1758 elif request == 'modules': self.listmodules()
1759 elif request[:8] == 'modules ':
1760 self.listmodules(split(request)[1])
Georg Brandlc3a98032008-12-27 18:20:04 +00001761 elif request in self.symbols: self.showsymbol(request)
Raymond Hettinger54f02222002-06-01 14:18:47 +00001762 elif request in self.keywords: self.showtopic(request)
1763 elif request in self.topics: self.showtopic(request)
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001764 elif request: doc(request, 'Help on %s:')
1765 elif isinstance(request, Helper): self()
1766 else: doc(request, 'Help on %s:')
1767 self.output.write('\n')
1768
1769 def intro(self):
1770 self.output.write('''
1771Welcome to Python %s! This is the online help utility.
1772
1773If this is your first time using Python, you should definitely check out
Georg Brandl0751d1a2008-01-21 17:13:03 +00001774the tutorial on the Internet at http://docs.python.org/tutorial/.
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001775
1776Enter the name of any module, keyword, or topic to get help on writing
1777Python programs and using Python modules. To quit this help utility and
1778return to the interpreter, just type "quit".
1779
1780To get a list of available modules, keywords, or topics, type "modules",
1781"keywords", or "topics". Each module also comes with a one-line summary
1782of what it does; to list the modules whose summaries contain a given word
1783such as "spam", type "modules spam".
1784''' % sys.version[:3])
1785
1786 def list(self, items, columns=4, width=80):
1787 items = items[:]
1788 items.sort()
1789 colw = width / columns
1790 rows = (len(items) + columns - 1) / columns
1791 for row in range(rows):
1792 for col in range(columns):
1793 i = col * rows + row
1794 if i < len(items):
1795 self.output.write(items[i])
1796 if col < columns - 1:
1797 self.output.write(' ' + ' ' * (colw-1 - len(items[i])))
1798 self.output.write('\n')
1799
1800 def listkeywords(self):
1801 self.output.write('''
1802Here is a list of the Python keywords. Enter any keyword to get more help.
1803
1804''')
1805 self.list(self.keywords.keys())
1806
Georg Brandlc3a98032008-12-27 18:20:04 +00001807 def listsymbols(self):
1808 self.output.write('''
1809Here is a list of the punctuation symbols which Python assigns special meaning
1810to. Enter any symbol to get more help.
1811
1812''')
1813 self.list(self.symbols.keys())
1814
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001815 def listtopics(self):
1816 self.output.write('''
1817Here is a list of available topics. Enter any topic name to get more help.
1818
1819''')
1820 self.list(self.topics.keys())
1821
Georg Brandlc3a98032008-12-27 18:20:04 +00001822 def showtopic(self, topic, more_xrefs=''):
Georg Brandl681001e2008-06-01 20:33:55 +00001823 try:
Georg Brandl43819252009-04-26 09:56:44 +00001824 import pydoc_data.topics
Georg Brandl681001e2008-06-01 20:33:55 +00001825 except ImportError:
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001826 self.output.write('''
Georg Brandl681001e2008-06-01 20:33:55 +00001827Sorry, topic and keyword documentation is not available because the
Georg Brandl43819252009-04-26 09:56:44 +00001828module "pydoc_data.topics" could not be found.
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001829''')
1830 return
1831 target = self.topics.get(topic, self.keywords.get(topic))
1832 if not target:
1833 self.output.write('no documentation found for %s\n' % repr(topic))
1834 return
1835 if type(target) is type(''):
Georg Brandlc3a98032008-12-27 18:20:04 +00001836 return self.showtopic(target, more_xrefs)
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001837
Georg Brandl681001e2008-06-01 20:33:55 +00001838 label, xrefs = target
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001839 try:
Georg Brandl43819252009-04-26 09:56:44 +00001840 doc = pydoc_data.topics.topics[label]
Georg Brandl681001e2008-06-01 20:33:55 +00001841 except KeyError:
1842 self.output.write('no documentation found for %s\n' % repr(topic))
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001843 return
Georg Brandl681001e2008-06-01 20:33:55 +00001844 pager(strip(doc) + '\n')
Georg Brandlc3a98032008-12-27 18:20:04 +00001845 if more_xrefs:
1846 xrefs = (xrefs or '') + ' ' + more_xrefs
Ka-Ping Yeeda793892001-04-13 11:02:51 +00001847 if xrefs:
Georg Brandl681001e2008-06-01 20:33:55 +00001848 import StringIO, formatter
Ka-Ping Yeeda793892001-04-13 11:02:51 +00001849 buffer = StringIO.StringIO()
1850 formatter.DumbWriter(buffer).send_flowing_data(
1851 'Related help topics: ' + join(split(xrefs), ', ') + '\n')
1852 self.output.write('\n%s\n' % buffer.getvalue())
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001853
Georg Brandlc3a98032008-12-27 18:20:04 +00001854 def showsymbol(self, symbol):
1855 target = self.symbols[symbol]
1856 topic, _, xrefs = target.partition(' ')
1857 self.showtopic(topic, xrefs)
1858
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001859 def listmodules(self, key=''):
1860 if key:
1861 self.output.write('''
1862Here is a list of matching modules. Enter any module name to get more help.
1863
1864''')
1865 apropos(key)
1866 else:
1867 self.output.write('''
1868Please wait a moment while I gather a list of all available modules...
1869
1870''')
1871 modules = {}
1872 def callback(path, modname, desc, modules=modules):
1873 if modname and modname[-9:] == '.__init__':
1874 modname = modname[:-9] + ' (package)'
1875 if find(modname, '.') < 0:
1876 modules[modname] = 1
Ka-Ping Yee9e0f1162008-01-13 11:25:13 +00001877 def onerror(modname):
1878 callback(None, modname, None)
1879 ModuleScanner().run(callback, onerror=onerror)
Ka-Ping Yee35cf0a32001-04-12 19:53:52 +00001880 self.list(modules.keys())
1881 self.output.write('''
1882Enter any module name to get more help. Or, type "modules spam" to search
1883for modules whose descriptions contain the word "spam".
1884''')
1885
1886help = Helper(sys.stdin, sys.stdout)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001887
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001888class Scanner:
1889 """A generic tree iterator."""
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001890 def __init__(self, roots, children, descendp):
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001891 self.roots = roots[:]
1892 self.state = []
1893 self.children = children
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001894 self.descendp = descendp
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001895
1896 def next(self):
1897 if not self.state:
1898 if not self.roots:
1899 return None
1900 root = self.roots.pop(0)
1901 self.state = [(root, self.children(root))]
1902 node, children = self.state[-1]
1903 if not children:
1904 self.state.pop()
1905 return self.next()
1906 child = children.pop(0)
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00001907 if self.descendp(child):
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001908 self.state.append((child, self.children(child)))
1909 return child
1910
Phillip J. Ebyceb30872006-04-18 00:59:55 +00001911
1912class ModuleScanner:
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001913 """An interruptible scanner that searches module synopses."""
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001914
Ka-Ping Yee9e0f1162008-01-13 11:25:13 +00001915 def run(self, callback, key=None, completer=None, onerror=None):
Ka-Ping Yee66246962001-04-12 11:59:50 +00001916 if key: key = lower(key)
Guido van Rossum8ca162f2002-04-07 06:36:23 +00001917 self.quit = False
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001918 seen = {}
1919
1920 for modname in sys.builtin_module_names:
Ka-Ping Yee239432a2001-03-02 02:45:08 +00001921 if modname != '__main__':
1922 seen[modname] = 1
Ka-Ping Yee66246962001-04-12 11:59:50 +00001923 if key is None:
1924 callback(None, modname, '')
1925 else:
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001926 desc = split(__import__(modname).__doc__ or '', '\n')[0]
Ka-Ping Yee66246962001-04-12 11:59:50 +00001927 if find(lower(modname + ' - ' + desc), key) >= 0:
1928 callback(None, modname, desc)
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001929
Ka-Ping Yee9e0f1162008-01-13 11:25:13 +00001930 for importer, modname, ispkg in pkgutil.walk_packages(onerror=onerror):
Phillip J. Ebyceb30872006-04-18 00:59:55 +00001931 if self.quit:
1932 break
1933 if key is None:
1934 callback(None, modname, '')
1935 else:
1936 loader = importer.find_module(modname)
1937 if hasattr(loader,'get_source'):
1938 import StringIO
1939 desc = source_synopsis(
1940 StringIO.StringIO(loader.get_source(modname))
1941 ) or ''
1942 if hasattr(loader,'get_filename'):
1943 path = loader.get_filename(modname)
Ka-Ping Yee66246962001-04-12 11:59:50 +00001944 else:
Phillip J. Ebyceb30872006-04-18 00:59:55 +00001945 path = None
1946 else:
1947 module = loader.load_module(modname)
1948 desc = (module.__doc__ or '').splitlines()[0]
1949 path = getattr(module,'__file__',None)
1950 if find(lower(modname + ' - ' + desc), key) >= 0:
1951 callback(path, modname, desc)
1952
1953 if completer:
1954 completer()
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001955
1956def apropos(key):
1957 """Print all the one-line module summaries that contain a substring."""
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001958 def callback(path, modname, desc):
1959 if modname[-9:] == '.__init__':
1960 modname = modname[:-9] + ' (package)'
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +00001961 print modname, desc and '- ' + desc
1962 try: import warnings
1963 except ImportError: pass
1964 else: warnings.filterwarnings('ignore') # ignore problems during import
Ka-Ping Yee66246962001-04-12 11:59:50 +00001965 ModuleScanner().run(callback, key)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001966
1967# --------------------------------------------------- web browser interface
1968
Ka-Ping Yee66246962001-04-12 11:59:50 +00001969def serve(port, callback=None, completer=None):
Ka-Ping Yeefd540692001-04-12 12:54:36 +00001970 import BaseHTTPServer, mimetools, select
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001971
1972 # Patch up mimetools.Message so it doesn't break if rfc822 is reloaded.
1973 class Message(mimetools.Message):
1974 def __init__(self, fp, seekable=1):
1975 Message = self.__class__
1976 Message.__bases__[0].__bases__[0].__init__(self, fp, seekable)
1977 self.encodingheader = self.getheader('content-transfer-encoding')
1978 self.typeheader = self.getheader('content-type')
1979 self.parsetype()
1980 self.parseplist()
1981
1982 class DocHandler(BaseHTTPServer.BaseHTTPRequestHandler):
1983 def send_document(self, title, contents):
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001984 try:
1985 self.send_response(200)
1986 self.send_header('Content-Type', 'text/html')
1987 self.end_headers()
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001988 self.wfile.write(html.page(title, contents))
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00001989 except IOError: pass
Ka-Ping Yeedd175342001-02-27 14:43:46 +00001990
1991 def do_GET(self):
1992 path = self.path
1993 if path[-5:] == '.html': path = path[:-5]
1994 if path[:1] == '/': path = path[1:]
1995 if path and path != '.':
1996 try:
Ka-Ping Yeedec96e92001-04-13 09:55:49 +00001997 obj = locate(path, forceload=1)
Ka-Ping Yee3bda8792001-03-23 13:17:50 +00001998 except ErrorDuringImport, value:
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00001999 self.send_document(path, html.escape(str(value)))
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002000 return
Ka-Ping Yee3bda8792001-03-23 13:17:50 +00002001 if obj:
2002 self.send_document(describe(obj), html.document(obj, path))
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002003 else:
2004 self.send_document(path,
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00002005'no Python documentation found for %s' % repr(path))
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002006 else:
2007 heading = html.heading(
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002008'<big><big><strong>Python: Index of Modules</strong></big></big>',
2009'#ffffff', '#7799ee')
Ka-Ping Yee239432a2001-03-02 02:45:08 +00002010 def bltinlink(name):
2011 return '<a href="%s.html">%s</a>' % (name, name)
Ka-Ping Yee3bda8792001-03-23 13:17:50 +00002012 names = filter(lambda x: x != '__main__',
2013 sys.builtin_module_names)
Ka-Ping Yee239432a2001-03-02 02:45:08 +00002014 contents = html.multicolumn(names, bltinlink)
2015 indices = ['<p>' + html.bigsection(
2016 'Built-in Modules', '#ffffff', '#ee77aa', contents)]
2017
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002018 seen = {}
Phillip J. Ebyceb30872006-04-18 00:59:55 +00002019 for dir in sys.path:
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002020 indices.append(html.index(dir, seen))
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00002021 contents = heading + join(indices) + '''<p align=right>
Tim Peters2306d242001-09-25 03:18:32 +00002022<font color="#909090" face="helvetica, arial"><strong>
2023pydoc</strong> by Ka-Ping Yee &lt;ping@lfw.org&gt;</font>'''
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002024 self.send_document('Index of Modules', contents)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002025
2026 def log_message(self, *args): pass
2027
Ka-Ping Yeefd540692001-04-12 12:54:36 +00002028 class DocServer(BaseHTTPServer.HTTPServer):
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002029 def __init__(self, port, callback):
Ka-Ping Yeedb8ed152001-03-02 05:58:17 +00002030 host = (sys.platform == 'mac') and '127.0.0.1' or 'localhost'
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00002031 self.address = ('', port)
Ka-Ping Yeedb8ed152001-03-02 05:58:17 +00002032 self.url = 'http://%s:%d/' % (host, port)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002033 self.callback = callback
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002034 self.base.__init__(self, self.address, self.handler)
2035
2036 def serve_until_quit(self):
2037 import select
Guido van Rossum8ca162f2002-04-07 06:36:23 +00002038 self.quit = False
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002039 while not self.quit:
2040 rd, wr, ex = select.select([self.socket.fileno()], [], [], 1)
2041 if rd: self.handle_request()
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002042
2043 def server_activate(self):
2044 self.base.server_activate(self)
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002045 if self.callback: self.callback(self)
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002046
2047 DocServer.base = BaseHTTPServer.HTTPServer
2048 DocServer.handler = DocHandler
2049 DocHandler.MessageClass = Message
2050 try:
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00002051 try:
2052 DocServer(port, callback).serve_until_quit()
2053 except (KeyboardInterrupt, select.error):
2054 pass
2055 finally:
Ka-Ping Yee66246962001-04-12 11:59:50 +00002056 if completer: completer()
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002057
2058# ----------------------------------------------------- graphical interface
2059
2060def gui():
2061 """Graphical interface (starts web server and pops up a control window)."""
2062 class GUI:
2063 def __init__(self, window, port=7464):
2064 self.window = window
2065 self.server = None
2066 self.scanner = None
2067
Georg Brandl6634bf22008-05-20 07:13:37 +00002068 import Tkinter
2069 self.server_frm = Tkinter.Frame(window)
2070 self.title_lbl = Tkinter.Label(self.server_frm,
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002071 text='Starting server...\n ')
Georg Brandl6634bf22008-05-20 07:13:37 +00002072 self.open_btn = Tkinter.Button(self.server_frm,
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002073 text='open browser', command=self.open, state='disabled')
Georg Brandl6634bf22008-05-20 07:13:37 +00002074 self.quit_btn = Tkinter.Button(self.server_frm,
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002075 text='quit serving', command=self.quit, state='disabled')
2076
Georg Brandl6634bf22008-05-20 07:13:37 +00002077 self.search_frm = Tkinter.Frame(window)
2078 self.search_lbl = Tkinter.Label(self.search_frm, text='Search for')
2079 self.search_ent = Tkinter.Entry(self.search_frm)
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002080 self.search_ent.bind('<Return>', self.search)
Georg Brandl6634bf22008-05-20 07:13:37 +00002081 self.stop_btn = Tkinter.Button(self.search_frm,
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002082 text='stop', pady=0, command=self.stop, state='disabled')
2083 if sys.platform == 'win32':
Ka-Ping Yeec92cdf72001-03-02 05:54:35 +00002084 # Trying to hide and show this button crashes under Windows.
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002085 self.stop_btn.pack(side='right')
2086
2087 self.window.title('pydoc')
2088 self.window.protocol('WM_DELETE_WINDOW', self.quit)
2089 self.title_lbl.pack(side='top', fill='x')
2090 self.open_btn.pack(side='left', fill='x', expand=1)
2091 self.quit_btn.pack(side='right', fill='x', expand=1)
2092 self.server_frm.pack(side='top', fill='x')
2093
2094 self.search_lbl.pack(side='left')
2095 self.search_ent.pack(side='right', fill='x', expand=1)
2096 self.search_frm.pack(side='top', fill='x')
2097 self.search_ent.focus_set()
2098
Ka-Ping Yeec92cdf72001-03-02 05:54:35 +00002099 font = ('helvetica', sys.platform == 'win32' and 8 or 10)
Georg Brandl6634bf22008-05-20 07:13:37 +00002100 self.result_lst = Tkinter.Listbox(window, font=font, height=6)
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002101 self.result_lst.bind('<Button-1>', self.select)
2102 self.result_lst.bind('<Double-Button-1>', self.goto)
Georg Brandl6634bf22008-05-20 07:13:37 +00002103 self.result_scr = Tkinter.Scrollbar(window,
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002104 orient='vertical', command=self.result_lst.yview)
2105 self.result_lst.config(yscrollcommand=self.result_scr.set)
2106
Georg Brandl6634bf22008-05-20 07:13:37 +00002107 self.result_frm = Tkinter.Frame(window)
2108 self.goto_btn = Tkinter.Button(self.result_frm,
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002109 text='go to selected', command=self.goto)
Georg Brandl6634bf22008-05-20 07:13:37 +00002110 self.hide_btn = Tkinter.Button(self.result_frm,
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002111 text='hide results', command=self.hide)
2112 self.goto_btn.pack(side='left', fill='x', expand=1)
2113 self.hide_btn.pack(side='right', fill='x', expand=1)
2114
2115 self.window.update()
2116 self.minwidth = self.window.winfo_width()
2117 self.minheight = self.window.winfo_height()
2118 self.bigminheight = (self.server_frm.winfo_reqheight() +
2119 self.search_frm.winfo_reqheight() +
2120 self.result_lst.winfo_reqheight() +
2121 self.result_frm.winfo_reqheight())
2122 self.bigwidth, self.bigheight = self.minwidth, self.bigminheight
2123 self.expanded = 0
2124 self.window.wm_geometry('%dx%d' % (self.minwidth, self.minheight))
2125 self.window.wm_minsize(self.minwidth, self.minheight)
Martin v. Löwis5b26abb2002-12-28 09:23:09 +00002126 self.window.tk.willdispatch()
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002127
2128 import threading
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00002129 threading.Thread(
2130 target=serve, args=(port, self.ready, self.quit)).start()
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002131
2132 def ready(self, server):
2133 self.server = server
2134 self.title_lbl.config(
2135 text='Python documentation server at\n' + server.url)
2136 self.open_btn.config(state='normal')
2137 self.quit_btn.config(state='normal')
2138
Ka-Ping Yee239432a2001-03-02 02:45:08 +00002139 def open(self, event=None, url=None):
2140 url = url or self.server.url
2141 try:
2142 import webbrowser
2143 webbrowser.open(url)
2144 except ImportError: # pre-webbrowser.py compatibility
Ka-Ping Yeec92cdf72001-03-02 05:54:35 +00002145 if sys.platform == 'win32':
Ka-Ping Yee239432a2001-03-02 02:45:08 +00002146 os.system('start "%s"' % url)
2147 elif sys.platform == 'mac':
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +00002148 try: import ic
Ka-Ping Yee239432a2001-03-02 02:45:08 +00002149 except ImportError: pass
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +00002150 else: ic.launchurl(url)
Ka-Ping Yee239432a2001-03-02 02:45:08 +00002151 else:
2152 rc = os.system('netscape -remote "openURL(%s)" &' % url)
2153 if rc: os.system('netscape "%s" &' % url)
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002154
2155 def quit(self, event=None):
2156 if self.server:
2157 self.server.quit = 1
2158 self.window.quit()
2159
2160 def search(self, event=None):
2161 key = self.search_ent.get()
2162 self.stop_btn.pack(side='right')
2163 self.stop_btn.config(state='normal')
2164 self.search_lbl.config(text='Searching for "%s"...' % key)
2165 self.search_ent.forget()
2166 self.search_lbl.pack(side='left')
2167 self.result_lst.delete(0, 'end')
2168 self.goto_btn.config(state='disabled')
2169 self.expand()
2170
2171 import threading
2172 if self.scanner:
2173 self.scanner.quit = 1
2174 self.scanner = ModuleScanner()
2175 threading.Thread(target=self.scanner.run,
Ka-Ping Yee6dcfa382001-04-12 20:27:31 +00002176 args=(self.update, key, self.done)).start()
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002177
2178 def update(self, path, modname, desc):
2179 if modname[-9:] == '.__init__':
2180 modname = modname[:-9] + ' (package)'
2181 self.result_lst.insert('end',
2182 modname + ' - ' + (desc or '(no description)'))
2183
2184 def stop(self, event=None):
2185 if self.scanner:
2186 self.scanner.quit = 1
2187 self.scanner = None
2188
2189 def done(self):
2190 self.scanner = None
2191 self.search_lbl.config(text='Search for')
2192 self.search_lbl.pack(side='left')
2193 self.search_ent.pack(side='right', fill='x', expand=1)
2194 if sys.platform != 'win32': self.stop_btn.forget()
2195 self.stop_btn.config(state='disabled')
2196
2197 def select(self, event=None):
2198 self.goto_btn.config(state='normal')
2199
2200 def goto(self, event=None):
2201 selection = self.result_lst.curselection()
2202 if selection:
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002203 modname = split(self.result_lst.get(selection[0]))[0]
Ka-Ping Yee239432a2001-03-02 02:45:08 +00002204 self.open(url=self.server.url + modname + '.html')
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002205
2206 def collapse(self):
2207 if not self.expanded: return
2208 self.result_frm.forget()
2209 self.result_scr.forget()
2210 self.result_lst.forget()
2211 self.bigwidth = self.window.winfo_width()
2212 self.bigheight = self.window.winfo_height()
2213 self.window.wm_geometry('%dx%d' % (self.minwidth, self.minheight))
2214 self.window.wm_minsize(self.minwidth, self.minheight)
2215 self.expanded = 0
2216
2217 def expand(self):
2218 if self.expanded: return
2219 self.result_frm.pack(side='bottom', fill='x')
2220 self.result_scr.pack(side='right', fill='y')
2221 self.result_lst.pack(side='top', fill='both', expand=1)
2222 self.window.wm_geometry('%dx%d' % (self.bigwidth, self.bigheight))
2223 self.window.wm_minsize(self.minwidth, self.bigminheight)
2224 self.expanded = 1
2225
2226 def hide(self, event=None):
2227 self.stop()
2228 self.collapse()
2229
Georg Brandl6634bf22008-05-20 07:13:37 +00002230 import Tkinter
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002231 try:
Georg Brandl6634bf22008-05-20 07:13:37 +00002232 root = Tkinter.Tk()
Martin v. Löwise09bd932004-08-22 16:13:26 +00002233 # Tk will crash if pythonw.exe has an XP .manifest
2234 # file and the root has is not destroyed explicitly.
2235 # If the problem is ever fixed in Tk, the explicit
2236 # destroy can go.
2237 try:
2238 gui = GUI(root)
2239 root.mainloop()
2240 finally:
2241 root.destroy()
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002242 except KeyboardInterrupt:
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002243 pass
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002244
2245# -------------------------------------------------- command-line interface
2246
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00002247def ispath(x):
Ka-Ping Yee45daeb02002-08-11 15:11:33 +00002248 return isinstance(x, str) and find(x, os.sep) >= 0
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00002249
Ka-Ping Yee1d384632001-03-01 00:24:32 +00002250def cli():
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002251 """Command-line interface (looks at sys.argv to decide what to do)."""
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002252 import getopt
2253 class BadUsage: pass
2254
Nick Coghlan11db64e2009-11-15 22:36:47 +00002255 # Scripts don't get the current directory in their path by default
2256 # unless they are run with the '-m' switch
2257 if '' not in sys.path:
2258 scriptdir = os.path.dirname(sys.argv[0])
2259 if scriptdir in sys.path:
2260 sys.path.remove(scriptdir)
2261 sys.path.insert(0, '.')
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002262
Ka-Ping Yee3bda8792001-03-23 13:17:50 +00002263 try:
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002264 opts, args = getopt.getopt(sys.argv[1:], 'gk:p:w')
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002265 writing = 0
2266
2267 for opt, val in opts:
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002268 if opt == '-g':
2269 gui()
2270 return
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002271 if opt == '-k':
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002272 apropos(val)
2273 return
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002274 if opt == '-p':
2275 try:
2276 port = int(val)
2277 except ValueError:
2278 raise BadUsage
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002279 def ready(server):
Ka-Ping Yee9aa0d902001-04-12 10:50:23 +00002280 print 'pydoc server ready at %s' % server.url
2281 def stopped():
2282 print 'pydoc server stopped'
2283 serve(port, ready, stopped)
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002284 return
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002285 if opt == '-w':
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002286 writing = 1
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002287
2288 if not args: raise BadUsage
2289 for arg in args:
Ka-Ping Yee45daeb02002-08-11 15:11:33 +00002290 if ispath(arg) and not os.path.exists(arg):
2291 print 'file %r does not exist' % arg
2292 break
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002293 try:
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00002294 if ispath(arg) and os.path.isfile(arg):
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002295 arg = importfile(arg)
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00002296 if writing:
2297 if ispath(arg) and os.path.isdir(arg):
2298 writedocs(arg)
2299 else:
2300 writedoc(arg)
2301 else:
Martin v. Löwisb8c084e2003-06-14 09:03:46 +00002302 help.help(arg)
Ka-Ping Yee3bda8792001-03-23 13:17:50 +00002303 except ErrorDuringImport, value:
Ka-Ping Yee37f7b382001-03-23 00:12:53 +00002304 print value
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002305
2306 except (getopt.error, BadUsage):
Martin v. Löwisf9b08b82003-10-31 13:05:21 +00002307 cmd = os.path.basename(sys.argv[0])
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002308 print """pydoc - the Python documentation tool
2309
2310%s <name> ...
2311 Show text documentation on something. <name> may be the name of a
Martin v. Löwisb8c084e2003-06-14 09:03:46 +00002312 Python keyword, topic, function, module, or package, or a dotted
2313 reference to a class or function within a module or module in a
2314 package. If <name> contains a '%s', it is used as the path to a
2315 Python source file to document. If name is 'keywords', 'topics',
2316 or 'modules', a listing of these things is displayed.
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002317
2318%s -k <keyword>
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002319 Search for a keyword in the synopsis lines of all available modules.
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002320
2321%s -p <port>
2322 Start an HTTP server on the given port on the local machine.
2323
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002324%s -g
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +00002325 Pop up a graphical interface for finding and serving documentation.
Ka-Ping Yeedd175342001-02-27 14:43:46 +00002326
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002327%s -w <name> ...
2328 Write out the HTML documentation for a module to a file in the current
Ka-Ping Yee5a804ed2001-04-10 11:46:02 +00002329 directory. If <name> contains a '%s', it is treated as a filename; if
2330 it names a directory, documentation is written for all the contents.
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002331""" % (cmd, os.sep, cmd, cmd, cmd, cmd, os.sep)
Ka-Ping Yee1d384632001-03-01 00:24:32 +00002332
Ka-Ping Yee66efbc72001-03-01 13:55:20 +00002333if __name__ == '__main__': cli()