1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 245 246 247 248 249 250 251 252 253 254 255 256 257 258 259 260 261 262 263 264 265 266 267 268 269 270 271 272
|
# Client green modes
You can also change the active global green mode at any time in your program:
```
>>> from tango import DeviceProxy, GreenMode
>>> from tango import set_green_mode, get_green_mode
>>> get_green_mode()
tango.GreenMode.Synchronous
>>> dev = DeviceProxy("sys/tg_test/1")
>>> dev.get_green_mode()
tango.GreenMode.Synchronous
>>> set_green_mode(GreenMode.Futures)
>>> get_green_mode()
tango.GreenMode.Futures
>>> dev.get_green_mode()
tango.GreenMode.Futures
```
As you can see by the example, the global green mode will affect any previously
created {class}`DeviceProxy` using the default *DeviceProxy* constructor
parameters.
You can specificy green mode on a {class}`DeviceProxy` at creation time.
You can also change the green mode at any time:
```
>>> from tango.futures import DeviceProxy
>>> dev = DeviceProxy("sys/tg_test/1")
>>> dev.get_green_mode()
tango.GreenMode.Futures
>>> dev.set_green_mode(GreenMode.Synchronous)
>>> dev.get_green_mode()
tango.GreenMode.Synchronous
```
## futures mode
Using {mod}`concurrent.futures` cooperative mode in PyTango is relatively easy:
```
>>> from tango.futures import DeviceProxy
>>> dev = DeviceProxy("sys/tg_test/1")
>>> dev.get_green_mode()
tango.GreenMode.Futures
>>> print(dev.state())
RUNNING
```
The {func}`tango.futures.DeviceProxy` API is exactly the same as the standard
{class}`~tango.DeviceProxy`. The difference is in the semantics of the methods
that involve synchronous network calls (constructor included) which may block
the execution for a relatively big amount of time.
The list of methods that have been modified to accept *futures* semantics are,
on the {func}`tango.futures.DeviceProxy`:
- Constructor
- {meth}`~DeviceProxy.state`
- {meth}`~DeviceProxy.status`
- {meth}`~DeviceProxy.read_attribute`
- {meth}`~DeviceProxy.write_attribute`
- {meth}`~DeviceProxy.write_read_attribute`
- {meth}`~DeviceProxy.read_attributes`
- {meth}`~DeviceProxy.write_attributes`
- {meth}`~DeviceProxy.ping`
So how does this work in fact? I see no difference from using the *standard*
{class}`~tango.DeviceProxy`.
Well, this is, in fact, one of the goals: be able to use a *futures* cooperation
without changing the API. Behind the scenes the methods mentioned before have
been modified to be able to work cooperatively.
All of the above methods have been boosted with two extra keyword arguments
*wait* and *timeout* which allow to fine tune the behaviour.
The *wait* parameter is by default set to `True` meaning wait for the request
to finish (the default semantics when not using green mode).
If *wait* is set to `True`, the timeout determines the maximum time to wait for
the method to execute. The default is `None` which means wait forever. If *wait*
is set to `False`, the *timeout* is ignored.
If *wait* is set to `True`, the result is the same as executing the
*standard* method on a {class}`~tango.DeviceProxy`.
If, *wait* is set to `False`, the result will be a
{class}`concurrent.futures.Future`. In this case, to get the actual value
you will need to do something like:
```
>>> from tango.futures import DeviceProxy
>>> dev = DeviceProxy("sys/tg_test/1")
>>> result = dev.state(wait=False)
>>> result
<Future at 0x16cb310 state=pending>
>>> # this will be the blocking code
>>> state = result.result()
>>> print(state)
RUNNING
```
Here is another example using {meth}`~DeviceProxy.read_attribute`:
```
>>> from tango.futures import DeviceProxy
>>> dev = DeviceProxy("sys/tg_test/1")
>>> result = dev.read_attribute('wave', wait=False)
>>> result
<Future at 0x16cbe50 state=pending>
>>> dev_attr = result.result()
>>> print(dev_attr)
DeviceAttribute[
data_format = tango.AttrDataFormat.SPECTRUM
dim_x = 256
dim_y = 0
has_failed = False
is_empty = False
name = 'wave'
nb_read = 256
nb_written = 0
quality = tango.AttrQuality.ATTR_VALID
r_dimension = AttributeDimension(dim_x = 256, dim_y = 0)
time = TimeVal(tv_nsec = 0, tv_sec = 1383923329, tv_usec = 451821)
type = tango.CmdArgType.DevDouble
value = array([ -9.61260664e-01, -9.65924853e-01, -9.70294813e-01,
-9.74369212e-01, -9.78146810e-01, -9.81626455e-01,
-9.84807087e-01, -9.87687739e-01, -9.90267531e-01,
...
5.15044507e-1])
w_dim_x = 0
w_dim_y = 0
w_dimension = AttributeDimension(dim_x = 0, dim_y = 0)
w_value = None]
```
## gevent mode
:::{warning}
Before using gevent mode please note that at the time of writing this
documentation, *tango.gevent* requires the latest version 1.0 of
gevent (which has been released the day before :-P).
:::
Using [gevent](https://www.gevent.org) cooperative mode in PyTango is relatively easy:
```
>>> from tango.gevent import DeviceProxy
>>> dev = DeviceProxy("sys/tg_test/1")
>>> dev.get_green_mode()
tango.GreenMode.Gevent
>>> print(dev.state())
RUNNING
```
The {func}`tango.gevent.DeviceProxy` API is exactly the same as the standard
{class}`~tango.DeviceProxy`. The difference is in the semantics of the methods
that involve synchronous network calls (constructor included) which may block
the execution for a relatively big amount of time.
The list of methods that have been modified to accept *gevent* semantics are,
on the {func}`tango.gevent.DeviceProxy`:
- Constructor
- {meth}`~DeviceProxy.state`
- {meth}`~DeviceProxy.status`
- {meth}`~DeviceProxy.read_attribute`
- {meth}`~DeviceProxy.write_attribute`
- {meth}`~DeviceProxy.write_read_attribute`
- {meth}`~DeviceProxy.read_attributes`
- {meth}`~DeviceProxy.write_attributes`
- {meth}`~DeviceProxy.ping`
So how does this work in fact? I see no difference from using the *standard*
{class}`~tango.DeviceProxy`.
Well, this is, in fact, one of the goals: be able to use a gevent cooperation
without changing the API. Behind the scenes the methods mentioned before have
been modified to be able to work cooperatively with other greenlets.
All of the above methods have been boosted with two extra keyword arguments
*wait* and *timeout* which allow to fine tune the behaviour.
The *wait* parameter is by default set to `True` meaning wait for the request
to finish (the default semantics when not using green mode).
If *wait* is set to `True`, the timeout determines the maximum time to wait for
the method to execute. The default timeout is `None` which means wait forever.
If *wait* is set to `False`, the *timeout* is ignored.
If *wait* is set to `True`, the result is the same as executing the
*standard* method on a {class}`~tango.DeviceProxy`.
If, *wait* is set to `False`, the result will be a
{class}`gevent.event.AsyncResult`. In this case, to get the actual value
you will need to do something like:
```
>>> from tango.gevent import DeviceProxy
>>> dev = DeviceProxy("sys/tg_test/1")
>>> result = dev.state(wait=False)
>>> result
<gevent.event.AsyncResult at 0x1a74050>
>>> # this will be the blocking code
>>> state = result.get()
>>> print(state)
RUNNING
```
Here is another example using {meth}`~DeviceProxy.read_attribute`:
```
>>> from tango.gevent import DeviceProxy
>>> dev = DeviceProxy("sys/tg_test/1")
>>> result = dev.read_attribute('wave', wait=False)
>>> result
<gevent.event.AsyncResult at 0x1aff54e>
>>> dev_attr = result.get()
>>> print(dev_attr)
DeviceAttribute[
data_format = tango.AttrDataFormat.SPECTRUM
dim_x = 256
dim_y = 0
has_failed = False
is_empty = False
name = 'wave'
nb_read = 256
nb_written = 0
quality = tango.AttrQuality.ATTR_VALID
r_dimension = AttributeDimension(dim_x = 256, dim_y = 0)
time = TimeVal(tv_nsec = 0, tv_sec = 1383923292, tv_usec = 886720)
type = tango.CmdArgType.DevDouble
value = array([ -9.61260664e-01, -9.65924853e-01, -9.70294813e-01,
-9.74369212e-01, -9.78146810e-01, -9.81626455e-01,
-9.84807087e-01, -9.87687739e-01, -9.90267531e-01,
...
5.15044507e-1])
w_dim_x = 0
w_dim_y = 0
w_dimension = AttributeDimension(dim_x = 0, dim_y = 0)
w_value = None]
```
:::{note}
due to the internal workings of gevent, setting the *wait* flag to
`True` (default) doesn't prevent other greenlets from running in *parallel*.
This is, in fact, one of the major bonus of working with {mod}`gevent` when
compared with {mod}`concurrent.futures`
:::
## asyncio mode
[Asyncio](https://docs.python.org/3/library/asyncio.html) mode is similar to gevent, but it uses explicit coroutines. You can compare gevent and asyncio examples.
```{literalinclude} ../../../examples/asyncio_green_mode/asyncio_simple_example.py
:linenos: true
```
Below you can find a TCP server example, which runs in an asynchronous mode and waits for a device's attribute name from a TCP client, then asks the device for a value and replies to the TCP client.
```{literalinclude} ../../../examples/asyncio_green_mode/tcp_server_example.py
:linenos: true
```
|