forked from VimPlug/jedi
temporary conversion of the imports module to separate import processing from actually importing things
This commit is contained in:
@@ -401,7 +401,7 @@ class Script(object):
|
|||||||
search_name = unicode(user_scope.name)
|
search_name = unicode(user_scope.name)
|
||||||
elif isinstance(user_stmt, pr.Import):
|
elif isinstance(user_stmt, pr.Import):
|
||||||
s, name_part = helpers.get_on_import_stmt(self._evaluator,
|
s, name_part = helpers.get_on_import_stmt(self._evaluator,
|
||||||
self._user_context, user_stmt)
|
self._user_context, user_stmt)
|
||||||
try:
|
try:
|
||||||
definitions = [s.follow(is_goto=True)[0]]
|
definitions = [s.follow(is_goto=True)[0]]
|
||||||
except IndexError:
|
except IndexError:
|
||||||
|
|||||||
@@ -400,6 +400,7 @@ class Completion(BaseDefinition):
|
|||||||
if self._definition.isinstance(pr.Statement):
|
if self._definition.isinstance(pr.Statement):
|
||||||
defs = self._evaluator.eval_statement(self._definition)
|
defs = self._evaluator.eval_statement(self._definition)
|
||||||
elif self._definition.isinstance(pr.Import):
|
elif self._definition.isinstance(pr.Import):
|
||||||
|
print(self._definition)
|
||||||
defs = imports.strip_imports(self._evaluator, [self._definition])
|
defs = imports.strip_imports(self._evaluator, [self._definition])
|
||||||
else:
|
else:
|
||||||
return [self]
|
return [self]
|
||||||
|
|||||||
@@ -51,26 +51,116 @@ class ImportPath(pr.Base):
|
|||||||
self.is_just_from = is_just_from
|
self.is_just_from = is_just_from
|
||||||
|
|
||||||
self.is_partial_import = bool(max(0, kill_count))
|
self.is_partial_import = bool(max(0, kill_count))
|
||||||
path = import_stmt.get_parent_until().path
|
|
||||||
self.file_path = os.path.dirname(path) if path is not None else None
|
|
||||||
|
|
||||||
# rest is import_path resolution
|
# rest is import_path resolution
|
||||||
self.import_path = []
|
import_path = []
|
||||||
if import_stmt.from_ns:
|
if import_stmt.from_ns:
|
||||||
self.import_path += import_stmt.from_ns.names
|
import_path += import_stmt.from_ns.names
|
||||||
if import_stmt.namespace:
|
if import_stmt.namespace:
|
||||||
if self._is_nested_import() and not direct_resolve:
|
if self._is_nested_import() and not direct_resolve:
|
||||||
self.import_path.append(import_stmt.namespace.names[0])
|
import_path.append(import_stmt.namespace.names[0])
|
||||||
else:
|
else:
|
||||||
self.import_path += import_stmt.namespace.names
|
import_path += import_stmt.namespace.names
|
||||||
self.import_path = [str(name_part) for name_part in self.import_path]
|
import_path = [str(name_part) for name_part in import_path]
|
||||||
|
|
||||||
for i in range(kill_count + int(is_like_search)):
|
for i in range(kill_count + int(is_like_search)):
|
||||||
self.import_path.pop()
|
import_path.pop()
|
||||||
|
|
||||||
|
module = import_stmt.get_parent_until()
|
||||||
|
self._importer = Importer(evaluator, import_path, module)
|
||||||
|
|
||||||
def __repr__(self):
|
def __repr__(self):
|
||||||
return '<%s: %s>' % (type(self).__name__, self.import_stmt)
|
return '<%s: %s>' % (type(self).__name__, self.import_stmt)
|
||||||
|
|
||||||
|
def get_defined_names(self, on_import_stmt=False):
|
||||||
|
names = []
|
||||||
|
for scope in self.follow():
|
||||||
|
if scope is ImportPath.GlobalNamespace:
|
||||||
|
if not self._is_relative_import():
|
||||||
|
names += self._get_module_names()
|
||||||
|
|
||||||
|
if self._importer.file_path is not None:
|
||||||
|
path = os.path.abspath(self._importer.file_path)
|
||||||
|
for i in range(self.import_stmt.relative_count - 1):
|
||||||
|
path = os.path.dirname(path)
|
||||||
|
names += self._get_module_names([path])
|
||||||
|
|
||||||
|
if self._is_relative_import():
|
||||||
|
rel_path = self._importer.get_relative_path() + '/__init__.py'
|
||||||
|
if os.path.exists(rel_path):
|
||||||
|
m = load_module(rel_path)
|
||||||
|
names += m.get_defined_names()
|
||||||
|
else:
|
||||||
|
if on_import_stmt and isinstance(scope, pr.Module) \
|
||||||
|
and scope.path.endswith('__init__.py'):
|
||||||
|
pkg_path = os.path.dirname(scope.path)
|
||||||
|
paths = self._namespace_packages(pkg_path, self._importer.import_path)
|
||||||
|
names += self._get_module_names([pkg_path] + paths)
|
||||||
|
if self.is_just_from:
|
||||||
|
# In the case of an import like `from x.` we don't need to
|
||||||
|
# add all the variables.
|
||||||
|
if ['os'] == self._importer.import_path and not self._is_relative_import():
|
||||||
|
# os.path is a hardcoded exception, because it's a
|
||||||
|
# ``sys.modules`` modification.
|
||||||
|
names.append(self._generate_name('path'))
|
||||||
|
continue
|
||||||
|
from jedi.evaluate import finder
|
||||||
|
for s, scope_names in finder.get_names_of_scope(self._evaluator,
|
||||||
|
scope, include_builtin=False):
|
||||||
|
for n in scope_names:
|
||||||
|
if self.import_stmt.from_ns is None \
|
||||||
|
or self.is_partial_import:
|
||||||
|
# from_ns must be defined to access module
|
||||||
|
# values plus a partial import means that there
|
||||||
|
# is something after the import, which
|
||||||
|
# automatically implies that there must not be
|
||||||
|
# any non-module scope.
|
||||||
|
continue
|
||||||
|
names.append(n)
|
||||||
|
return names
|
||||||
|
|
||||||
|
def _generate_name(self, name):
|
||||||
|
parent = self.import_stmt
|
||||||
|
inf_pos = float('inf'), float('inf')
|
||||||
|
# Generate a statement that reflects autocompletion names.
|
||||||
|
#if is_like_search:
|
||||||
|
# parent = pr.Import(self.GlobalNamespace, inf_pos, inf_pos, n
|
||||||
|
return pr.Name(self.GlobalNamespace, [(name, inf_pos)],
|
||||||
|
inf_pos, inf_pos, parent)
|
||||||
|
|
||||||
|
def _get_module_names(self, search_path=None):
|
||||||
|
"""
|
||||||
|
Get the names of all modules in the search_path. This means file names
|
||||||
|
and not names defined in the files.
|
||||||
|
"""
|
||||||
|
|
||||||
|
print(self.import_stmt)
|
||||||
|
names = []
|
||||||
|
# add builtin module names
|
||||||
|
if search_path is None:
|
||||||
|
names += [self._generate_name(name) for name in sys.builtin_module_names]
|
||||||
|
|
||||||
|
if search_path is None:
|
||||||
|
search_path = self._sys_path_with_modifications()
|
||||||
|
for module_loader, name, is_pkg in pkgutil.iter_modules(search_path):
|
||||||
|
names.append(self._generate_name(name))
|
||||||
|
return names
|
||||||
|
|
||||||
|
def _sys_path_with_modifications(self):
|
||||||
|
# If you edit e.g. gunicorn, there will be imports like this:
|
||||||
|
# `from gunicorn import something`. But gunicorn is not in the
|
||||||
|
# sys.path. Therefore look if gunicorn is a parent directory, #56.
|
||||||
|
in_path = []
|
||||||
|
if self._importer.import_path:
|
||||||
|
parts = self._importer.file_path.split(os.path.sep)
|
||||||
|
for i, p in enumerate(parts):
|
||||||
|
if p == self._importer.import_path[0]:
|
||||||
|
new = os.path.sep.join(parts[:i])
|
||||||
|
in_path.append(new)
|
||||||
|
|
||||||
|
module = self.import_stmt.get_parent_until()
|
||||||
|
return in_path + sys_path.sys_path_with_modifications(module)
|
||||||
|
|
||||||
def _is_nested_import(self):
|
def _is_nested_import(self):
|
||||||
"""
|
"""
|
||||||
This checks for the special case of nested imports, without aliases and
|
This checks for the special case of nested imports, without aliases and
|
||||||
@@ -98,105 +188,53 @@ class ImportPath(pr.Base):
|
|||||||
debug.dbg('Generated a nested import: %s', new)
|
debug.dbg('Generated a nested import: %s', new)
|
||||||
return new
|
return new
|
||||||
|
|
||||||
def get_defined_names(self, on_import_stmt=False):
|
def _is_relative_import(self):
|
||||||
names = []
|
return bool(self.import_stmt.relative_count)
|
||||||
for scope in self.follow():
|
|
||||||
if scope is ImportPath.GlobalNamespace:
|
|
||||||
if self._is_relative_import() == 0:
|
|
||||||
names += self._get_module_names()
|
|
||||||
|
|
||||||
if self.file_path is not None:
|
def follow(self, is_goto=False):
|
||||||
path = os.path.abspath(self.file_path)
|
if self._evaluator.recursion_detector.push_stmt(self.import_stmt):
|
||||||
for i in range(self.import_stmt.relative_count - 1):
|
# check recursion
|
||||||
path = os.path.dirname(path)
|
return []
|
||||||
names += self._get_module_names([path])
|
try:
|
||||||
|
return self._importer.follow(is_goto)
|
||||||
|
finally:
|
||||||
|
self._evaluator.recursion_detector.pop_stmt()
|
||||||
|
|
||||||
if self._is_relative_import():
|
|
||||||
rel_path = self._get_relative_path() + '/__init__.py'
|
|
||||||
if os.path.exists(rel_path):
|
|
||||||
m = load_module(rel_path)
|
|
||||||
names += m.get_defined_names()
|
|
||||||
else:
|
|
||||||
if on_import_stmt and isinstance(scope, pr.Module) \
|
|
||||||
and scope.path.endswith('__init__.py'):
|
|
||||||
pkg_path = os.path.dirname(scope.path)
|
|
||||||
paths = self._namespace_packages(pkg_path, self.import_path)
|
|
||||||
names += self._get_module_names([pkg_path] + paths)
|
|
||||||
if self.is_just_from:
|
|
||||||
# In the case of an import like `from x.` we don't need to
|
|
||||||
# add all the variables.
|
|
||||||
if ['os'] == self.import_path and not self._is_relative_import():
|
|
||||||
# os.path is a hardcoded exception, because it's a
|
|
||||||
# ``sys.modules`` modification.
|
|
||||||
p = (0, 0)
|
|
||||||
names.append(pr.Name(self.GlobalNamespace, [('path', p)],
|
|
||||||
p, p, self.import_stmt))
|
|
||||||
continue
|
|
||||||
from jedi.evaluate import finder
|
|
||||||
for s, scope_names in finder.get_names_of_scope(self._evaluator,
|
|
||||||
scope, include_builtin=False):
|
|
||||||
for n in scope_names:
|
|
||||||
if self.import_stmt.from_ns is None \
|
|
||||||
or self.is_partial_import:
|
|
||||||
# from_ns must be defined to access module
|
|
||||||
# values plus a partial import means that there
|
|
||||||
# is something after the import, which
|
|
||||||
# automatically implies that there must not be
|
|
||||||
# any non-module scope.
|
|
||||||
continue
|
|
||||||
names.append(n)
|
|
||||||
return names
|
|
||||||
|
|
||||||
def _get_module_names(self, search_path=None):
|
class Importer(object):
|
||||||
|
def __init__(self, evaluator, import_path, module, level=0):
|
||||||
"""
|
"""
|
||||||
Get the names of all modules in the search_path. This means file names
|
An implementation similar to ``__import__``.
|
||||||
and not names defined in the files.
|
|
||||||
|
*level* specifies whether to use absolute or relative imports. 0 (the
|
||||||
|
default) means only perform absolute imports. Positive values for level
|
||||||
|
indicate the number of parent directories to search relative to the
|
||||||
|
directory of the module calling ``__import__()`` (see PEP 328 for the
|
||||||
|
details).
|
||||||
|
|
||||||
|
:param import_path: List of namespaces (strings).
|
||||||
"""
|
"""
|
||||||
def generate_name(name):
|
self.import_path = import_path
|
||||||
return pr.Name(self.GlobalNamespace, [(name, inf_pos)],
|
self._module = module
|
||||||
inf_pos, inf_pos, self.import_stmt)
|
path = module.path
|
||||||
|
# TODO abspath
|
||||||
|
self.file_path = os.path.dirname(path) if path is not None else None
|
||||||
|
|
||||||
names = []
|
def get_relative_path(self):
|
||||||
inf_pos = float('inf'), float('inf')
|
path = self.file_path
|
||||||
# add builtin module names
|
for i in range(self.import_stmt.relative_count - 1):
|
||||||
if search_path is None:
|
path = os.path.dirname(path)
|
||||||
names += [generate_name(name) for name in sys.builtin_module_names]
|
return path
|
||||||
|
|
||||||
if search_path is None:
|
|
||||||
search_path = self._sys_path_with_modifications()
|
|
||||||
for module_loader, name, is_pkg in pkgutil.iter_modules(search_path):
|
|
||||||
names.append(generate_name(name))
|
|
||||||
return names
|
|
||||||
|
|
||||||
def _sys_path_with_modifications(self):
|
|
||||||
# If you edit e.g. gunicorn, there will be imports like this:
|
|
||||||
# `from gunicorn import something`. But gunicorn is not in the
|
|
||||||
# sys.path. Therefore look if gunicorn is a parent directory, #56.
|
|
||||||
in_path = []
|
|
||||||
if self.import_path:
|
|
||||||
parts = self.file_path.split(os.path.sep)
|
|
||||||
for i, p in enumerate(parts):
|
|
||||||
if p == self.import_path[0]:
|
|
||||||
new = os.path.sep.join(parts[:i])
|
|
||||||
in_path.append(new)
|
|
||||||
|
|
||||||
module = self.import_stmt.get_parent_until()
|
|
||||||
return in_path + sys_path.sys_path_with_modifications(module)
|
|
||||||
|
|
||||||
def follow(self, is_goto=False):
|
def follow(self, is_goto=False):
|
||||||
"""
|
"""
|
||||||
Returns the imported modules.
|
Returns the imported modules.
|
||||||
"""
|
"""
|
||||||
if self._evaluator.recursion_detector.push_stmt(self.import_stmt):
|
|
||||||
# check recursion
|
|
||||||
return []
|
|
||||||
|
|
||||||
if self.import_path:
|
if self.import_path:
|
||||||
try:
|
try:
|
||||||
scope, rest = self._follow_file_system()
|
scope, rest = self._follow_file_system()
|
||||||
except ModuleNotFound:
|
except ModuleNotFound:
|
||||||
debug.warning('Module not found: %s', self.import_stmt)
|
debug.warning('Module not found: %s', self.import_path)
|
||||||
self._evaluator.recursion_detector.pop_stmt()
|
|
||||||
return []
|
return []
|
||||||
|
|
||||||
scopes = [scope]
|
scopes = [scope]
|
||||||
@@ -229,17 +267,28 @@ class ImportPath(pr.Base):
|
|||||||
scopes = [ImportPath.GlobalNamespace]
|
scopes = [ImportPath.GlobalNamespace]
|
||||||
debug.dbg('after import: %s', scopes)
|
debug.dbg('after import: %s', scopes)
|
||||||
|
|
||||||
self._evaluator.recursion_detector.pop_stmt()
|
|
||||||
return scopes
|
return scopes
|
||||||
|
|
||||||
def _is_relative_import(self):
|
def _follow_file_system(self):
|
||||||
return bool(self.import_stmt.relative_count)
|
if self.file_path:
|
||||||
|
sys_path_mod = list(self._sys_path_with_modifications())
|
||||||
|
if not self._module.has_explicit_absolute_import:
|
||||||
|
# If the module explicitly asks for absolute imports,
|
||||||
|
# there's probably a bogus local one.
|
||||||
|
sys_path_mod.insert(0, self.file_path)
|
||||||
|
|
||||||
def _get_relative_path(self):
|
# First the sys path is searched normally and if that doesn't
|
||||||
path = self.file_path
|
# succeed, try to search the parent directories, because sometimes
|
||||||
for i in range(self.import_stmt.relative_count - 1):
|
# Jedi doesn't recognize sys.path modifications (like py.test
|
||||||
path = os.path.dirname(path)
|
# stuff).
|
||||||
return path
|
old_path, temp_path = self.file_path, os.path.dirname(self.file_path)
|
||||||
|
while old_path != temp_path:
|
||||||
|
sys_path_mod.append(temp_path)
|
||||||
|
old_path, temp_path = temp_path, os.path.dirname(temp_path)
|
||||||
|
else:
|
||||||
|
sys_path_mod = list(sys_path.get_sys_path())
|
||||||
|
|
||||||
|
return self._follow_sys_path(sys_path_mod)
|
||||||
|
|
||||||
def _namespace_packages(self, found_path, import_path):
|
def _namespace_packages(self, found_path, import_path):
|
||||||
"""
|
"""
|
||||||
@@ -270,28 +319,6 @@ class ImportPath(pr.Base):
|
|||||||
return follow_path(iter(import_path), sys.path)
|
return follow_path(iter(import_path), sys.path)
|
||||||
return []
|
return []
|
||||||
|
|
||||||
def _follow_file_system(self):
|
|
||||||
if self.file_path:
|
|
||||||
sys_path_mod = list(self._sys_path_with_modifications())
|
|
||||||
module = self.import_stmt.get_parent_until()
|
|
||||||
if not module.has_explicit_absolute_import:
|
|
||||||
# If the module explicitly asks for absolute imports,
|
|
||||||
# there's probably a bogus local one.
|
|
||||||
sys_path_mod.insert(0, self.file_path)
|
|
||||||
|
|
||||||
# First the sys path is searched normally and if that doesn't
|
|
||||||
# succeed, try to search the parent directories, because sometimes
|
|
||||||
# Jedi doesn't recognize sys.path modifications (like py.test
|
|
||||||
# stuff).
|
|
||||||
old_path, temp_path = self.file_path, os.path.dirname(self.file_path)
|
|
||||||
while old_path != temp_path:
|
|
||||||
sys_path_mod.append(temp_path)
|
|
||||||
old_path, temp_path = temp_path, os.path.dirname(temp_path)
|
|
||||||
else:
|
|
||||||
sys_path_mod = list(sys_path.get_sys_path())
|
|
||||||
|
|
||||||
return self._follow_sys_path(sys_path_mod)
|
|
||||||
|
|
||||||
def _follow_sys_path(self, sys_path):
|
def _follow_sys_path(self, sys_path):
|
||||||
"""
|
"""
|
||||||
Find a module with a path (of the module, like usb.backend.libusb10).
|
Find a module with a path (of the module, like usb.backend.libusb10).
|
||||||
@@ -302,7 +329,7 @@ class ImportPath(pr.Base):
|
|||||||
if ns_path:
|
if ns_path:
|
||||||
path = ns_path
|
path = ns_path
|
||||||
elif self._is_relative_import():
|
elif self._is_relative_import():
|
||||||
path = self._get_relative_path()
|
path = self.get_relative_path()
|
||||||
|
|
||||||
if path is not None:
|
if path is not None:
|
||||||
importing = find_module(string, [path])
|
importing = find_module(string, [path])
|
||||||
@@ -328,7 +355,7 @@ class ImportPath(pr.Base):
|
|||||||
_continue = False
|
_continue = False
|
||||||
if self._is_relative_import() and len(self.import_path) == 1:
|
if self._is_relative_import() and len(self.import_path) == 1:
|
||||||
# follow `from . import some_variable`
|
# follow `from . import some_variable`
|
||||||
rel_path = self._get_relative_path()
|
rel_path = self.get_relative_path()
|
||||||
with common.ignored(ImportError):
|
with common.ignored(ImportError):
|
||||||
current_namespace = follow_str(rel_path, '__init__')
|
current_namespace = follow_str(rel_path, '__init__')
|
||||||
elif current_namespace[2]: # is a package
|
elif current_namespace[2]: # is a package
|
||||||
|
|||||||
@@ -51,7 +51,7 @@ SCOPE_CONTENTS = 'asserts', 'subscopes', 'imports', 'statements', 'returns'
|
|||||||
|
|
||||||
class GetCodeState(object):
|
class GetCodeState(object):
|
||||||
"""A helper class for passing the state of get_code in a thread-safe
|
"""A helper class for passing the state of get_code in a thread-safe
|
||||||
manner"""
|
manner."""
|
||||||
__slots__ = ("last_pos",)
|
__slots__ = ("last_pos",)
|
||||||
|
|
||||||
def __init__(self):
|
def __init__(self):
|
||||||
|
|||||||
@@ -26,7 +26,7 @@ def test_import_not_in_sys_path():
|
|||||||
|
|
||||||
|
|
||||||
def test_import_empty():
|
def test_import_empty():
|
||||||
""" github #340, imports have """
|
""" github #340, return the full word. """
|
||||||
completion = jedi.Script("import ").completions()[0]
|
completion = jedi.Script("import ").completions()[0]
|
||||||
definition = completion.follow_definition()[0]
|
definition = completion.follow_definition()[0]
|
||||||
print(definition)
|
print(definition)
|
||||||
|
|||||||
Reference in New Issue
Block a user