Files
jedi/functions.py
2012-08-03 00:14:14 +02:00

311 lines
9.3 KiB
Python

import re
import parsing
import evaluate
import modules
import debug
import imports
import os
class NotFoundError(Exception):
""" A custom error to avoid catching the wrong exceptions """
pass
class Completion(object):
def __init__(self, name, needs_dot, like_name_length):
self.name = name
self.needs_dot = needs_dot
self.like_name_length = like_name_length
@property
def complete(self):
dot = '.' if self.needs_dot else ''
return dot + self.name.names[-1][self.like_name_length:]
@property
def description(self):
return str(self.name.parent)
@property
def help(self):
try:
return str(self.name.parent.docstr)
except AttributeError:
return ''
def get_type(self):
return type(self.name.parent)
def get_vim_type(self):
"""
This is the only function, which is vim specific, it returns the vim
type, see help(complete-items)
"""
typ = self.get_type()
if typ == parsing.Statement:
return 'v' # variable
elif typ == parsing.Function:
return 'f' # function / method
elif typ in [parsing.Class, evaluate.Instance]:
return 't' # typedef -> abused as class
elif typ == parsing.Import:
return 'd' # define -> abused as import
if typ == parsing.Param:
return 'm' # member -> abused as param
else:
debug.dbg('other python type: ', typ)
return ''
def __str__(self):
return self.name.names[-1]
class Definition(object):
def __init__(self, definition):
""" The definition of a function """
self.definition = definition
def get_name(self):
try:
# is a func / class
return self.definition.name
except AttributeError:
try:
# is an array
return self.definition.type
except:
# is a statement
return self.definition.get_code()
@property
def module_name(self):
path = self.module_path
try:
return path[path.rindex('/') + 1:]
except ValueError:
return path
@property
def module_path(self):
par = self.definition
while True:
if par.parent is not None:
par = par.parent
else:
break
return str(par.path)
def in_builtin_module(self):
return not self.module_path.endswith('.py')
@property
def line_nr(self):
return self.definition.start_pos[0]
@property
def column(self):
return self.definition.start_pos[1]
@property
def description(self):
d = self.definition
if isinstance(d, evaluate.InstanceElement):
d = d.var
if isinstance(d, evaluate.parsing.Name):
d = d.parent
if isinstance(d, (evaluate.Class, evaluate.Instance)):
d = 'class ' + str(d.name)
elif isinstance(d, (evaluate.Function, evaluate.parsing.Function)):
d = 'def ' + str(d.name)
elif isinstance(d, evaluate.parsing.Module):
p = str(d.path)
# only show module name
p = re.sub(r'^.*?([\w\d]+)(/__init__)?.py$', r'\1', p)
d = 'module ' + p
else:
d = d.get_code().replace('\n', '')
return d
def __str__(self):
if self.module_path[0] == '/':
position = '@%s' % (self.line_nr)
else:
# no path - is a builtin
position = ''
return "%s:%s%s" % (self.module_name, self.get_name(), position)
def __repr__(self):
return "<%s %s>" % (self.__class__.__name__, self.definition)
def get_completion_parts(path):
"""
Returns the parts for the completion
:return: tuple - (path, dot, like)
"""
match = re.match(r'^(.*?)(\.|)(\w?[\w\d]*)$', path, flags=re.S)
return match.groups()
def complete(source, line, column, source_path):
"""
An auto completer for python files.
:param source: The source code of the current file
:type source: string
:param line: The line to complete in.
:type line: int
:param col: The column to complete in.
:type col: int
:param source_path: The path in the os, the current module is in.
:type source_path: str
:return: list of Completion objects.
:rtype: list
"""
pos = (line, column)
f = modules.ModuleWithCursor(source_path, source=source, position=pos)
path = f.get_path_until_cursor()
path, dot, like = get_completion_parts(path)
try:
scopes = prepare_goto(source, pos, source_path, f, path, True)
except NotFoundError:
scope_generator = evaluate.get_names_for_scope(f.parser.user_scope)
completions = []
for dummy, name_list in scope_generator:
completions += name_list
else:
completions = []
debug.dbg('possible scopes', scopes)
for s in scopes:
# TODO is this really the right way? just ignore the functions? \
# do the magic functions first? and then recheck here?
if not isinstance(s, evaluate.Function):
completions += s.get_defined_names()
completions = [c for c in completions
if c.names[-1].lower().startswith(like.lower())]
_clear_caches()
needs_dot = not dot and path
return [Completion(c, needs_dot, len(like)) for c in set(completions)]
def prepare_goto(source, position, source_path, module, goto_path,
is_like_search=False):
scope = module.parser.user_scope
debug.dbg('start: %s in %s' % (goto_path, scope))
user_stmt = module.parser.user_stmt
if isinstance(user_stmt, parsing.Import):
scopes = [imports.ImportPath(user_stmt, is_like_search)]
else:
# just parse one statement, take it and evaluate it
r = parsing.PyFuzzyParser(goto_path, source_path)
try:
stmt = r.top.statements[0]
except IndexError:
raise NotFoundError()
else:
stmt.start_pos = position
stmt.parent = scope
scopes = evaluate.follow_statement(stmt)
return scopes
def get_definitions(source, line, column, source_path):
"""
Returns the definitions of a the path under the cursor.
This is not a goto function! This follows complicated paths and returns the
end, not the first definition.
:param source: The source code of the current file
:type source: string
:param line: The line to complete in.
:type line: int
:param col: The column to complete in.
:type col: int
:param source_path: The path in the os, the current module is in.
:type source_path: int
:return: list of Definition objects, which are basically scopes.
:rtype: list
"""
pos = (line, column)
f = modules.ModuleWithCursor(source_path, source=source, position=pos)
goto_path = f.get_path_under_cursor()
scopes = prepare_goto(source, pos, source_path, f, goto_path)
_clear_caches()
return [Definition(s) for s in set(scopes)]
def goto(source, line, column, source_path):
pos = (line, column)
f = modules.ModuleWithCursor(source_path, source=source, position=pos)
goto_path = f.get_path_under_cursor()
goto_path, dot, search_name = get_completion_parts(goto_path)
# define goto path the right way
if not dot:
goto_path = search_name
search_name = None
scopes = prepare_goto(source, pos, source_path, f, goto_path)
if not dot:
try:
definitions = [evaluate.statement_path[1]]
except IndexError:
definitions = []
for s in scopes:
if isinstance(s, imports.ImportPath):
s = s.follow()[0]
definitions.append(s)
else:
def remove_unreal_imports(names):
"""
These imports are only virtual, because of multi-line imports.
"""
new_names = []
for n in names:
par = n.parent
if isinstance(par, parsing.Import) and not par.start_pos[0]:
# this is a special case: If
for scope in imports.ImportPath(par).follow():
if isinstance(scope, parsing.Import):
temp = scope.get_defined_names()
new_names += remove_unreal_imports(temp)
elif isinstance(scope, parsing.Module):
new_names.append(scope.get_module_name(n.names))
else:
new_names.append(n)
return new_names
names = []
for s in scopes:
names += s.get_defined_names()
names = remove_unreal_imports(names)
definitions = [n for n in names if n.names[-1] == search_name]
_clear_caches()
return [Definition(d) for d in set(definitions)]
def set_debug_function(func_cb):
"""
You can define a callback debug function to get all the debug messages.
:param func_cb: The callback function for debug messages, with n params.
"""
debug.debug_function = func_cb
def _clear_caches():
evaluate.clear_caches()