-
-
Notifications
You must be signed in to change notification settings - Fork 178
Coroutine with asyncio.sleep() hangs when run in a different thread #312
Comments
Event loop is coupled to thread. You cannot share the loop between different threads, except See https://docs.python.org/3/library/asyncio-dev.html#concurrency-and-multithreading You can have a loop per thread though. |
Oh right, I think I remember reading a while back that |
Assuming that the child thread has to create the event loop, here is updated code that creates a new event loop and runs it in the child thread. It still has the same hanging problem as before. import asyncio
from threading import Thread, Semaphore
async def onRequest(data):
print('got request: {}'.format(data))
await asyncio.sleep(1)
print('finished processing request: {}'.format(data))
loop_created_sem = Semaphore(0)
def child_executor():
global loop
loop = asyncio.new_event_loop()
# signal to the main thread that we've created the event loop
loop_created_sem.release()
loop.run_forever()
child = Thread(target=child_executor, name="child")
def request(data):
global loop
asyncio.run_coroutine_threadsafe(onRequest(data), loop)
child.start()
# wait for the child thread to create its event loop
# before we advance
loop_created_sem.acquire()
request(4) Expected Output:
Output (control^C pressed after 10 seconds)
|
That's because you don't pass explicit event loop to As an option you may setup thread-local loop:
|
I think it's one of most unclear things about asyncio :) Just started a discussion issue #332 related to that. |
see also bugs.python.org/issue26969 |
py.test asyncio plugin has a feature that detects the frowned-upon use of the default event loop |
Honestly I recommend to newer use implicit event loop in your code but always pass Moreover global implicit loop should be disabled by |
On 18 July 2016 at 22:46, Andrew Svetlov notifications@github.com wrote:
Gustavo J. A. M. Carneiro |
As I understand it, one should set a customized event loop policy globally, The thing is that the documentation is (or used to be) light on the topic, 2016-07-19 11:44 GMT+02:00 Gustavo J. A. M. Carneiro <
Martin http://www.martiusweb.net Richard |
I wanted to mention that in OP's code, the actual exception is caught and set in the
Regarding the proper way to handle the event loop, my recommendation for users would be to never pass it explicitly and always rely on
My opinion is that passing an event loop around might make sense for libraries to be unit-testable, but it is annoying and unnecessary for user code. |
@vxgmichel do you mean that non-libraries should not be unit-testable? |
@asvetlov My mistake, that's not what I meant. Relying on
Even
I can only see two issues with a library not supporting explicit loop passing:
For small libraries, those two points are probably not worth the trouble of passing the event loop around. But I guess it is also a matter of taste. |
Thanks to the updated behaviour of |
read more here python/asyncio#312 and here python/asyncio#332
Here is my code:
This produces the following output:
and then it just hangs, without the 'finished processing request' message. If I press ctrl^C, it outputs the following message, which doesn't look very interesting:
The text was updated successfully, but these errors were encountered: