How to access the asyncio / uvloop loop from C

1.1k Views Asked by At

I'm completely new to python, but have an async python app using uvloop which uses a C api module I created which also needs access to the async loop.

1) asyncio does not yet have a c-api for this? Any hacks to get an event loop usable in C? Is this being discussed anywhere?

2) uvloop uses libuv which I am familiar with in C. If I can grab the uv_loop_t pointer I could hook into the loop. I assume I can either:

A) With a PyObject * to uvloop's loop calculate the offset to the uv_loop_t* and use that? Assuming I knew the length of PyObject_HEAD?

libuv_loop = (uv_loop_t*)((void*)(loop)+0x8);

struct __pyx_obj_6uvloop_4loop_Loop {
    PyObject_HEAD
    uv_loop_t *uvloop;

B) Or non hacky modify uvloop to expose the loop pointer. I'm completely clueless here as I've never looked at cython code. Can I create a python function on the loop, call it from my C code and get the C pointer? Like:

(uv_loop_t*)PyObject_CallFunctionObjArgs( getLoop, NULL )

By adding getLoop to here:

https://github.com/MagicStack/uvloop/blob/master/uvloop/loop.pyx

cdef uv.uv_loop_t* _getLoop(self):
    return self.uvloop
2

There are 2 best solutions below

1
On

asyncio has no C API yet.

We have a plan for adding it in future Python versions (3.8 maybe).

Right now you should use PyObject_* api.

uvloop is written in Cython but the library has no Public C API as well. You may access private uvloop API but exposed function names and data structures can be changed in any moment without public notice because they are considered private, users should never use it.

0
On

Was looking for this too, and coincidentally, it just so happens that uvloop added a loop.get_uv_loop_t_ptr() method a few days ago :)

https://github.com/MagicStack/uvloop/pull/310

Now we just have to wait for a new release (v0.17 ?) that includes this PR (or build it ourselves).