1
0
forked from VimPlug/jedi
Files
jedi-fork/jedi/utils.py
2013-08-15 14:10:41 +04:30

111 lines
4.2 KiB
Python

"""
Utilities for end-users.
"""
from __future__ import absolute_import
import __main__
import re
from jedi._compatibility import builtins
from jedi import Interpreter
def setup_readline(namespace=__main__.__dict__):
"""
Install Jedi completer to :mod:`readline`.
This function setups :mod:`readline` to use Jedi in Python interactive
shell. If you want to use a custom ``PYTHONSTARTUP`` file (typically
``$HOME/.pythonrc.py``), you can add this piece of code::
try:
from jedi.utils import setup_readline
setup_readline()
except ImportError:
# Fallback to the stdlib readline completer if it is installed.
# Taken from http://docs.python.org/2/library/rlcompleter.html
print("Jedi is not installed, falling back to readline")
try:
import readline
import rlcompleter
readline.parse_and_bind("tab: complete")
except ImportError:
print("Readline is not installed either. No tab completion is enabled.")
This will fallback to the readline completer if Jedi is not installed.
The readline completer will only complete names in the global namespace,
so for example,
>>> ran<TAB> # doctest: +SKIP
will complete to ``range``
with both Jedi and readline, but
>>> range(10).cou<TAB> # doctest: +SKIP
will show complete to ``range(10).count`` only with Jedi.
You'll also need to add ``export PYTHONSTARTUP=$HOME/.pythonrc.py`` to
your shell profile (usually ``.bash_profile`` or ``.profile`` if you use
bash).
"""
try:
import readline
except ImportError:
print("Module readline not available.")
else:
class JediRL():
def complete(self, text, state):
"""
This complete stuff is pretty weird, a generator would make
a lot more sense, but probably due to backwards compatibility
this is still the way how it works.
The only important part is stuff in the ``state == 0`` flow,
everything else has been copied from the ``rlcompleter`` std.
library module.
"""
if state == 0:
interpreter = Interpreter(text, [namespace])
# The following part is a bit hackish, because it tries to
# directly access some jedi internals. The goal is to just
# use the "default" completion with ``getattr`` if
# possible.
path, dot, like = interpreter._get_completion_parts()
# Shouldn't be an import statement and just a simple path
# with dots.
if not re.match('^\s*(import|from) ', text) \
and (not path or re.match('^[\w][\w\d.]*$', path)):
paths = path.split('.') if path else []
namespaces = (namespace, builtins.__dict__)
for p in paths:
old, namespaces = namespaces, []
for n in old:
try:
namespaces.append(n[p].__dict__)
except (KeyError, AttributeError):
pass
self.matches = []
for n in namespaces:
for name in n.keys():
if name.lower().startswith(like.lower()):
self.matches.append(path + dot + name)
else:
completions = interpreter.completions()
self.matches = [path + dot + c.name_with_symbols
for c in completions]
try:
return self.matches[state]
except IndexError:
return None
readline.set_completer(JediRL().complete)
readline.parse_and_bind("tab: complete")
# No delimiters, Jedi handles that.
readline.set_completer_delims('')