mirror of
https://github.com/davidhalter/jedi.git
synced 2025-12-06 22:14:27 +08:00
Merge the async branch
This commit is contained in:
@@ -23,11 +23,11 @@ matrix:
|
|||||||
- JEDI_TEST_ENVIRONMENT=36
|
- JEDI_TEST_ENVIRONMENT=36
|
||||||
- python: 3.7-dev
|
- python: 3.7-dev
|
||||||
include:
|
include:
|
||||||
- python: 3.5
|
- python: 3.6
|
||||||
env:
|
env:
|
||||||
- TOXENV=cov
|
- TOXENV=cov
|
||||||
- JEDI_TEST_ENVIRONMENT=36
|
- JEDI_TEST_ENVIRONMENT=36
|
||||||
- python: 3.5
|
- python: 3.6
|
||||||
env: TOXENV=sith
|
env: TOXENV=sith
|
||||||
# For now ignore pypy, there are so many issues that we don't really need
|
# For now ignore pypy, there are so many issues that we don't really need
|
||||||
# to run it.
|
# to run it.
|
||||||
|
|||||||
@@ -63,10 +63,13 @@ class Context(BaseContext):
|
|||||||
arguments = ValuesArguments([ContextSet(value) for value in value_list])
|
arguments = ValuesArguments([ContextSet(value) for value in value_list])
|
||||||
return self.execute(arguments)
|
return self.execute(arguments)
|
||||||
|
|
||||||
def iterate(self, contextualized_node=None):
|
def iterate(self, contextualized_node=None, is_async=False):
|
||||||
debug.dbg('iterate')
|
debug.dbg('iterate')
|
||||||
try:
|
try:
|
||||||
iter_method = self.py__iter__
|
if is_async:
|
||||||
|
iter_method = self.py__aiter__
|
||||||
|
else:
|
||||||
|
iter_method = self.py__iter__
|
||||||
except AttributeError:
|
except AttributeError:
|
||||||
if contextualized_node is not None:
|
if contextualized_node is not None:
|
||||||
from jedi.evaluate import analysis
|
from jedi.evaluate import analysis
|
||||||
@@ -242,9 +245,9 @@ class ContextSet(BaseContextSet):
|
|||||||
def py__class__(self):
|
def py__class__(self):
|
||||||
return ContextSet.from_iterable(c.py__class__() for c in self._set)
|
return ContextSet.from_iterable(c.py__class__() for c in self._set)
|
||||||
|
|
||||||
def iterate(self, contextualized_node=None):
|
def iterate(self, contextualized_node=None, is_async=False):
|
||||||
from jedi.evaluate.lazy_context import get_merged_lazy_context
|
from jedi.evaluate.lazy_context import get_merged_lazy_context
|
||||||
type_iters = [c.iterate(contextualized_node) for c in self._set]
|
type_iters = [c.iterate(contextualized_node, is_async=is_async) for c in self._set]
|
||||||
for lazy_contexts in zip_longest(*type_iters):
|
for lazy_contexts in zip_longest(*type_iters):
|
||||||
yield get_merged_lazy_context(
|
yield get_merged_lazy_context(
|
||||||
[l for l in lazy_contexts if l is not None]
|
[l for l in lazy_contexts if l is not None]
|
||||||
|
|||||||
@@ -63,11 +63,19 @@ class FunctionContext(use_metaclass(CachedMetaClass, TreeContext)):
|
|||||||
"""
|
"""
|
||||||
Created to be used by inheritance.
|
Created to be used by inheritance.
|
||||||
"""
|
"""
|
||||||
yield_exprs = get_yield_exprs(self.evaluator, self.tree_node)
|
is_coroutine = self.tree_node.parent.type == 'async_stmt'
|
||||||
if yield_exprs:
|
is_generator = bool(get_yield_exprs(self.evaluator, self.tree_node))
|
||||||
return ContextSet(iterable.Generator(self.evaluator, function_execution))
|
|
||||||
|
if is_coroutine:
|
||||||
|
if is_generator:
|
||||||
|
return ContextSet(iterable.AsyncGenerator(self.evaluator, function_execution))
|
||||||
|
else:
|
||||||
|
return ContextSet(iterable.Coroutine(self.evaluator, function_execution))
|
||||||
else:
|
else:
|
||||||
return function_execution.get_return_values()
|
if is_generator:
|
||||||
|
return ContextSet(iterable.Generator(self.evaluator, function_execution))
|
||||||
|
else:
|
||||||
|
return function_execution.get_return_values()
|
||||||
|
|
||||||
def get_function_execution(self, arguments=None):
|
def get_function_execution(self, arguments=None):
|
||||||
if arguments is None:
|
if arguments is None:
|
||||||
@@ -171,7 +179,8 @@ class FunctionExecutionContext(TreeContext):
|
|||||||
yield LazyTreeContext(self, node)
|
yield LazyTreeContext(self, node)
|
||||||
|
|
||||||
@recursion.execution_recursion_decorator(default=iter([]))
|
@recursion.execution_recursion_decorator(default=iter([]))
|
||||||
def get_yield_lazy_contexts(self):
|
def get_yield_lazy_contexts(self, is_async=False):
|
||||||
|
# TODO: if is_async, wrap yield statements in Awaitable/async_generator_asend
|
||||||
for_parents = [(y, tree.search_ancestor(y, 'for_stmt', 'funcdef',
|
for_parents = [(y, tree.search_ancestor(y, 'for_stmt', 'funcdef',
|
||||||
'while_stmt', 'if_stmt'))
|
'while_stmt', 'if_stmt'))
|
||||||
for y in get_yield_exprs(self.evaluator, self.tree_node)]
|
for y in get_yield_exprs(self.evaluator, self.tree_node)]
|
||||||
|
|||||||
@@ -39,6 +39,19 @@ from jedi.evaluate.base_context import ContextSet, NO_CONTEXTS, Context, \
|
|||||||
TreeContext, ContextualizedNode
|
TreeContext, ContextualizedNode
|
||||||
from jedi.parser_utils import get_comp_fors
|
from jedi.parser_utils import get_comp_fors
|
||||||
|
|
||||||
|
try:
|
||||||
|
from types import CoroutineType
|
||||||
|
except ImportError:
|
||||||
|
HAS_COROUTINE = False
|
||||||
|
else:
|
||||||
|
HAS_COROUTINE = True
|
||||||
|
|
||||||
|
try:
|
||||||
|
from types import AsyncGeneratorType
|
||||||
|
except ImportError:
|
||||||
|
HAS_ASYNC_GENERATOR = False
|
||||||
|
else:
|
||||||
|
HAS_ASYNC_GENERATOR = True
|
||||||
|
|
||||||
class AbstractIterable(Context):
|
class AbstractIterable(Context):
|
||||||
builtin_methods = {}
|
builtin_methods = {}
|
||||||
@@ -55,6 +68,83 @@ class AbstractIterable(Context):
|
|||||||
return compiled.CompiledContextName(self, self.array_type)
|
return compiled.CompiledContextName(self, self.array_type)
|
||||||
|
|
||||||
|
|
||||||
|
@has_builtin_methods
|
||||||
|
class CoroutineMixin(object):
|
||||||
|
array_type = None
|
||||||
|
|
||||||
|
def get_filters(self, search_global, until_position=None, origin_scope=None):
|
||||||
|
gen_obj = compiled.create(self.evaluator, CoroutineType)
|
||||||
|
yield SpecialMethodFilter(self, self.builtin_methods, gen_obj)
|
||||||
|
for filter in gen_obj.get_filters(search_global):
|
||||||
|
yield filter
|
||||||
|
|
||||||
|
def py__bool__(self):
|
||||||
|
return True
|
||||||
|
|
||||||
|
def py__class__(self):
|
||||||
|
gen_obj = compiled.create(self.evaluator, CoroutineType)
|
||||||
|
return gen_obj.py__class__()
|
||||||
|
|
||||||
|
@property
|
||||||
|
def name(self):
|
||||||
|
return compiled.CompiledContextName(self, 'coroutine')
|
||||||
|
|
||||||
|
|
||||||
|
class Coroutine(CoroutineMixin, Context):
|
||||||
|
def __init__(self, evaluator, func_execution_context):
|
||||||
|
if not HAS_COROUTINE:
|
||||||
|
raise ImportError("Need python3.5 to support coroutines.")
|
||||||
|
super(Coroutine, self).__init__(evaluator, parent_context=evaluator.BUILTINS)
|
||||||
|
self._func_execution_context = func_execution_context
|
||||||
|
|
||||||
|
def execute_await(self):
|
||||||
|
return self._func_execution_context.get_return_values()
|
||||||
|
|
||||||
|
def __repr__(self):
|
||||||
|
return "<%s of %s>" % (type(self).__name__, self._func_execution_context)
|
||||||
|
|
||||||
|
|
||||||
|
@has_builtin_methods
|
||||||
|
class AsyncGeneratorMixin(object):
|
||||||
|
array_type = None
|
||||||
|
|
||||||
|
@register_builtin_method('__anext__')
|
||||||
|
def py__anext__(self):
|
||||||
|
return ContextSet.from_sets(lazy_context.infer() for lazy_context in self.py__aiter__())
|
||||||
|
|
||||||
|
def get_filters(self, search_global, until_position=None, origin_scope=None):
|
||||||
|
gen_obj = compiled.create(self.evaluator, AsyncGeneratorType)
|
||||||
|
yield SpecialMethodFilter(self, self.builtin_methods, gen_obj)
|
||||||
|
for filter in gen_obj.get_filters(search_global):
|
||||||
|
yield filter
|
||||||
|
|
||||||
|
def py__bool__(self):
|
||||||
|
return True
|
||||||
|
|
||||||
|
def py__class__(self):
|
||||||
|
gen_obj = compiled.create(self.evaluator, AsyncGeneratorType)
|
||||||
|
return gen_obj.py__class__()
|
||||||
|
|
||||||
|
@property
|
||||||
|
def name(self):
|
||||||
|
return compiled.CompiledContextName(self, 'asyncgenerator')
|
||||||
|
|
||||||
|
|
||||||
|
class AsyncGenerator(AsyncGeneratorMixin, Context):
|
||||||
|
"""Handling of `yield` functions."""
|
||||||
|
def __init__(self, evaluator, func_execution_context):
|
||||||
|
if not HAS_ASYNC_GENERATOR:
|
||||||
|
raise ImportError("Need python3.6 to support async generators.")
|
||||||
|
super(AsyncGenerator, self).__init__(evaluator, parent_context=evaluator.BUILTINS)
|
||||||
|
self._func_execution_context = func_execution_context
|
||||||
|
|
||||||
|
def py__aiter__(self):
|
||||||
|
return self._func_execution_context.get_yield_values(is_async=True)
|
||||||
|
|
||||||
|
def __repr__(self):
|
||||||
|
return "<%s of %s>" % (type(self).__name__, self._func_execution_context)
|
||||||
|
|
||||||
|
|
||||||
@has_builtin_methods
|
@has_builtin_methods
|
||||||
class GeneratorMixin(object):
|
class GeneratorMixin(object):
|
||||||
array_type = None
|
array_type = None
|
||||||
@@ -128,17 +218,18 @@ class Comprehension(AbstractIterable):
|
|||||||
cls = ListComprehension
|
cls = ListComprehension
|
||||||
return cls(evaluator, context, atom)
|
return cls(evaluator, context, atom)
|
||||||
|
|
||||||
def __init__(self, evaluator, defining_context, atom):
|
def __init__(self, evaluator, defining_context, atom, is_async=False):
|
||||||
super(Comprehension, self).__init__(evaluator)
|
super(Comprehension, self).__init__(evaluator)
|
||||||
self._defining_context = defining_context
|
self._defining_context = defining_context
|
||||||
self._atom = atom
|
self._atom = atom
|
||||||
|
|
||||||
def _get_comprehension(self):
|
def _get_comprehension(self):
|
||||||
|
"return 'a for a in b'"
|
||||||
# The atom contains a testlist_comp
|
# The atom contains a testlist_comp
|
||||||
return self._atom.children[1]
|
return self._atom.children[1]
|
||||||
|
|
||||||
def _get_comp_for(self):
|
def _get_comp_for(self):
|
||||||
# The atom contains a testlist_comp
|
"return CompFor('for a in b')"
|
||||||
return self._get_comprehension().children[1]
|
return self._get_comprehension().children[1]
|
||||||
|
|
||||||
def _eval_node(self, index=0):
|
def _eval_node(self, index=0):
|
||||||
@@ -156,13 +247,17 @@ class Comprehension(AbstractIterable):
|
|||||||
|
|
||||||
def _nested(self, comp_fors, parent_context=None):
|
def _nested(self, comp_fors, parent_context=None):
|
||||||
comp_for = comp_fors[0]
|
comp_for = comp_fors[0]
|
||||||
input_node = comp_for.children[3]
|
|
||||||
|
is_async = 'async' == comp_for.children[comp_for.children.index('for') - 1]
|
||||||
|
|
||||||
|
input_node = comp_for.children[comp_for.children.index('in') + 1]
|
||||||
parent_context = parent_context or self._defining_context
|
parent_context = parent_context or self._defining_context
|
||||||
input_types = parent_context.eval_node(input_node)
|
input_types = parent_context.eval_node(input_node)
|
||||||
|
# TODO: simulate await if self.is_async
|
||||||
|
|
||||||
cn = ContextualizedNode(parent_context, input_node)
|
cn = ContextualizedNode(parent_context, input_node)
|
||||||
iterated = input_types.iterate(cn)
|
iterated = input_types.iterate(cn, is_async=is_async)
|
||||||
exprlist = comp_for.children[1]
|
exprlist = comp_for.children[comp_for.children.index('for') + 1]
|
||||||
for i, lazy_context in enumerate(iterated):
|
for i, lazy_context in enumerate(iterated):
|
||||||
types = lazy_context.infer()
|
types = lazy_context.infer()
|
||||||
dct = unpack_tuple_to_dict(parent_context, types, exprlist)
|
dct = unpack_tuple_to_dict(parent_context, types, exprlist)
|
||||||
@@ -675,7 +770,7 @@ class _ArrayInstance(object):
|
|||||||
for addition in additions:
|
for addition in additions:
|
||||||
yield addition
|
yield addition
|
||||||
|
|
||||||
def iterate(self, contextualized_node=None):
|
def iterate(self, contextualized_node=None, is_async=False):
|
||||||
return self.py__iter__()
|
return self.py__iter__()
|
||||||
|
|
||||||
|
|
||||||
|
|||||||
@@ -70,22 +70,38 @@ def eval_node(context, element):
|
|||||||
return eval_expr_stmt(context, element)
|
return eval_expr_stmt(context, element)
|
||||||
elif typ in ('power', 'atom_expr'):
|
elif typ in ('power', 'atom_expr'):
|
||||||
first_child = element.children[0]
|
first_child = element.children[0]
|
||||||
if not (first_child.type == 'keyword' and first_child.value == 'await'):
|
children = element.children[1:]
|
||||||
context_set = eval_atom(context, first_child)
|
had_await = False
|
||||||
for trailer in element.children[1:]:
|
if first_child.type == 'keyword' and first_child.value == 'await':
|
||||||
if trailer == '**': # has a power operation.
|
had_await = True
|
||||||
right = evaluator.eval_element(context, element.children[2])
|
first_child = children.pop(0)
|
||||||
context_set = _eval_comparison(
|
|
||||||
evaluator,
|
context_set = eval_atom(context, first_child)
|
||||||
context,
|
for trailer in children:
|
||||||
context_set,
|
if trailer == '**': # has a power operation.
|
||||||
trailer,
|
right = evaluator.eval_element(context, children[1])
|
||||||
right
|
context_set = _eval_comparison(
|
||||||
)
|
evaluator,
|
||||||
break
|
context,
|
||||||
context_set = eval_trailer(context, context_set, trailer)
|
context_set,
|
||||||
return context_set
|
trailer,
|
||||||
return NO_CONTEXTS
|
right
|
||||||
|
)
|
||||||
|
break
|
||||||
|
context_set = eval_trailer(context, context_set, trailer)
|
||||||
|
|
||||||
|
if had_await:
|
||||||
|
await_context_set = ContextSet()
|
||||||
|
for context in context_set:
|
||||||
|
try:
|
||||||
|
func = context.execute_await
|
||||||
|
except AttributeError:
|
||||||
|
debug.warning('Tried to run execute_await on context %s', context)
|
||||||
|
pass
|
||||||
|
else:
|
||||||
|
await_context_set |= func()
|
||||||
|
return await_context_set
|
||||||
|
return context_set
|
||||||
elif typ in ('testlist_star_expr', 'testlist',):
|
elif typ in ('testlist_star_expr', 'testlist',):
|
||||||
# The implicit tuple in statements.
|
# The implicit tuple in statements.
|
||||||
return ContextSet(iterable.SequenceLiteralContext(evaluator, context, element))
|
return ContextSet(iterable.SequenceLiteralContext(evaluator, context, element))
|
||||||
|
|||||||
@@ -6,17 +6,39 @@ raise errors or return extremely strange results.
|
|||||||
"""
|
"""
|
||||||
|
|
||||||
async def x():
|
async def x():
|
||||||
|
return 1
|
||||||
|
|
||||||
|
#? []
|
||||||
|
x.cr_awai
|
||||||
|
|
||||||
|
#? ['cr_await']
|
||||||
|
x().cr_awai
|
||||||
|
|
||||||
|
a = await x()
|
||||||
|
#? int()
|
||||||
|
a
|
||||||
|
|
||||||
|
async def y():
|
||||||
argh = await x()
|
argh = await x()
|
||||||
#?
|
#? int()
|
||||||
argh
|
argh
|
||||||
return 2
|
return 2
|
||||||
|
|
||||||
#? int()
|
async def asgen():
|
||||||
x()
|
yield 1
|
||||||
|
await asyncio.sleep(0)
|
||||||
|
yield 2
|
||||||
|
|
||||||
a = await x()
|
async def wrapper():
|
||||||
#?
|
#? int()
|
||||||
a
|
[x async for x in asgen()][0]
|
||||||
|
|
||||||
|
async for y in asgen():
|
||||||
|
# TODO: make this an int()
|
||||||
|
y
|
||||||
|
|
||||||
|
#? ['__anext__']
|
||||||
|
asgen().__ane
|
||||||
|
|
||||||
|
|
||||||
async def x2():
|
async def x2():
|
||||||
|
|||||||
Reference in New Issue
Block a user