11

I am trying to learn about asyncio for a websockets client. Every piece of code I try gets the following error:

RuntimeError: asyncio.run() cannot be called from a running event loop

I have tried the most simple code and it always gives that RuntimeError. I tried installing the full anaconda distribution again, etc, and can´t find what the problem might be.

I am using Spyder 3.3.3 with Python 3.7.3

An example of code that should work:

import asyncio

async def main():
    print('hello')
    await asyncio.sleep(1)
    print('world')

asyncio.run(main())

Error message:

File "C:\Users\jmart\Anaconda3\lib\site-packages\spyder_kernels\customize\spydercustomize.py", line 786, in runfile
  execfile(filename, namespace)
File "C:\Users\jmart\Anaconda3\lib\site-packages\spyder_kernels\customize\spydercustomize.py", line 110, in execfile
  exec(compile(f.read(), filename, 'exec'), namespace)
File "C:/Users/jmart/Documents/asynk2.py", line 8, in <module>
  asyncio.run(main())
File "C:\Users\jmart\Anaconda3\lib\asyncio\runners.py", line 34, in run
  "asyncio.run() cannot be called from a running event loop")
RuntimeError: asyncio.run() cannot be called from a running event loop
Gino Mempin
  • 19,150
  • 23
  • 79
  • 104
Javier Martin
  • 123
  • 1
  • 2
  • 8

2 Answers2

17

It's a known problem related to IPython.

One way as you already found is to use nest_asyncio:

import nest_asyncio
nest_asyncio.apply()

The other one is to install older version of tornado:

pip3 install tornado==4.5.3
Mikhail Gerasimov
  • 32,558
  • 15
  • 103
  • 148
  • 1
    The first solution seems to give some running problems for me, installing the older version of tornado seems to work perfectly! – Javier Martin May 15 '19 at 20:39
  • 1
    Beware, nested run calls will starve out loops. This monkey patch is very fragile on python / stdlib upgrades. – Blaze Oct 22 '20 at 13:50
  • 1
    Is there any reason for not simply using [`await main()`](https://stackoverflow.com/a/55409674/1720199) with spyder? – cglacet Apr 05 '21 at 19:11
5

Problem origin

Spyder runs its own event loop

print(asyncio.get_running_loop().is_running()) 
Returns: True

but only one is allowed per thread

cannot be called when another asyncio event loop is running in the same thread

That is why, when we try to create a new event loop with
asyncio.run(main()) it gives us an error:
RuntimeError: asyncio.run() cannot be called from a running event loop

Solution

Except what already proposed with nest_asyncio and tornado I came up with

  1. Attach to an existing Spyder thread event loop by creating a new task
import asyncio

async def main():
    print('Hello world!')

asyncio.create_task(main())
  1. Create new thread (by executing in external terminal) which allows to run our own event loop.
    Top menu Run -> Run configuration per file... -> Execute in the external system terminal enter image description here

Now the code runs in new terminal and works

import asyncio

async def main():
    print('Hello world!')
asyncio.run(main())

Stepan
  • 464
  • 5
  • 12
  • This is definetly the better answer. Perfect would be a interpreter-independent version with `try: ....create_task(main()) except RuntimeError: asyncio.run(main())`. This works fine in Spyder, but still produces some warnings in python shell :( – PythoNic Feb 17 '21 at 02:43