Server green modes

PyTango server API from version 9.2.0 supports two green modes: Gevent and Asyncio. Both can be used in writing new device servers in an asynchronous way.

Note

If your device server has multiple devices they must all use the same green mode.

Warning

These green modes disable Tango’s device server serialisation, i.e., tango.SerialModel.NO_SYNC is automatically passed to tango.Util.set_serial_model(), when the device server starts. From those docs: “This is an exotic kind of serialization and should be used with extreme care only with devices which are fully thread safe.”

gevent mode

This mode lets you convert your existing devices to asynchronous devices easily. You just add green_mode = tango.GreenMode.Gevent line to your device class. Consider this example:

class GeventDevice(Device):
    green_mode = tango.GreenMode.Gevent

Every method in your device class will be treated as a coroutine implicitly. This can be beneficial, but also potentially dangerous as it is a lot harder to debug. You should use this green mode with care. Gevent green mode is useful when you don’t want to change too much in your existing code (or you don’t feel comfortable with writing syntax of asynchronous calls).

Another thing to keep in mind is that when using Gevent green mode is that the Tango monitor lock is disabled, so the client requests can be processed concurrently.

Greenlets can also be used to spawn tasks in the background.

asyncio mode

The way asyncio green mode on the server side works is it redirects all user code to an event loop. This means that all user methods become coroutines, so in Python > 3.5 you should define them with async keyword. This also means that in order to convert existing code of your devices to Asyncio green mode you will have to introduce at least those changes. But, of course, to truly benefit from this green mode (and asynchronous approach in general), you should introduce more far-fetched changes!

The main benefit of asynchronous programing approach is that it lets you control precisely when code is run sequentially without interruptions and when control can be given back to the event loop. It’s especially useful if you want to perform some long operations and don’t want to prevent clients from accessing other parts of your device (attributes, in particular). This means that in Asyncio green mode there is no monitor lock!

The example below shows how asyncio can be used to write an asynchronous Tango device:

 1"""Demo Tango Device Server using asyncio green mode"""
 2
 3import asyncio
 4from tango import DevState, GreenMode
 5from tango.server import Device, command, attribute
 6
 7
 8class AsyncioDevice(Device):
 9    green_mode = GreenMode.Asyncio
10
11    async def init_device(self):
12        await super().init_device()
13        self.set_state(DevState.ON)
14
15    @command
16    async def long_running_command(self):
17        self.set_state(DevState.OPEN)
18        await asyncio.sleep(2)
19        self.set_state(DevState.CLOSE)
20
21    @command
22    async def background_task_command(self):
23        loop = asyncio.get_event_loop()
24        future = loop.create_task(self.coroutine_target())
25
26    async def coroutine_target(self):
27        self.set_state(DevState.INSERT)
28        await asyncio.sleep(15)
29        self.set_state(DevState.EXTRACT)
30
31    @attribute
32    async def test_attribute(self):
33        await asyncio.sleep(2)
34        return 42
35
36
37if __name__ == "__main__":
38    AsyncioDevice.run_server()