| 
									
										
										
										
											1997-09-26 22:04:56 +00:00
										 |  |  | """Word completion for GNU readline 2.0.
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-03-23 23:44:51 +00: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 22:04:56 +00: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 15:58:32 +00:00
										 |  |  | It's very cool to do "import sys" type "sys.", hit the | 
					
						
							| 
									
										
										
										
											1997-09-26 22:04:56 +00: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 15:58:32 +00:00
										 |  |  | sys module! | 
					
						
							| 
									
										
										
										
											1997-09-26 22:04:56 +00: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 readline | 
					
						
							|  |  |  | import __builtin__ | 
					
						
							|  |  |  | import __main__ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-02-15 22:15:14 +00:00
										 |  |  | __all__ = ["Completer"] | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1997-09-26 22:04:56 +00:00
										 |  |  | class Completer: | 
					
						
							| 
									
										
										
										
											2002-03-23 23:44:51 +00: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-16 01:38:40 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-03-23 23:44:51 +00: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 22:04:56 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def complete(self, text, state): | 
					
						
							| 
									
										
										
										
											1998-03-26 21:13:24 +00:00
										 |  |  |         """Return the next possible completion for 'text'.
 | 
					
						
							| 
									
										
										
										
											1997-09-26 22:04:56 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1998-03-26 21:13:24 +00:00
										 |  |  |         This is called successively with state == 0, 1, 2, ... until it | 
					
						
							|  |  |  |         returns None.  The completion should begin with 'text'. | 
					
						
							| 
									
										
										
										
											1997-09-26 22:04:56 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1998-03-26 21:13:24 +00:00
										 |  |  |         """
 | 
					
						
							| 
									
										
										
										
											2002-03-23 23:44:51 +00:00
										 |  |  |         if self.use_main_ns: | 
					
						
							|  |  |  |             self.namespace = __main__.__dict__ | 
					
						
							| 
									
										
										
										
											2002-04-16 01:38:40 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1998-03-26 21:13:24 +00:00
										 |  |  |         if state == 0: | 
					
						
							|  |  |  |             if "." in text: | 
					
						
							|  |  |  |                 self.matches = self.attr_matches(text) | 
					
						
							|  |  |  |             else: | 
					
						
							|  |  |  |                 self.matches = self.global_matches(text) | 
					
						
							| 
									
										
										
										
											1998-06-12 19:42:14 +00:00
										 |  |  |         try: | 
					
						
							|  |  |  |             return self.matches[state] | 
					
						
							|  |  |  |         except IndexError: | 
					
						
							|  |  |  |             return None | 
					
						
							| 
									
										
										
										
											1997-09-26 22:04:56 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def global_matches(self, text): | 
					
						
							| 
									
										
										
										
											1998-03-26 21:13:24 +00:00
										 |  |  |         """Compute matches when text is a simple name.
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-03-23 23:44:51 +00:00
										 |  |  |         Return a list of all keywords, built-in functions and names currently | 
					
						
							|  |  |  |         defined in self.namespace that match. | 
					
						
							| 
									
										
										
										
											1998-03-26 21:13:24 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         import keyword | 
					
						
							|  |  |  |         matches = [] | 
					
						
							|  |  |  |         n = len(text) | 
					
						
							|  |  |  |         for list in [keyword.kwlist, | 
					
						
							| 
									
										
										
										
											2002-06-02 18:55:56 +00:00
										 |  |  |                      __builtin__.__dict__, | 
					
						
							|  |  |  |                      self.namespace]: | 
					
						
							| 
									
										
										
										
											1998-03-26 21:13:24 +00:00
										 |  |  |             for word in list: | 
					
						
							| 
									
										
										
										
											2000-05-31 14:31:00 +00:00
										 |  |  |                 if word[:n] == text and word != "__builtins__": | 
					
						
							| 
									
										
										
										
											1998-03-26 21:13:24 +00:00
										 |  |  |                     matches.append(word) | 
					
						
							|  |  |  |         return matches | 
					
						
							| 
									
										
										
										
											1997-09-26 22:04:56 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def attr_matches(self, text): | 
					
						
							| 
									
										
										
										
											1998-03-26 21:13:24 +00:00
										 |  |  |         """Compute matches when text contains a dot.
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         Assuming the text is of the form NAME.NAME....[NAME], and is | 
					
						
							| 
									
										
										
										
											2002-03-23 23:44:51 +00: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 14:01:56 +00:00
										 |  |  |         instances, class members are also considered.) | 
					
						
							| 
									
										
										
										
											1998-03-26 21:13:24 +00: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: | 
					
						
							|  |  |  |             return | 
					
						
							|  |  |  |         expr, attr = m.group(1, 3) | 
					
						
							| 
									
										
										
										
											2002-03-23 23:44:51 +00:00
										 |  |  |         object = eval(expr, self.namespace) | 
					
						
							| 
									
										
										
										
											1999-10-26 13:09:08 +00:00
										 |  |  |         words = dir(object) | 
					
						
							|  |  |  |         if hasattr(object,'__class__'): | 
					
						
							|  |  |  |             words.append('__class__') | 
					
						
							|  |  |  |             words = words + get_class_members(object.__class__) | 
					
						
							| 
									
										
										
										
											1998-03-26 21:13:24 +00:00
										 |  |  |         matches = [] | 
					
						
							|  |  |  |         n = len(attr) | 
					
						
							|  |  |  |         for word in words: | 
					
						
							| 
									
										
										
										
											2000-05-31 14:31:00 +00:00
										 |  |  |             if word[:n] == attr and word != "__builtins__": | 
					
						
							| 
									
										
										
										
											1998-03-26 21:13:24 +00:00
										 |  |  |                 matches.append("%s.%s" % (expr, word)) | 
					
						
							|  |  |  |         return matches | 
					
						
							| 
									
										
										
										
											1997-09-26 22:04:56 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1999-10-26 13:09:08 +00: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 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1997-09-26 22:04:56 +00:00
										 |  |  | readline.set_completer(Completer().complete) |