##// END OF EJS Templates
SVN completer
SVN completer

File last commit:

r390:94d39235
r397:684683b2
Show More
OInspect.py
548 lines | 18.9 KiB | text/x-python | PythonLexer
fperez
Reorganized the directory for ipython/ to have its own dir, which is a bit...
r0 # -*- coding: utf-8 -*-
"""Tools for inspecting Python objects.
Uses syntax highlighting for presenting the various information elements.
Similar in spirit to the inspect module, but all calls take a name argument to
reference the name under which an object is being read.
fptest
- Work around pexcept buglet which causes wraparound problems with long...
r390 $Id: OInspect.py 1850 2006-10-28 19:48:13Z fptest $
fperez
Reorganized the directory for ipython/ to have its own dir, which is a bit...
r0 """
#*****************************************************************************
# Copyright (C) 2001-2004 Fernando Perez <fperez@colorado.edu>
#
# Distributed under the terms of the BSD License. The full license is in
# the file COPYING, distributed as part of this software.
#*****************************************************************************
from IPython import Release
__author__ = '%s <%s>' % Release.authors['Fernando']
__license__ = Release.license
__all__ = ['Inspector','InspectColors']
# stdlib modules
fperez
Wildcard system cleanup, ipmaker speedups, bugfix in globals handling...
r41 import __builtin__
fperez
Cosmetic cleanups: put all imports in a single line, and sort them...
r52 import inspect
import linecache
import string
import StringIO
import types
vivainio
Oinspect.py: Only show docstring for nonexisting/binary files...
r270 import os
vivainio
revert fakemodule and oinspect mods, they broke unittest. monkeypatch is back
r363 import sys
fperez
Reorganized the directory for ipython/ to have its own dir, which is a bit...
r0 # IPython's own
fperez
New wildcard support. Lightly tested, so proceed with caution. We need to...
r37 from IPython import PyColorize
fperez
Cosmetic cleanups: put all imports in a single line, and sort them...
r52 from IPython.genutils import page,indent,Term,mkdict
fperez
Reorganized the directory for ipython/ to have its own dir, which is a bit...
r0 from IPython.Itpl import itpl
fperez
Wildcard system cleanup, ipmaker speedups, bugfix in globals handling...
r41 from IPython.wildcard import list_namespace
fperez
Reorganized the directory for ipython/ to have its own dir, which is a bit...
r0 from IPython.ColorANSI import *
#****************************************************************************
vivainio
revert fakemodule and oinspect mods, they broke unittest. monkeypatch is back
r363 # HACK!!! This is a crude fix for bugs in python 2.3's inspect module. We
# simply monkeypatch inspect with code copied from python 2.4.
if sys.version_info[:2] == (2,3):
from inspect import ismodule, getabsfile, modulesbyfile
def getmodule(object):
"""Return the module an object was defined in, or None if not found."""
if ismodule(object):
return object
if hasattr(object, '__module__'):
return sys.modules.get(object.__module__)
try:
file = getabsfile(object)
except TypeError:
return None
if file in modulesbyfile:
return sys.modules.get(modulesbyfile[file])
for module in sys.modules.values():
if hasattr(module, '__file__'):
modulesbyfile[
os.path.realpath(
getabsfile(module))] = module.__name__
if file in modulesbyfile:
return sys.modules.get(modulesbyfile[file])
main = sys.modules['__main__']
if not hasattr(object, '__name__'):
return None
if hasattr(main, object.__name__):
mainobject = getattr(main, object.__name__)
if mainobject is object:
return main
builtin = sys.modules['__builtin__']
if hasattr(builtin, object.__name__):
builtinobject = getattr(builtin, object.__name__)
if builtinobject is object:
return builtin
inspect.getmodule = getmodule
#****************************************************************************
fperez
Reorganized the directory for ipython/ to have its own dir, which is a bit...
r0 # Builtin color schemes
Colors = TermColors # just a shorthand
# Build a few color schemes
NoColor = ColorScheme(
'NoColor',{
'header' : Colors.NoColor,
'normal' : Colors.NoColor # color off (usu. Colors.Normal)
} )
LinuxColors = ColorScheme(
'Linux',{
'header' : Colors.LightRed,
'normal' : Colors.Normal # color off (usu. Colors.Normal)
} )
LightBGColors = ColorScheme(
'LightBG',{
'header' : Colors.Red,
'normal' : Colors.Normal # color off (usu. Colors.Normal)
} )
# Build table of color schemes (needed by the parser)
InspectColors = ColorSchemeTable([NoColor,LinuxColors,LightBGColors],
'Linux')
#****************************************************************************
# Auxiliary functions
def getdoc(obj):
"""Stable wrapper around inspect.getdoc.
This can't crash because of attribute problems.
It also attempts to call a getdoc() method on the given object. This
allows objects which provide their docstrings via non-standard mechanisms
(like Pyro proxies) to still be inspected by ipython's ? system."""
ds = None # default return value
try:
ds = inspect.getdoc(obj)
except:
# Harden against an inspect failure, which can occur with
# SWIG-wrapped extensions.
pass
# Allow objects to offer customized documentation via a getdoc method:
try:
ds2 = obj.getdoc()
except:
pass
else:
# if we get extra info, we add it to the normal docstring.
if ds is None:
ds = ds2
else:
ds = '%s\n%s' % (ds,ds2)
return ds
fptest
- Work around pexcept buglet which causes wraparound problems with long...
r390 def getsource(obj,is_binary=False):
"""Wrapper around inspect.getsource.
This can be modified by other projects to provide customized source
extraction.
Inputs:
- obj: an object whose source code we will attempt to extract.
Optional inputs:
- is_binary: whether the object is known to come from a binary source.
This implementation will skip returning any output for binary objects, but
custom extractors may know how to meaninfully process them."""
if is_binary:
return None
else:
return inspect.getsource(obj)
fperez
Reorganized the directory for ipython/ to have its own dir, which is a bit...
r0 #****************************************************************************
# Class definitions
class myStringIO(StringIO.StringIO):
"""Adds a writeln method to normal StringIO."""
def writeln(self,*arg,**kw):
"""Does a write() and then a write('\n')"""
self.write(*arg,**kw)
self.write('\n')
class Inspector:
fperez
- fix bug where aliases would shadow variables when autocall was fully off....
r299 def __init__(self,color_table,code_color_table,scheme,
str_detail_level=0):
fperez
Reorganized the directory for ipython/ to have its own dir, which is a bit...
r0 self.color_table = color_table
self.parser = PyColorize.Parser(code_color_table,out='str')
self.format = self.parser.format
fperez
- fix bug where aliases would shadow variables when autocall was fully off....
r299 self.str_detail_level = str_detail_level
fperez
Reorganized the directory for ipython/ to have its own dir, which is a bit...
r0 self.set_active_scheme(scheme)
def __getargspec(self,obj):
"""Get the names and default values of a function's arguments.
A tuple of four things is returned: (args, varargs, varkw, defaults).
'args' is a list of the argument names (it may contain nested lists).
'varargs' and 'varkw' are the names of the * and ** arguments or None.
'defaults' is an n-tuple of the default values of the last n arguments.
Modified version of inspect.getargspec from the Python Standard
Library."""
if inspect.isfunction(obj):
func_obj = obj
elif inspect.ismethod(obj):
func_obj = obj.im_func
else:
raise TypeError, 'arg is not a Python function'
args, varargs, varkw = inspect.getargs(func_obj.func_code)
return args, varargs, varkw, func_obj.func_defaults
def __getdef(self,obj,oname=''):
"""Return the definition header for any callable object.
If any exception is generated, None is returned instead and the
exception is suppressed."""
try:
return oname + inspect.formatargspec(*self.__getargspec(obj))
except:
return None
def __head(self,h):
"""Return a header string with proper colors."""
return '%s%s%s' % (self.color_table.active_colors.header,h,
self.color_table.active_colors.normal)
def set_active_scheme(self,scheme):
self.color_table.set_active_scheme(scheme)
self.parser.color_table.set_active_scheme(scheme)
def noinfo(self,msg,oname):
"""Generic message when no information is found."""
print 'No %s found' % msg,
if oname:
print 'for %s' % oname
else:
print
def pdef(self,obj,oname=''):
"""Print the definition header for any callable object.
If the object is a class, print the constructor information."""
if not callable(obj):
print 'Object is not callable.'
return
header = ''
if type(obj) is types.ClassType:
header = self.__head('Class constructor information:\n')
obj = obj.__init__
elif type(obj) is types.InstanceType:
obj = obj.__call__
output = self.__getdef(obj,oname)
if output is None:
self.noinfo('definition header',oname)
else:
print >>Term.cout, header,self.format(output),
def pdoc(self,obj,oname='',formatter = None):
"""Print the docstring for any object.
Optional:
-formatter: a function to run the docstring through for specially
formatted docstrings."""
head = self.__head # so that itpl can find it even if private
ds = getdoc(obj)
if formatter:
ds = formatter(ds)
if type(obj) is types.ClassType:
init_ds = getdoc(obj.__init__)
output = itpl('$head("Class Docstring:")\n'
'$indent(ds)\n'
'$head("Constructor Docstring"):\n'
'$indent(init_ds)')
elif type(obj) is types.InstanceType and hasattr(obj,'__call__'):
call_ds = getdoc(obj.__call__)
if call_ds:
output = itpl('$head("Class Docstring:")\n$indent(ds)\n'
'$head("Calling Docstring:")\n$indent(call_ds)')
else:
output = ds
else:
output = ds
if output is None:
self.noinfo('documentation',oname)
return
page(output)
def psource(self,obj,oname=''):
"""Print the source code for an object."""
# Flush the source cache because inspect can return out-of-date source
linecache.checkcache()
try:
fptest
- Work around pexcept buglet which causes wraparound problems with long...
r390 src = getsource(obj)
fperez
Reorganized the directory for ipython/ to have its own dir, which is a bit...
r0 except:
self.noinfo('source',oname)
else:
page(self.format(src))
def pfile(self,obj,oname=''):
"""Show the whole file where an object was defined."""
try:
sourcelines,lineno = inspect.getsourcelines(obj)
except:
self.noinfo('file',oname)
else:
# run contents of file through pager starting at line
vivainio
Merge from branches/0.7.1 into trunk, revs 1052-1057
r136 # where the object is defined
ofile = inspect.getabsfile(obj)
if (ofile.endswith('.so') or ofile.endswith('.dll')):
print 'File %r is binary, not printing.' % ofile
vivainio
Oinspect.py: Only show docstring for nonexisting/binary files...
r270 elif not os.path.isfile(ofile):
print 'File %r does not exist, not printing.' % ofile
vivainio
Merge from branches/0.7.1 into trunk, revs 1052-1057
r136 else:
# Print only text files, not extension binaries.
page(self.format(open(ofile).read()),lineno)
#page(self.format(open(inspect.getabsfile(obj)).read()),lineno)
fperez
Reorganized the directory for ipython/ to have its own dir, which is a bit...
r0
def pinfo(self,obj,oname='',formatter=None,info=None,detail_level=0):
"""Show detailed information about an object.
Optional arguments:
- oname: name of the variable pointing to the object.
- formatter: special formatter for docstrings (see pdoc)
- info: a structure with some information fields which may have been
precomputed already.
- detail_level: if set to 1, more information is given.
"""
obj_type = type(obj)
header = self.__head
if info is None:
ismagic = 0
isalias = 0
ospace = ''
else:
ismagic = info.ismagic
isalias = info.isalias
ospace = info.namespace
# Get docstring, special-casing aliases:
if isalias:
ds = "Alias to the system command:\n %s" % obj[1]
else:
ds = getdoc(obj)
vivainio
Added ipapi, the extension api for ipython....
r109 if ds is None:
ds = '<no docstring>'
fperez
Reorganized the directory for ipython/ to have its own dir, which is a bit...
r0 if formatter is not None:
ds = formatter(ds)
# store output in a list which gets joined with \n at the end.
out = myStringIO()
string_max = 200 # max size of strings to show (snipped if longer)
shalf = int((string_max -5)/2)
if ismagic:
obj_type_name = 'Magic function'
elif isalias:
obj_type_name = 'System alias'
else:
obj_type_name = obj_type.__name__
out.writeln(header('Type:\t\t')+obj_type_name)
try:
bclass = obj.__class__
out.writeln(header('Base Class:\t')+str(bclass))
except: pass
# String form, but snip if too long in ? form (full in ??)
fperez
- fix bug where aliases would shadow variables when autocall was fully off....
r299 if detail_level >= self.str_detail_level:
try:
ostr = str(obj)
str_head = 'String Form:'
if not detail_level and len(ostr)>string_max:
ostr = ostr[:shalf] + ' <...> ' + ostr[-shalf:]
ostr = ("\n" + " " * len(str_head.expandtabs())).\
join(map(string.strip,ostr.split("\n")))
if ostr.find('\n') > -1:
# Print multi-line strings starting at the next line.
str_sep = '\n'
else:
str_sep = '\t'
out.writeln("%s%s%s" % (header(str_head),str_sep,ostr))
except:
pass
fperez
Reorganized the directory for ipython/ to have its own dir, which is a bit...
r0
if ospace:
out.writeln(header('Namespace:\t')+ospace)
# Length (for strings and lists)
try:
length = str(len(obj))
out.writeln(header('Length:\t\t')+length)
except: pass
# Filename where object was defined
vivainio
Merge from branches/0.7.1 into trunk, revs 1052-1057
r136 binary_file = False
fperez
Reorganized the directory for ipython/ to have its own dir, which is a bit...
r0 try:
vivainio
Merge from branches/0.7.1 into trunk, revs 1052-1057
r136 fname = inspect.getabsfile(obj)
if fname.endswith('<string>'):
fname = 'Dynamically generated function. No source code available.'
vivainio
Oinspect.py: Only show docstring for nonexisting/binary files...
r270 if (fname.endswith('.so') or fname.endswith('.dll') or
not os.path.isfile(fname)):
vivainio
Merge from branches/0.7.1 into trunk, revs 1052-1057
r136 binary_file = True
out.writeln(header('File:\t\t')+fname)
except:
# if anything goes wrong, we don't want to show source, so it's as
# if the file was binary
binary_file = True
fperez
Reorganized the directory for ipython/ to have its own dir, which is a bit...
r0
# reconstruct the function definition and print it:
defln = self.__getdef(obj,oname)
if defln:
out.write(header('Definition:\t')+self.format(defln))
# Docstrings only in detail 0 mode, since source contains them (we
# avoid repetitions). If source fails, we add them back, see below.
if ds and detail_level == 0:
out.writeln(header('Docstring:\n') + indent(ds))
vivainio
Oinspect.py: Only show docstring for nonexisting/binary files...
r270
fperez
Reorganized the directory for ipython/ to have its own dir, which is a bit...
r0 # Original source code for any callable
if detail_level:
# Flush the source cache because inspect can return out-of-date source
linecache.checkcache()
vivainio
Oinspect.py: Only show docstring for nonexisting/binary files...
r270 source_success = False
fperez
Reorganized the directory for ipython/ to have its own dir, which is a bit...
r0 try:
fptest
- Work around pexcept buglet which causes wraparound problems with long...
r390 source = self.format(getsource(obj,binary_file))
if source:
vivainio
Merge from branches/0.7.1 into trunk, revs 1052-1057
r136 out.write(header('Source:\n')+source.rstrip())
vivainio
Oinspect.py: Only show docstring for nonexisting/binary files...
r270 source_success = True
fptest
- Work around pexcept buglet which causes wraparound problems with long...
r390 except Exception, msg:
vivainio
Oinspect.py: Only show docstring for nonexisting/binary files...
r270 pass
if ds and not source_success:
fptest
- Work around pexcept buglet which causes wraparound problems with long...
r390 out.writeln(header('Docstring [source file open failed]:\n')
+ indent(ds))
fperez
Reorganized the directory for ipython/ to have its own dir, which is a bit...
r0
# Constructor docstring for classes
if obj_type is types.ClassType:
# reconstruct the function definition and print it:
try:
obj_init = obj.__init__
except AttributeError:
init_def = init_ds = None
else:
init_def = self.__getdef(obj_init,oname)
init_ds = getdoc(obj_init)
if init_def or init_ds:
out.writeln(header('\nConstructor information:'))
if init_def:
out.write(header('Definition:\t')+ self.format(init_def))
if init_ds:
out.writeln(header('Docstring:\n') + indent(init_ds))
# and class docstring for instances:
elif obj_type is types.InstanceType:
# First, check whether the instance docstring is identical to the
# class one, and print it separately if they don't coincide. In
# most cases they will, but it's nice to print all the info for
# objects which use instance-customized docstrings.
if ds:
class_ds = getdoc(obj.__class__)
if class_ds and ds != class_ds:
out.writeln(header('Class Docstring:\n') +
indent(class_ds))
# Next, try to show constructor docstrings
try:
init_ds = getdoc(obj.__init__)
except AttributeError:
init_ds = None
if init_ds:
out.writeln(header('Constructor Docstring:\n') +
indent(init_ds))
# Call form docstring for callable instances
if hasattr(obj,'__call__'):
out.writeln(header('Callable:\t')+'Yes')
call_def = self.__getdef(obj.__call__,oname)
if call_def is None:
out.write(header('Call def:\t')+
'Calling definition not available.')
else:
out.write(header('Call def:\t')+self.format(call_def))
call_ds = getdoc(obj.__call__)
if call_ds:
out.writeln(header('Call docstring:\n') + indent(call_ds))
# Finally send to printer/pager
output = out.getvalue()
if output:
page(output)
# end pinfo
fperez
New wildcard support. Lightly tested, so proceed with caution. We need to...
r37
fperez
Wildcard system cleanup, ipmaker speedups, bugfix in globals handling...
r41 def psearch(self,pattern,ns_table,ns_search=[],
ignore_case=False,show_all=False):
fperez
wildcard fixes for subclasses
r40 """Search namespaces with wildcards for objects.
fperez
New wildcard support. Lightly tested, so proceed with caution. We need to...
r37
fperez
Wildcard system cleanup, ipmaker speedups, bugfix in globals handling...
r41 Arguments:
- pattern: string containing shell-like wildcards to use in namespace
searches and optionally a type specification to narrow the search to
objects of that type.
- ns_table: dict of name->namespaces for search.
fperez
New wildcard support. Lightly tested, so proceed with caution. We need to...
r37 Optional arguments:
fperez
Wildcard system cleanup, ipmaker speedups, bugfix in globals handling...
r41 - ns_search: list of namespace names to include in search.
fperez
wildcard fixes for subclasses
r40
fperez
Wildcard system cleanup, ipmaker speedups, bugfix in globals handling...
r41 - ignore_case(False): make the search case-insensitive.
fperez
wildcard fixes for subclasses
r40
fperez
Wildcard system cleanup, ipmaker speedups, bugfix in globals handling...
r41 - show_all(False): show all names, including those starting with
underscores.
fperez
wildcard fixes for subclasses
r40 """
fperez
Wildcard system cleanup, ipmaker speedups, bugfix in globals handling...
r41 # defaults
fperez
wildcard fixes for subclasses
r40 type_pattern = 'all'
fperez
Wildcard system cleanup, ipmaker speedups, bugfix in globals handling...
r41 filter = ''
cmds = pattern.split()
fperez
wildcard fixes for subclasses
r40 len_cmds = len(cmds)
if len_cmds == 1:
fperez
Wildcard system cleanup, ipmaker speedups, bugfix in globals handling...
r41 # Only filter pattern given
filter = cmds[0]
fperez
wildcard fixes for subclasses
r40 elif len_cmds == 2:
fperez
Wildcard system cleanup, ipmaker speedups, bugfix in globals handling...
r41 # Both filter and type specified
fperez
wildcard fixes for subclasses
r40 filter,type_pattern = cmds
fperez
Wildcard system cleanup, ipmaker speedups, bugfix in globals handling...
r41 else:
raise ValueError('invalid argument string for psearch: <%s>' %
pattern)
fperez
wildcard fixes for subclasses
r40
fperez
Wildcard system cleanup, ipmaker speedups, bugfix in globals handling...
r41 # filter search namespaces
for name in ns_search:
if name not in ns_table:
raise ValueError('invalid namespace <%s>. Valid names: %s' %
(name,ns_table.keys()))
fperez
wildcard fixes for subclasses
r40
fperez
Wildcard system cleanup, ipmaker speedups, bugfix in globals handling...
r41 #print 'type_pattern:',type_pattern # dbg
fperez
wildcard fixes for subclasses
r40 search_result = []
fperez
Wildcard system cleanup, ipmaker speedups, bugfix in globals handling...
r41 for ns_name in ns_search:
ns = ns_table[ns_name]
fperez
wildcard fixes for subclasses
r40 tmp_res = list(list_namespace(ns,type_pattern,filter,
fperez
Wildcard system cleanup, ipmaker speedups, bugfix in globals handling...
r41 ignore_case=ignore_case,
show_all=show_all))
fperez
wildcard fixes for subclasses
r40 search_result.extend(tmp_res)
search_result.sort()
page('\n'.join(search_result))