Skip to content

gsmecher/awaitless

Folders and files

NameName
Last commit message
Last commit date

Latest commit

 

History

29 Commits
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 

Repository files navigation

Awaitless

If you're writing network-oriented code in Python, you're probably looking at an asynchronous framework of some kind. Let's say you're also writing code that's intended to be used interactively in IPython or Jupyter notebooks.

(Why would you ever use async interactively in IPython/JupyterLab? There's a longer screed associated with an earlier attempt here.)

That's a drag, because async is irrelevant for interactive use -- except you need to sprinkle "await" and "async" keywords in exactly the right places or it doesn't work.

For example, let's pretend "release_kraken" is a complicated piece of async machinery. It might take arguments, interacts with the world, and might return a value. Here's a placeholder definition:

>>> async def release_kraken():
...     import aiohttp, http
...     async with aiohttp.ClientSession() as cs:
...         resp = await cs.post('http://httpbin.org/post')
...         return resp.status == http.HTTPStatus.OK

Let's say you're calling release_kraken() in an interactive ipython session:

>>> release_kraken() # doctest: +SKIP
Out[1]: <coroutine object release_kraken ...>

The kraken is not released - your function never actually executes. All you get is a "RuntimeWarning: coroutine was never awaited" slap on the wrist. You forgot the "await":

>>> await release_kraken()
Out[1]: True

That's ... better? I mean, it's now "correct" according to asyncio conventions, and we're making use of ipython's autoawait magic, but there is never any ambiguity about a coroutine at the top level of an interactive session. The user wanted to run some code, and forcing them to slavishly type "await" every time is so pedantic it's user-hostile.

Let's do better:

>>> %load_ext awaitless
>>> release_kraken()
Out[1]: ...<Task finished ... result=True>

What did that do?

  • The async code ran to completion, even though we didn't await it (note the 'finished' and 'result')
  • The return value is now a Task, not a coroutine(). However, both coroutines and Tasks are awaitable, so the switcheroo is (largely) API compatible.

Why?

Yes, "await" is only 6 extra keys to hit, but it's distracting and unnecessary. You could have the same discussion about print()ing the results in a REPL rather than just displaying them -- and ipython cares enough about this to make it configurable. (Try get_ipython().ast_node_interactivity='all' if you're curious.)

I find myself returning to this article every year or so. The author says:

I ain't gonna mock Tcl-scriptable tools no more. I understand what made the authors choose Tcl, and no, it's not just a bad habit. On a level, they chose probably the best thing available today in terms of ease-of-programming/ease-of-use trade-off (assuming they bundle an interactive command shell). If I need to automate something related to those tools, I'll delve into it more happily than previously.

In short: languages designed for interactive use (tcl, bash, etc) make syntax decisions that are different than languages that are designed for programming. Interactive languages tend to prioritize "shallow" tasks like command invocation at the expense of language composition (functions, classes, modules). IPython is a reasonable compromise - but not when coroutines are involved.

Installing

Something like:

$ pip install awaitless

In IPython or JupyterLab, you can now run

In [1]: %load_ext awaitless

You can make this setting persistent by following the instructions here.

Testing

Run the following:

awaitless$ pytest

About

ipython + REPL + coroutines - suffering

Resources

License

Stars

Watchers

Forks

Packages

No packages published

Languages