Kaydet (Commit) ab824222 authored tarafından Serhiy Storchaka's avatar Serhiy Storchaka

Issue #25011: rlcomplete now omits private and special attribute names unless

the prefix starts with underscores.
üst 8ace8e99
...@@ -103,6 +103,14 @@ operator ...@@ -103,6 +103,14 @@ operator
(Contributed by Joe Jevnik in :issue:`24379`.) (Contributed by Joe Jevnik in :issue:`24379`.)
rlcomplete
----------
Private and special attribute names now are omitted unless the prefix starts
with underscores. A space or a colon can be added after completed keyword.
(Contributed by Serhiy Storchaka in :issue:`25011` and :issue:`25209`.)
Optimizations Optimizations
============= =============
......
...@@ -142,20 +142,35 @@ class Completer: ...@@ -142,20 +142,35 @@ 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: if attr == '':
if word[:n] == attr and hasattr(thisobject, word): noprefix = '_'
val = getattr(thisobject, word) elif attr == '_':
word = self._callable_postfix(val, "%s.%s" % (expr, word)) noprefix = '__'
matches.append(word) else:
noprefix = None
while True:
for word in words:
if (word[:n] == attr and
not (noprefix and word[:n+1] == noprefix) and
hasattr(thisobject, word)):
val = getattr(thisobject, word)
word = self._callable_postfix(val, "%s.%s" % (expr, word))
matches.append(word)
if matches or not noprefix:
break
if noprefix == '_':
noprefix = '__'
else:
noprefix = None
matches.sort()
return matches return matches
def get_class_members(klass): def get_class_members(klass):
......
...@@ -5,6 +5,7 @@ import rlcompleter ...@@ -5,6 +5,7 @@ import rlcompleter
class CompleteMe: class CompleteMe:
""" Trivial class used in testing rlcompleter.Completer. """ """ Trivial class used in testing rlcompleter.Completer. """
spam = 1 spam = 1
_ham = 2
class TestRlcompleter(unittest.TestCase): class TestRlcompleter(unittest.TestCase):
...@@ -51,11 +52,25 @@ class TestRlcompleter(unittest.TestCase): ...@@ -51,11 +52,25 @@ class TestRlcompleter(unittest.TestCase):
['str.{}('.format(x) for x in dir(str) ['str.{}('.format(x) for x in dir(str)
if x.startswith('s')]) if x.startswith('s')])
self.assertEqual(self.stdcompleter.attr_matches('tuple.foospamegg'), []) self.assertEqual(self.stdcompleter.attr_matches('tuple.foospamegg'), [])
expected = sorted({'None.%s%s' % (x, '(' if x != '__doc__' else '')
for x in dir(None)})
self.assertEqual(self.stdcompleter.attr_matches('None.'), expected)
self.assertEqual(self.stdcompleter.attr_matches('None._'), expected)
self.assertEqual(self.stdcompleter.attr_matches('None.__'), expected)
# test with a customized namespace # test with a customized namespace
self.assertEqual(self.completer.attr_matches('CompleteMe.sp'), self.assertEqual(self.completer.attr_matches('CompleteMe.sp'),
['CompleteMe.spam']) ['CompleteMe.spam'])
self.assertEqual(self.completer.attr_matches('Completeme.egg'), []) self.assertEqual(self.completer.attr_matches('Completeme.egg'), [])
self.assertEqual(self.completer.attr_matches('CompleteMe.'),
['CompleteMe.mro(', 'CompleteMe.spam'])
self.assertEqual(self.completer.attr_matches('CompleteMe._'),
['CompleteMe._ham'])
matches = self.completer.attr_matches('CompleteMe.__')
for x in matches:
self.assertTrue(x.startswith('CompleteMe.__'), x)
self.assertIn('CompleteMe.__name__', matches)
self.assertIn('CompleteMe.__new__(', matches)
CompleteMe.me = CompleteMe CompleteMe.me = CompleteMe
self.assertEqual(self.completer.attr_matches('CompleteMe.me.me.sp'), self.assertEqual(self.completer.attr_matches('CompleteMe.me.me.sp'),
......
...@@ -27,6 +27,9 @@ Core and Builtins ...@@ -27,6 +27,9 @@ Core and Builtins
Library Library
------- -------
- Issue #25011: rlcomplete now omits private and special attribute names unless
the prefix starts with underscores.
- Issue #25209: rlcomplete now can add a space or a colon after completed keyword. - Issue #25209: rlcomplete now can add a space or a colon after completed keyword.
- Issue #22241: timezone.utc name is now plain 'UTC', not 'UTC-00:00'. - Issue #22241: timezone.utc name is now plain 'UTC', not 'UTC-00:00'.
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment