##// END OF EJS Templates
reformat with black
Matthias Bussonnier -
Show More
@@ -1,90 +1,95 b''
1 """
1 """
2 Async helper function that are invalid syntax on Python 3.5 and below.
2 Async helper function that are invalid syntax on Python 3.5 and below.
3
3
4 Known limitation and possible improvement.
4 Known limitation and possible improvement.
5
5
6 Top level code that contain a return statement (instead of, or in addition to
6 Top level code that contain a return statement (instead of, or in addition to
7 await) will be detected as requiring being wrapped in async calls. This should
7 await) will be detected as requiring being wrapped in async calls. This should
8 be prevented as early return will not work.
8 be prevented as early return will not work.
9 """
9 """
10
10
11
11
12
13 import ast
12 import ast
14 import sys
13 import sys
15 import inspect
14 import inspect
16 from textwrap import dedent, indent
15 from textwrap import dedent, indent
17 from types import CodeType
16 from types import CodeType
18
17
19
18
20 def _asyncio_runner(coro):
19 def _asyncio_runner(coro):
21 """
20 """
22 Handler for asyncio autoawait
21 Handler for asyncio autoawait
23 """
22 """
24 import asyncio
23 import asyncio
24
25 return asyncio.get_event_loop().run_until_complete(coro)
25 return asyncio.get_event_loop().run_until_complete(coro)
26
26
27
27
28 def _curio_runner(coroutine):
28 def _curio_runner(coroutine):
29 """
29 """
30 handler for curio autoawait
30 handler for curio autoawait
31 """
31 """
32 import curio
32 import curio
33
33 return curio.run(coroutine)
34 return curio.run(coroutine)
34
35
35
36
36 if sys.version_info > (3, 5):
37 if sys.version_info > (3, 5):
37 # nose refuses to avoid this file and async def is invalidsyntax
38 # nose refuses to avoid this file and async def is invalidsyntax
38 s = dedent('''
39 s = dedent(
40 '''
39 def _trio_runner(function):
41 def _trio_runner(function):
40 import trio
42 import trio
41 async def loc(coro):
43 async def loc(coro):
42 """
44 """
43 We need the dummy no-op async def to protect from
45 We need the dummy no-op async def to protect from
44 trio's internal. See https://github.com/python-trio/trio/issues/89
46 trio's internal. See https://github.com/python-trio/trio/issues/89
45 """
47 """
46 return await coro
48 return await coro
47 return trio.run(loc, function)
49 return trio.run(loc, function)
48 ''')
50 '''
51 )
49 exec(s, globals(), locals())
52 exec(s, globals(), locals())
50
53
51
54
52 def _asyncify(code: str) -> str:
55 def _asyncify(code: str) -> str:
53 """wrap code in async def definition.
56 """wrap code in async def definition.
54
57
55 And setup a bit of context to run it later.
58 And setup a bit of context to run it later.
56 """
59 """
57 res = dedent("""
60 res = dedent(
61 """
58 async def __wrapper__():
62 async def __wrapper__():
59 try:
63 try:
60 {usercode}
64 {usercode}
61 finally:
65 finally:
62 locals()
66 locals()
63 """).format(usercode=indent(code, ' ' * 8)[8:])
67 """
68 ).format(usercode=indent(code, " " * 8)[8:])
64 return res
69 return res
65
70
66
71
67 def _should_be_async(cell: str) -> bool:
72 def _should_be_async(cell: str) -> bool:
68 """Detect if a block of code need to be wrapped in an `async def`
73 """Detect if a block of code need to be wrapped in an `async def`
69
74
70 Attempt to parse the block of code, it it compile we're fine.
75 Attempt to parse the block of code, it it compile we're fine.
71 Otherwise we wrap if and try to compile.
76 Otherwise we wrap if and try to compile.
72
77
73 If it works, assume it should be async. Otherwise Return False.
78 If it works, assume it should be async. Otherwise Return False.
74
79
75 Not handled yet: If the block of code has a return statement as the top
80 Not handled yet: If the block of code has a return statement as the top
76 level, it will be seen as async. This is a know limitation.
81 level, it will be seen as async. This is a know limitation.
77 """
82 """
78
83
79 try:
84 try:
80 # we can't limit ourself to ast.parse, as it __accepts__ to parse on
85 # we can't limit ourself to ast.parse, as it __accepts__ to parse on
81 # 3.7+, but just does not _compile_
86 # 3.7+, but just does not _compile_
82 compile(cell, '<>', 'exec')
87 compile(cell, "<>", "exec")
83 return False
88 return False
84 except SyntaxError:
89 except SyntaxError:
85 try:
90 try:
86 ast.parse(_asyncify(cell))
91 ast.parse(_asyncify(cell))
87 except SyntaxError:
92 except SyntaxError:
88 return False
93 return False
89 return True
94 return True
90 return False
95 return False
General Comments 0
You need to be logged in to leave comments. Login now