1997-09-26 19:04:56 -03:00
|
|
|
"""Word completion for GNU readline 2.0.
|
|
|
|
|
2002-03-23 19:44:51 -04:00
|
|
|
This requires the latest extension to the readline module. The completer
|
|
|
|
completes keywords, built-ins and globals in a selectable namespace (which
|
|
|
|
defaults to __main__); when completing NAME.NAME..., it evaluates (!) the
|
|
|
|
expression up to the last dot and completes its attributes.
|
1997-09-26 19:04:56 -03:00
|
|
|
|
Remove uses of the string and types modules:
x in string.whitespace => x.isspace()
type(x) in types.StringTypes => isinstance(x, basestring)
isinstance(x, types.StringTypes) => isinstance(x, basestring)
type(x) is types.StringType => isinstance(x, str)
type(x) == types.StringType => isinstance(x, str)
string.split(x, ...) => x.split(...)
string.join(x, y) => y.join(x)
string.zfill(x, ...) => x.zfill(...)
string.count(x, ...) => x.count(...)
hasattr(types, "UnicodeType") => try: unicode except NameError:
type(x) != types.TupleTuple => not isinstance(x, tuple)
isinstance(x, types.TupleType) => isinstance(x, tuple)
type(x) is types.IntType => isinstance(x, int)
Do not mention the string module in the rlcompleter docstring.
This partially applies SF patch http://www.python.org/sf/562373
(with basestring instead of string). (It excludes the changes to
unittest.py and does not change the os.stat stuff.)
2002-06-03 12:58:32 -03:00
|
|
|
It's very cool to do "import sys" type "sys.", hit the
|
1997-09-26 19:04:56 -03:00
|
|
|
completion key (twice), and see the list of names defined by the
|
Remove uses of the string and types modules:
x in string.whitespace => x.isspace()
type(x) in types.StringTypes => isinstance(x, basestring)
isinstance(x, types.StringTypes) => isinstance(x, basestring)
type(x) is types.StringType => isinstance(x, str)
type(x) == types.StringType => isinstance(x, str)
string.split(x, ...) => x.split(...)
string.join(x, y) => y.join(x)
string.zfill(x, ...) => x.zfill(...)
string.count(x, ...) => x.count(...)
hasattr(types, "UnicodeType") => try: unicode except NameError:
type(x) != types.TupleTuple => not isinstance(x, tuple)
isinstance(x, types.TupleType) => isinstance(x, tuple)
type(x) is types.IntType => isinstance(x, int)
Do not mention the string module in the rlcompleter docstring.
This partially applies SF patch http://www.python.org/sf/562373
(with basestring instead of string). (It excludes the changes to
unittest.py and does not change the os.stat stuff.)
2002-06-03 12:58:32 -03:00
|
|
|
sys module!
|
1997-09-26 19:04:56 -03:00
|
|
|
|
|
|
|
Tip: to use the tab key as the completion key, call
|
|
|
|
|
|
|
|
readline.parse_and_bind("tab: complete")
|
|
|
|
|
|
|
|
Notes:
|
|
|
|
|
|
|
|
- Exceptions raised by the completer function are *ignored* (and
|
|
|
|
generally cause the completion to fail). This is a feature -- since
|
|
|
|
readline sets the tty device in raw (or cbreak) mode, printing a
|
|
|
|
traceback wouldn't work well without some complicated hoopla to save,
|
|
|
|
reset and restore the tty state.
|
|
|
|
|
|
|
|
- The evaluation of the NAME.NAME... form may cause arbitrary
|
|
|
|
application defined code to be executed if an object with a
|
|
|
|
__getattr__ hook is found. Since it is the responsibility of the
|
|
|
|
application (or the user) to enable this feature, I consider this an
|
|
|
|
acceptable risk. More complicated expressions (e.g. function calls or
|
|
|
|
indexing operations) are *not* evaluated.
|
|
|
|
|
|
|
|
- GNU readline is also used by the built-in functions input() and
|
|
|
|
raw_input(), and thus these also benefit/suffer from the completer
|
|
|
|
features. Clearly an interactive application can benefit by
|
|
|
|
specifying its own completer function and using raw_input() for all
|
|
|
|
its input.
|
|
|
|
|
|
|
|
- When the original stdin is not a tty device, GNU readline is never
|
|
|
|
used, and this module (and the readline module) are silently inactive.
|
|
|
|
|
|
|
|
"""
|
|
|
|
|
|
|
|
import __builtin__
|
|
|
|
import __main__
|
|
|
|
|
2001-02-15 18:15:14 -04:00
|
|
|
__all__ = ["Completer"]
|
|
|
|
|
1997-09-26 19:04:56 -03:00
|
|
|
class Completer:
|
2002-03-23 19:44:51 -04:00
|
|
|
def __init__(self, namespace = None):
|
|
|
|
"""Create a new completer for the command line.
|
|
|
|
|
|
|
|
Completer([namespace]) -> completer instance.
|
|
|
|
|
|
|
|
If unspecified, the default namespace where completions are performed
|
|
|
|
is __main__ (technically, __main__.__dict__). Namespaces should be
|
|
|
|
given as dictionaries.
|
|
|
|
|
|
|
|
Completer instances should be used as the completion mechanism of
|
|
|
|
readline via the set_completer() call:
|
|
|
|
|
|
|
|
readline.set_completer(Completer(my_namespace).complete)
|
|
|
|
"""
|
2002-04-15 22:38:40 -03:00
|
|
|
|
2002-03-23 19:44:51 -04:00
|
|
|
if namespace and not isinstance(namespace, dict):
|
|
|
|
raise TypeError,'namespace must be a dictionary'
|
|
|
|
|
|
|
|
# Don't bind to namespace quite yet, but flag whether the user wants a
|
|
|
|
# specific namespace or to use __main__.__dict__. This will allow us
|
|
|
|
# to bind to __main__.__dict__ at completion time, not now.
|
|
|
|
if namespace is None:
|
|
|
|
self.use_main_ns = 1
|
|
|
|
else:
|
|
|
|
self.use_main_ns = 0
|
|
|
|
self.namespace = namespace
|
1997-09-26 19:04:56 -03:00
|
|
|
|
|
|
|
def complete(self, text, state):
|
1998-03-26 17:13:24 -04:00
|
|
|
"""Return the next possible completion for 'text'.
|
1997-09-26 19:04:56 -03:00
|
|
|
|
1998-03-26 17:13:24 -04:00
|
|
|
This is called successively with state == 0, 1, 2, ... until it
|
|
|
|
returns None. The completion should begin with 'text'.
|
1997-09-26 19:04:56 -03:00
|
|
|
|
1998-03-26 17:13:24 -04:00
|
|
|
"""
|
2002-03-23 19:44:51 -04:00
|
|
|
if self.use_main_ns:
|
|
|
|
self.namespace = __main__.__dict__
|
2002-04-15 22:38:40 -03:00
|
|
|
|
1998-03-26 17:13:24 -04:00
|
|
|
if state == 0:
|
|
|
|
if "." in text:
|
|
|
|
self.matches = self.attr_matches(text)
|
|
|
|
else:
|
|
|
|
self.matches = self.global_matches(text)
|
1998-06-12 16:42:14 -03:00
|
|
|
try:
|
|
|
|
return self.matches[state]
|
|
|
|
except IndexError:
|
|
|
|
return None
|
1997-09-26 19:04:56 -03:00
|
|
|
|
2008-07-02 13:52:55 -03:00
|
|
|
def _callable_postfix(self, val, word):
|
|
|
|
if callable(val):
|
|
|
|
word = word + "("
|
|
|
|
return word
|
|
|
|
|
1997-09-26 19:04:56 -03:00
|
|
|
def global_matches(self, text):
|
1998-03-26 17:13:24 -04:00
|
|
|
"""Compute matches when text is a simple name.
|
|
|
|
|
2002-03-23 19:44:51 -04:00
|
|
|
Return a list of all keywords, built-in functions and names currently
|
|
|
|
defined in self.namespace that match.
|
1998-03-26 17:13:24 -04:00
|
|
|
|
|
|
|
"""
|
|
|
|
import keyword
|
|
|
|
matches = []
|
|
|
|
n = len(text)
|
2008-07-02 13:52:55 -03:00
|
|
|
for word in keyword.kwlist:
|
|
|
|
if word[:n] == text:
|
|
|
|
matches.append(word)
|
|
|
|
for nspace in [__builtin__.__dict__, self.namespace]:
|
|
|
|
for word, val in nspace.items():
|
2000-05-31 11:31:00 -03:00
|
|
|
if word[:n] == text and word != "__builtins__":
|
2008-07-02 13:52:55 -03:00
|
|
|
matches.append(self._callable_postfix(val, word))
|
1998-03-26 17:13:24 -04:00
|
|
|
return matches
|
1997-09-26 19:04:56 -03:00
|
|
|
|
|
|
|
def attr_matches(self, text):
|
1998-03-26 17:13:24 -04:00
|
|
|
"""Compute matches when text contains a dot.
|
|
|
|
|
|
|
|
Assuming the text is of the form NAME.NAME....[NAME], and is
|
2002-03-23 19:44:51 -04:00
|
|
|
evaluatable in self.namespace, it will be evaluated and its attributes
|
|
|
|
(as revealed by dir()) are used as possible completions. (For class
|
2003-10-20 11:01:56 -03:00
|
|
|
instances, class members are also considered.)
|
1998-03-26 17:13:24 -04:00
|
|
|
|
|
|
|
WARNING: this can still invoke arbitrary C code, if an object
|
|
|
|
with a __getattr__ hook is evaluated.
|
|
|
|
|
|
|
|
"""
|
|
|
|
import re
|
|
|
|
m = re.match(r"(\w+(\.\w+)*)\.(\w*)", text)
|
|
|
|
if not m:
|
2008-03-06 03:43:02 -04:00
|
|
|
return []
|
1998-03-26 17:13:24 -04:00
|
|
|
expr, attr = m.group(1, 3)
|
2008-05-11 18:03:42 -03:00
|
|
|
try:
|
2008-07-21 11:28:17 -03:00
|
|
|
thisobject = eval(expr, self.namespace)
|
2008-05-11 18:03:42 -03:00
|
|
|
except Exception:
|
|
|
|
return []
|
2008-07-21 11:28:17 -03:00
|
|
|
|
|
|
|
# get the content of the object, except __builtins__
|
|
|
|
words = dir(thisobject)
|
|
|
|
if "__builtins__" in words:
|
|
|
|
words.remove("__builtins__")
|
|
|
|
|
|
|
|
if hasattr(thisobject, '__class__'):
|
1999-10-26 10:09:08 -03:00
|
|
|
words.append('__class__')
|
2008-07-21 11:28:17 -03:00
|
|
|
words.extend(get_class_members(thisobject.__class__))
|
1998-03-26 17:13:24 -04:00
|
|
|
matches = []
|
|
|
|
n = len(attr)
|
|
|
|
for word in words:
|
2008-07-21 11:28:17 -03:00
|
|
|
if word[:n] == attr and hasattr(thisobject, word):
|
|
|
|
val = getattr(thisobject, word)
|
2008-07-02 13:52:55 -03:00
|
|
|
word = self._callable_postfix(val, "%s.%s" % (expr, word))
|
|
|
|
matches.append(word)
|
1998-03-26 17:13:24 -04:00
|
|
|
return matches
|
1997-09-26 19:04:56 -03:00
|
|
|
|
1999-10-26 10:09:08 -03:00
|
|
|
def get_class_members(klass):
|
|
|
|
ret = dir(klass)
|
|
|
|
if hasattr(klass,'__bases__'):
|
|
|
|
for base in klass.__bases__:
|
|
|
|
ret = ret + get_class_members(base)
|
|
|
|
return ret
|
|
|
|
|
2006-04-30 15:14:54 -03:00
|
|
|
try:
|
|
|
|
import readline
|
|
|
|
except ImportError:
|
|
|
|
pass
|
|
|
|
else:
|
|
|
|
readline.set_completer(Completer().complete)
|