uGreen is an implementation of green threads on top of the :doc:`uWSGI async platform<Async>`.
It is very similar to Python's greenlet but built on top of the POSIX swapcontext()
function. To take advantage of uGreen you have to set the number of async cores that will be mapped to green threads.
For example if you want to spawn 30 green threads:
./uwsgi -w tests.cpubound_green -s :3031 --async 30 --ugreen
The ugreen
option will enable uGreen on top of async mode.
Now when you call :py:func:`uwsgi.suspend` in your app, you'll be switched off to another green thread.
To ensure (relative) isolation of green threads, every stack area is protected by so called "guard pages".
An attempt to write out of the stack area of a green thread will result in a segmentation fault/bus error (and the process manager, if enabled, will respawn the worker without too much damage).
The context switch is very fast, we can see it as:
- On switch
- Save the Python Frame pointer
- Save the recursion depth of the Python environment (it is simply an int)
- Switch to the main stack
- On return
- Re-set the uGreen stack
- Re-set the recursion depth
- Re-set the frame pointer
The stack/registers switch is done by the POSIX swapcontext()
call and we don't have to worry about it.
For managing async I/O you can use the Async mode FD wait functions :py:func:`uwsgi.wait_fd_read` and :py:func:`uwsgi.wait_fd_write`.
You can choose the uGreen stack size using the ugreen-stacksize <pages>
option. The argument is in pages, not bytes.
Weeeeelll... it depends. uGreen is faster (the stack is preallocated) but requires more memory (to allocate a stack area for every core). Stackless and Greenlet probably require less memory... but Stackless requires a heavily patched version of Python.
If you're heavily invested in making your app as async-snappy as possible, it's always best to do some tests to choose the best one for you. As far as uWSGI is concerned, you can move from async engine to another without changing your code.
Lots of uGreen has been inspired by it. The author's way to map Python threads to their implementation allows python-coev
to be a little more "trustworthy" than Stackless Python. However, like Stackless, it requires a patched version of Python... :(
Yeah! Sure! Go ahead. In the distribution you will find the ugreenchat.py
script. It is a simple/dumb multiuser Comet chat. If you want to test it (for example 30 users) run it with
./uwsgi -s :3031 -w ugreenchat --async 30 --ugreen
The code has comments for every ugreen-related line. You'll need Bottle, an amazing Python web micro framework to use it.
uGreen can benefit from the new psycopg2 async extensions and the psycogreen project. See the :file:`tests/psycopg2_green.py` and :file:`tests/psycogreen_green.py` files for examples.