##// END OF EJS Templates
Remove IPython/utils/ulinecache.py deprecated since IPython 6
r29007:ec5f39ac
Show More
test_async_helpers.py
325 lines | 8.7 KiB | text/x-python | PythonLexer
/ IPython / core / tests / test_async_helpers.py
Matthias Bussonnier
Prototype async REPL using IPython, take III...
r24463 """
farisachugthai
BUG: Fix mypy error...
r26040 Test for async helpers.
Matthias Bussonnier
Prototype async REPL using IPython, take III...
r24463
Should only trigger on python 3.5+ or will have syntax errors.
"""
Paul Ganssle
Refactor and tweak async-in-function tests...
r24489 from itertools import chain, repeat
Paul Ganssle
Add tests for SyntaxError with top-level return
r24486 from textwrap import dedent, indent
Matthias Bussonnier
Drop Python 3.9...
r28529 from typing import TYPE_CHECKING
Matthias Bussonnier
Prototype async REPL using IPython, take III...
r24463 from unittest import TestCase
Matthias Bussonnier
Drop Python 3.9...
r28529
import pytest
from IPython.core.async_helpers import _should_be_async
Matthias Bussonnier
remove trio from test requirement and skipp if not there
r24491 from IPython.testing.decorators import skip_without
farisachugthai
BUG: Fix mypy error...
r26040
if TYPE_CHECKING:
from IPython import get_ipython
Matthias Bussonnier
Update IPython/core/tests/test_async_helpers.py
r26074
farisachugthai
BUG: Fix mypy error...
r26040 ip = get_ipython()
Matthias Bussonnier
Remove assignment to ip in test files...
r25094
Matthias Bussonnier
Drop Python 3.9...
r28529 def iprc(x):
return ip.run_cell(dedent(x)).raise_error()
def iprc_nr(x):
return ip.run_cell(dedent(x))
Matthias Bussonnier
Prototype async REPL using IPython, take III...
r24463
kousik
Removed codepath for Python < 3.6 from test functions #11949
r25246
class AsyncTest(TestCase):
def test_should_be_async(self):
Samuel Gaist
[core][tests][async_helpers] Remove nose
r26887 self.assertFalse(_should_be_async("False"))
self.assertTrue(_should_be_async("await bar()"))
self.assertTrue(_should_be_async("x = await bar()"))
self.assertFalse(
kousik
Removed codepath for Python < 3.6 from test functions #11949
r25246 _should_be_async(
dedent(
"""
async def awaitable():
pass
"""
Matthias Bussonnier
runblack on new tests
r24478 )
)
kousik
Removed codepath for Python < 3.6 from test functions #11949
r25246 )
def _get_top_level_cases(self):
# These are test cases that should be valid in a function
# but invalid outside of a function.
test_cases = []
test_cases.append(('basic', "{val}"))
# Note, in all conditional cases, I use True instead of
# False so that the peephole optimizer won't optimize away
# the return, so CPython will see this as a syntax error:
#
# while True:
# break
# return
#
# But not this:
#
# while False:
# return
#
# See https://bugs.python.org/issue1875
test_cases.append(('if', dedent("""
if True:
{val}
""")))
test_cases.append(('while', dedent("""
while True:
{val}
break
""")))
test_cases.append(('try', dedent("""
try:
{val}
except:
pass
""")))
test_cases.append(('except', dedent("""
try:
pass
except:
{val}
""")))
test_cases.append(('finally', dedent("""
try:
pass
except:
pass
finally:
{val}
""")))
test_cases.append(('for', dedent("""
for _ in range(4):
{val}
""")))
test_cases.append(('nested', dedent("""
if True:
while True:
Paul Ganssle
Add tests for SyntaxError with top-level return
r24486 {val}
kousik
Removed codepath for Python < 3.6 from test functions #11949
r25246 break
""")))
Paul Ganssle
Add tests for SyntaxError with top-level return
r24486
kousik
Removed codepath for Python < 3.6 from test functions #11949
r25246 test_cases.append(('deep-nested', dedent("""
if True:
Paul Ganssle
Add tests for SyntaxError with top-level return
r24486 while True:
break
kousik
Removed codepath for Python < 3.6 from test functions #11949
r25246 for x in range(3):
if True:
while True:
for x in range(3):
{val}
""")))
Paul Ganssle
Add tests for SyntaxError with top-level return
r24486
kousik
Removed codepath for Python < 3.6 from test functions #11949
r25246 return test_cases
Paul Ganssle
Add tests for SyntaxError with top-level return
r24486
kousik
Removed codepath for Python < 3.6 from test functions #11949
r25246 def _get_ry_syntax_errors(self):
# This is a mix of tests that should be a syntax error if
# return or yield whether or not they are in a function
Paul Ganssle
Add tests for SyntaxError with top-level return
r24486
kousik
Removed codepath for Python < 3.6 from test functions #11949
r25246 test_cases = []
Paul Ganssle
Add tests for SyntaxError with top-level return
r24486
kousik
Removed codepath for Python < 3.6 from test functions #11949
r25246 test_cases.append(('class', dedent("""
class V:
{val}
""")))
test_cases.append(('nested-class', dedent("""
class V:
class C:
Paul Ganssle
Add tests for SyntaxError with top-level return
r24486 {val}
kousik
Removed codepath for Python < 3.6 from test functions #11949
r25246 """)))
Paul Ganssle
Add tests for SyntaxError with top-level return
r24486
kousik
Removed codepath for Python < 3.6 from test functions #11949
r25246 return test_cases
Paul Ganssle
Add tests for SyntaxError with top-level return
r24486
kousik
Removed codepath for Python < 3.6 from test functions #11949
r25246 def test_top_level_return_error(self):
tl_err_test_cases = self._get_top_level_cases()
tl_err_test_cases.extend(self._get_ry_syntax_errors())
vals = ('return', 'yield', 'yield from (_ for _ in range(3))',
dedent('''
def f():
pass
return
'''),
)
for test_name, test_case in tl_err_test_cases:
# This example should work if 'pass' is used as the value
with self.subTest((test_name, 'pass')):
iprc(test_case.format(val='pass'))
# It should fail with all the values
for val in vals:
with self.subTest((test_name, val)):
msg = "Syntax error not raised for %s, %s" % (test_name, val)
with self.assertRaises(SyntaxError, msg=msg):
iprc(test_case.format(val=val))
def test_in_func_no_error(self):
# Test that the implementation of top-level return/yield
# detection isn't *too* aggressive, and works inside a function
func_contexts = []
func_contexts.append(('func', False, dedent("""
def f():""")))
func_contexts.append(('method', False, dedent("""
class MyClass:
def __init__(self):
""")))
func_contexts.append(('async-func', True, dedent("""
async def f():""")))
func_contexts.append(('async-method', True, dedent("""
class MyClass:
async def f(self):""")))
func_contexts.append(('closure', False, dedent("""
def f():
def g():
""")))
def nest_case(context, case):
# Detect indentation
lines = context.strip().splitlines()
prefix_len = 0
for c in lines[-1]:
if c != ' ':
Paul Ganssle
Add tests for SyntaxError with top-level return
r24486 break
kousik
Removed codepath for Python < 3.6 from test functions #11949
r25246 prefix_len += 1
Paul Ganssle
Add tests for SyntaxError with top-level return
r24486
kousik
Removed codepath for Python < 3.6 from test functions #11949
r25246 indented_case = indent(case, ' ' * (prefix_len + 4))
return context + '\n' + indented_case
Paul Ganssle
Add tests for SyntaxError with top-level return
r24486
kousik
Removed codepath for Python < 3.6 from test functions #11949
r25246 # Gather and run the tests
Paul Ganssle
Add tests for SyntaxError with top-level return
r24486
kousik
Removed codepath for Python < 3.6 from test functions #11949
r25246 # yield is allowed in async functions, starting in Python 3.6,
# and yield from is not allowed in any version
vals = ('return', 'yield', 'yield from (_ for _ in range(3))')
Paul Ganssle
Add tests for SyntaxError with top-level return
r24486
kousik
Removed codepath for Python < 3.6 from test functions #11949
r25246 success_tests = zip(self._get_top_level_cases(), repeat(False))
failure_tests = zip(self._get_ry_syntax_errors(), repeat(True))
Paul Ganssle
Add tests for SyntaxError with top-level return
r24486
kousik
Removed codepath for Python < 3.6 from test functions #11949
r25246 tests = chain(success_tests, failure_tests)
felixzhuologist
add test for nonlocal case
r24667
kousik
Removed codepath for Python < 3.6 from test functions #11949
r25246 for context_name, async_func, context in func_contexts:
for (test_name, test_case), should_fail in tests:
nested_case = nest_case(context, test_case)
Paul Ganssle
Add tests for SyntaxError with top-level return
r24486
BaoGiang HoangVu
`async_safe` cleanup
r26167 for val in vals:
kousik
Removed codepath for Python < 3.6 from test functions #11949
r25246 test_id = (context_name, test_name, val)
cell = nested_case.format(val=val)
Matthias Bussonnier
Prototype async REPL using IPython, take III...
r24463
kousik
Removed codepath for Python < 3.6 from test functions #11949
r25246 with self.subTest(test_id):
BaoGiang HoangVu
`async_safe` cleanup
r26167 if should_fail:
kousik
Removed codepath for Python < 3.6 from test functions #11949
r25246 msg = ("SyntaxError not raised for %s" %
str(test_id))
with self.assertRaises(SyntaxError, msg=msg):
iprc(cell)
Matthias Bussonnier
Prototype async REPL using IPython, take III...
r24463
kousik
Removed codepath for Python < 3.6 from test functions #11949
r25246 print(cell)
else:
iprc(cell)
Matthias Bussonnier
Prototype async REPL using IPython, take III...
r24463
kousik
Removed codepath for Python < 3.6 from test functions #11949
r25246 def test_nonlocal(self):
# fails if outer scope is not a function scope or if var not defined
with self.assertRaises(SyntaxError):
iprc("nonlocal x")
iprc("""
x = 1
def f():
nonlocal x
x = 10000
yield x
Matthias Bussonnier
Test that using wrong runner/coroutine pair does not crash....
r24496 """)
kousik
Removed codepath for Python < 3.6 from test functions #11949
r25246 iprc("""
def f():
def g():
nonlocal x
x = 10000
yield x
Matthias Bussonnier
Test that using wrong runner/coroutine pair does not crash....
r24496 """)
kousik
Removed codepath for Python < 3.6 from test functions #11949
r25246 # works if outer scope is a function scope and var exists
iprc("""
def f():
x = 20
def g():
nonlocal x
x = 10000
yield x
""")
def test_execute(self):
iprc("""
import asyncio
await asyncio.sleep(0.001)
"""
)
def test_autoawait(self):
iprc("%autoawait False")
iprc("%autoawait True")
iprc("""
from asyncio import sleep
await sleep(0.1)
"""
)
farisachugthai
BUG: Fix mypy error...
r26040
Matthias Bussonnier
MAINT: remove support and testing on Python 3.8...
r28219 def test_memory_error(self):
"""
The pgen parser in 3.8 or before use to raise MemoryError on too many
nested parens anymore"""
iprc("(" * 200 + ")" * 200)
yangyang
add test
r25430
Matthias Bussonnier
Drop Python 3.9...
r28529 @pytest.mark.xfail(reason="fail on curio 1.6 and before on Python 3.12")
@pytest.mark.skip(
reason="skip_without(curio) fails on 3.12 for now even with other skip so must uncond skip"
)
# @skip_without("curio")
kousik
Removed codepath for Python < 3.6 from test functions #11949
r25246 def test_autoawait_curio(self):
iprc("%autoawait curio")
@skip_without('trio')
def test_autoawait_trio(self):
iprc("%autoawait trio")
@skip_without('trio')
def test_autoawait_trio_wrong_sleep(self):
iprc("%autoawait trio")
res = iprc_nr("""
import asyncio
await asyncio.sleep(0)
""")
Samuel Gaist
[core][tests][async_helpers] Remove nose
r26887 with self.assertRaises(TypeError):
kousik
Removed codepath for Python < 3.6 from test functions #11949
r25246 res.raise_error()
@skip_without('trio')
def test_autoawait_asyncio_wrong_sleep(self):
iprc("%autoawait asyncio")
res = iprc_nr("""
import trio
await trio.sleep(0)
""")
Samuel Gaist
[core][tests][async_helpers] Remove nose
r26887 with self.assertRaises(RuntimeError):
kousik
Removed codepath for Python < 3.6 from test functions #11949
r25246 res.raise_error()
def tearDown(self):
ip.loop_runner = "asyncio"