forked from VimPlug/jedi
Further import completion improvements.
This commit is contained in:
+44
-4
@@ -67,10 +67,10 @@ class ErrorStatement(object):
|
||||
self._next_start_pos = next_start_pos
|
||||
|
||||
def __repr__(self):
|
||||
return '<%s next: %s@%s>' % (
|
||||
return '<%s %s@%s>' % (
|
||||
type(self).__name__,
|
||||
repr(self.next_token),
|
||||
self.next_start_pos
|
||||
self.end_pos
|
||||
)
|
||||
|
||||
@property
|
||||
@@ -104,6 +104,32 @@ class ErrorStatement(object):
|
||||
first = next(iterator)
|
||||
return first.get_code(include_prefix=include_prefix) + ''.join(node.get_code() for node in iterator)
|
||||
|
||||
def get_next_leaf(self):
|
||||
for child in self.parent.children:
|
||||
if child.start_pos == self.end_pos:
|
||||
return child.first_leaf()
|
||||
|
||||
if child.start_pos > self.end_pos:
|
||||
raise NotImplementedError('Node not found, must be in error statements.')
|
||||
raise ValueError("Doesn't have a next leaf")
|
||||
|
||||
def set_parent(self, root_node):
|
||||
"""
|
||||
Used by the parser at the end of parsing. The error statements parents
|
||||
have to be calculated at the end, because they are basically ripped out
|
||||
of the stack at which time its parents don't yet exist..
|
||||
"""
|
||||
start_pos = self.start_pos
|
||||
for c in root_node.children:
|
||||
if c.start_pos < start_pos <= c.end_pos:
|
||||
return self.set_parent(c)
|
||||
|
||||
self.parent = root_node
|
||||
|
||||
|
||||
class ErrorToken(tree.LeafWithNewLines):
|
||||
type = 'error_token'
|
||||
|
||||
|
||||
class ParserSyntaxError(object):
|
||||
def __init__(self, message, position):
|
||||
@@ -193,6 +219,9 @@ class Parser(object):
|
||||
if self._added_newline:
|
||||
self.remove_last_newline()
|
||||
|
||||
for e in self._error_statements:
|
||||
e.set_parent(self.get_parsed_node())
|
||||
|
||||
def get_parsed_node(self):
|
||||
return self._parsed
|
||||
|
||||
@@ -381,13 +410,23 @@ class ParserWithRecovery(Parser):
|
||||
stack[index]
|
||||
|
||||
#print('err', token.tok_name[typ], repr(value), start_pos, len(stack), index)
|
||||
self._stack_removal(grammar, stack, arcs, index + 1, value, start_pos)
|
||||
if self._stack_removal(grammar, stack, arcs, index + 1, value, start_pos):
|
||||
#add_token_callback(typ, value, prefix, start_pos)
|
||||
pass
|
||||
else:
|
||||
#error_leaf = ErrorToken(self.position_modifier, value, start_pos, prefix)
|
||||
#stack = [(None, [error_leaf])]
|
||||
# TODO document the shizzle!
|
||||
#self._error_statements.append(ErrorStatement(stack, None, None,
|
||||
# self.position_modifier, error_leaf.end_pos))
|
||||
return
|
||||
|
||||
if typ == INDENT:
|
||||
# For every deleted INDENT we have to delete a DEDENT as well.
|
||||
# Otherwise the parser will get into trouble and DEDENT too early.
|
||||
self._omit_dedent_list.append(self._indent_counter)
|
||||
|
||||
if value in ('import', 'class', 'def', 'try', 'while', 'return'):
|
||||
if value in ('import', 'class', 'def', 'try', 'while', 'return', '\n'):
|
||||
# Those can always be new statements.
|
||||
add_token_callback(typ, value, prefix, start_pos)
|
||||
elif typ == DEDENT and symbol == 'suite':
|
||||
@@ -435,6 +474,7 @@ class ParserWithRecovery(Parser):
|
||||
self._last_failed_start_pos = start_pos
|
||||
|
||||
stack[start_index:] = []
|
||||
return failed_stack
|
||||
|
||||
def _tokenize(self, tokenizer):
|
||||
for typ, value, start_pos, prefix in tokenizer:
|
||||
|
||||
+58
-2
@@ -37,6 +37,7 @@ import re
|
||||
from inspect import cleandoc
|
||||
from itertools import chain
|
||||
import textwrap
|
||||
import abc
|
||||
|
||||
from jedi import common
|
||||
from jedi._compatibility import (Python3Method, encoding, is_py3, utf8_repr,
|
||||
@@ -194,6 +195,7 @@ class Base(object):
|
||||
# Default is not being a scope. Just inherit from Scope.
|
||||
return False
|
||||
|
||||
@abc.abstractmethod
|
||||
def nodes_to_execute(self, last_added=False):
|
||||
raise NotImplementedError()
|
||||
|
||||
@@ -238,6 +240,12 @@ class Leaf(Base):
|
||||
def start_pos(self, value):
|
||||
self._start_pos = value[0] - self.position_modifier.line, value[1]
|
||||
|
||||
def get_start_pos_of_prefix(self):
|
||||
try:
|
||||
return self.get_previous().end_pos
|
||||
except IndexError:
|
||||
return 1, 0 # It's the first leaf.
|
||||
|
||||
@property
|
||||
def end_pos(self):
|
||||
return (self._start_pos[0] + self.position_modifier.line,
|
||||
@@ -250,6 +258,8 @@ class Leaf(Base):
|
||||
def get_previous(self):
|
||||
"""
|
||||
Returns the previous leaf in the parser tree.
|
||||
Raises an IndexError if it's the first element.
|
||||
# TODO rename to get_previous_leaf
|
||||
"""
|
||||
node = self
|
||||
while True:
|
||||
@@ -269,6 +279,33 @@ class Leaf(Base):
|
||||
except AttributeError: # A Leaf doesn't have children.
|
||||
return node
|
||||
|
||||
def first_leaf(self):
|
||||
return self
|
||||
|
||||
def get_next_leaf(self):
|
||||
"""
|
||||
Returns the previous leaf in the parser tree.
|
||||
Raises an IndexError if it's the last element.
|
||||
"""
|
||||
node = self
|
||||
while True:
|
||||
c = node.parent.children
|
||||
i = c.index(self)
|
||||
if i == len(c) - 1:
|
||||
node = node.parent
|
||||
if node.parent is None:
|
||||
raise IndexError('Cannot access the next element of the last one.')
|
||||
else:
|
||||
node = c[i + 1]
|
||||
break
|
||||
|
||||
while True:
|
||||
try:
|
||||
node = node.children[0]
|
||||
except AttributeError: # A Leaf doesn't have children.
|
||||
return node
|
||||
|
||||
|
||||
def get_code(self, normalized=False, include_prefix=True):
|
||||
if normalized:
|
||||
return self.value
|
||||
@@ -474,6 +511,9 @@ class BaseNode(Base):
|
||||
def start_pos(self):
|
||||
return self.children[0].start_pos
|
||||
|
||||
def get_start_pos_of_prefix(self):
|
||||
return self.children[0].get_start_pos_of_prefix()
|
||||
|
||||
@property
|
||||
def end_pos(self):
|
||||
return self.children[-1].end_pos
|
||||
@@ -498,9 +538,14 @@ class BaseNode(Base):
|
||||
return result
|
||||
return None
|
||||
|
||||
def get_leaf_for_position(self, position):
|
||||
def get_leaf_for_position(self, position, include_prefixes=False):
|
||||
for c in self.children:
|
||||
if c.start_pos <= position <= c.end_pos:
|
||||
if include_prefixes:
|
||||
start_pos = c.get_start_pos_with_prefix()
|
||||
else:
|
||||
start_pos = c.start_pos
|
||||
|
||||
if start_pos <= position <= c.end_pos:
|
||||
try:
|
||||
return c.get_leaf_for_position(position)
|
||||
except AttributeError:
|
||||
@@ -528,6 +573,17 @@ class BaseNode(Base):
|
||||
except AttributeError:
|
||||
return self.children[0]
|
||||
|
||||
def get_next_leaf(self):
|
||||
"""
|
||||
Raises an IndexError if it's the last node. (Would be the module)
|
||||
"""
|
||||
c = self.parent.children
|
||||
index = c.index(self)
|
||||
if index == len(c) - 1:
|
||||
return self.get_next_leaf()
|
||||
else:
|
||||
return c[index + 1]
|
||||
|
||||
@utf8_repr
|
||||
def __repr__(self):
|
||||
code = self.get_code().replace('\n', ' ').strip()
|
||||
|
||||
Reference in New Issue
Block a user