]> granicus.if.org Git - python/commitdiff
Issue #25590: Make rlcompleter only call getattr() once per attribute
authorMartin Panter <vadmium+py@gmail.com>
Fri, 13 Nov 2015 22:47:00 +0000 (22:47 +0000)
committerMartin Panter <vadmium+py@gmail.com>
Fri, 13 Nov 2015 22:47:00 +0000 (22:47 +0000)
Previously it was called another time via hasattr(), and both calls were
made once for dir(f) and again for dir(f.__class__).  This includes a
backport of changing from a list to a set from revision 4dbb315fe667.

Lib/rlcompleter.py
Lib/test/test_rlcompleter.py
Misc/NEWS

index d517c0e2d39d24270a36c89ae5603d33b5c80dc4..be8aee0f7e904f336e01bb294cbeed9cef07071a 100644 (file)
@@ -136,20 +136,23 @@ class Completer:
             return []
 
         # get the content of the object, except __builtins__
-        words = dir(thisobject)
-        if "__builtins__" in words:
-            words.remove("__builtins__")
+        words = set(dir(thisobject))
+        words.discard("__builtins__")
 
         if hasattr(thisobject, '__class__'):
-            words.append('__class__')
-            words.extend(get_class_members(thisobject.__class__))
+            words.add('__class__')
+            words.update(get_class_members(thisobject.__class__))
         matches = []
         n = len(attr)
         for word in words:
-            if word[:n] == attr and hasattr(thisobject, word):
-                val = getattr(thisobject, word)
+            if word[:n] == attr:
+                try:
+                    val = getattr(thisobject, word)
+                except Exception:
+                    continue  # Exclude properties that are not set
                 word = self._callable_postfix(val, "%s.%s" % (expr, word))
                 matches.append(word)
+        matches.sort()
         return matches
 
 def get_class_members(klass):
index 2da7fce3f03f2da38503a81cfd07560064d5d1d9..927df34534870baa9459e35c42b82540c75d8ab6 100644 (file)
@@ -64,6 +64,19 @@ class TestRlcompleter(unittest.TestCase):
                          ['egg.{}('.format(x) for x in dir(str)
                           if x.startswith('s')])
 
+    def test_excessive_getattr(self):
+        # Ensure getattr() is invoked no more than once per attribute
+        class Foo:
+            calls = 0
+            @property
+            def bar(self):
+                self.calls += 1
+                return None
+        f = Foo()
+        completer = rlcompleter.Completer(dict(f=f))
+        self.assertEqual(completer.complete('f.b', 0), 'f.bar')
+        self.assertEqual(f.calls, 1)
+
     def test_complete(self):
         completer = rlcompleter.Completer()
         self.assertEqual(completer.complete('', 0), '\t')
index aacd51ea079036b78efcca8f8acad0e2ef590f1c..ec2b4afecd67cc74ed21dd01c9cc4fb9ed14df68 100644 (file)
--- a/Misc/NEWS
+++ b/Misc/NEWS
@@ -103,6 +103,9 @@ Core and Builtins
 Library
 -------
 
+- Issue #25590: In the Readline completer, only call getattr() once per
+  attribute.
+
 - Issue #25498: Fix a crash when garbage-collecting ctypes objects created
   by wrapping a memoryview.  This was a regression made in 3.4.3.  Based
   on patch by Eryksun.