mirror of
https://github.com/python/cpython.git
synced 2025-10-07 07:31:46 +00:00
Issue #25590: Make rlcompleter only call getattr() once per attribute
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.
This commit is contained in:
parent
1bb651540e
commit
06622ead80
3 changed files with 26 additions and 7 deletions
|
@ -136,20 +136,23 @@ class Completer:
|
||||||
return []
|
return []
|
||||||
|
|
||||||
# get the content of the object, except __builtins__
|
# get the content of the object, except __builtins__
|
||||||
words = dir(thisobject)
|
words = set(dir(thisobject))
|
||||||
if "__builtins__" in words:
|
words.discard("__builtins__")
|
||||||
words.remove("__builtins__")
|
|
||||||
|
|
||||||
if hasattr(thisobject, '__class__'):
|
if hasattr(thisobject, '__class__'):
|
||||||
words.append('__class__')
|
words.add('__class__')
|
||||||
words.extend(get_class_members(thisobject.__class__))
|
words.update(get_class_members(thisobject.__class__))
|
||||||
matches = []
|
matches = []
|
||||||
n = len(attr)
|
n = len(attr)
|
||||||
for word in words:
|
for word in words:
|
||||||
if word[:n] == attr and hasattr(thisobject, word):
|
if word[:n] == attr:
|
||||||
|
try:
|
||||||
val = getattr(thisobject, word)
|
val = getattr(thisobject, word)
|
||||||
|
except Exception:
|
||||||
|
continue # Exclude properties that are not set
|
||||||
word = self._callable_postfix(val, "%s.%s" % (expr, word))
|
word = self._callable_postfix(val, "%s.%s" % (expr, word))
|
||||||
matches.append(word)
|
matches.append(word)
|
||||||
|
matches.sort()
|
||||||
return matches
|
return matches
|
||||||
|
|
||||||
def get_class_members(klass):
|
def get_class_members(klass):
|
||||||
|
|
|
@ -64,6 +64,19 @@ class TestRlcompleter(unittest.TestCase):
|
||||||
['egg.{}('.format(x) for x in dir(str)
|
['egg.{}('.format(x) for x in dir(str)
|
||||||
if x.startswith('s')])
|
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):
|
def test_complete(self):
|
||||||
completer = rlcompleter.Completer()
|
completer = rlcompleter.Completer()
|
||||||
self.assertEqual(completer.complete('', 0), '\t')
|
self.assertEqual(completer.complete('', 0), '\t')
|
||||||
|
|
|
@ -103,6 +103,9 @@ Core and Builtins
|
||||||
Library
|
Library
|
||||||
-------
|
-------
|
||||||
|
|
||||||
|
- Issue #25590: In the Readline completer, only call getattr() once per
|
||||||
|
attribute.
|
||||||
|
|
||||||
- Issue #25498: Fix a crash when garbage-collecting ctypes objects created
|
- 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
|
by wrapping a memoryview. This was a regression made in 3.4.3. Based
|
||||||
on patch by Eryksun.
|
on patch by Eryksun.
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue