cgse 2023.38.0__py3-none-any.whl → 2024.1.3__py3-none-any.whl

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (653) hide show
  1. README.md +27 -0
  2. bump.py +77 -0
  3. cgse-2024.1.3.dist-info/METADATA +41 -0
  4. cgse-2024.1.3.dist-info/RECORD +5 -0
  5. {cgse-2023.38.0.dist-info → cgse-2024.1.3.dist-info}/WHEEL +1 -2
  6. cgse-2023.38.0.dist-info/COPYING +0 -674
  7. cgse-2023.38.0.dist-info/COPYING.LESSER +0 -165
  8. cgse-2023.38.0.dist-info/METADATA +0 -144
  9. cgse-2023.38.0.dist-info/RECORD +0 -649
  10. cgse-2023.38.0.dist-info/entry_points.txt +0 -75
  11. cgse-2023.38.0.dist-info/top_level.txt +0 -2
  12. egse/__init__.py +0 -12
  13. egse/__main__.py +0 -32
  14. egse/aeu/aeu.py +0 -5235
  15. egse/aeu/aeu_awg.yaml +0 -265
  16. egse/aeu/aeu_crio.yaml +0 -273
  17. egse/aeu/aeu_cs.py +0 -626
  18. egse/aeu/aeu_devif.py +0 -321
  19. egse/aeu/aeu_main_ui.py +0 -912
  20. egse/aeu/aeu_metrics.py +0 -131
  21. egse/aeu/aeu_protocol.py +0 -463
  22. egse/aeu/aeu_psu.yaml +0 -204
  23. egse/aeu/aeu_ui.py +0 -873
  24. egse/aeu/arbdata/FccdRead.arb +0 -2
  25. egse/aeu/arbdata/FccdRead_min_points.arb +0 -2
  26. egse/aeu/arbdata/HeaterSync_FccdRead.arb +0 -2
  27. egse/aeu/arbdata/HeaterSync_ccdRead25.arb +0 -2
  28. egse/aeu/arbdata/HeaterSync_ccdRead31_25.arb +0 -2
  29. egse/aeu/arbdata/HeaterSync_ccdRead37_50.arb +0 -2
  30. egse/aeu/arbdata/HeaterSync_ccdRead43_75.arb +0 -2
  31. egse/aeu/arbdata/HeaterSync_ccdRead50.arb +0 -2
  32. egse/aeu/arbdata/Heater_FccdRead_min_points.arb +0 -2
  33. egse/aeu/arbdata/ccdRead25.arb +0 -2
  34. egse/aeu/arbdata/ccdRead25_150ms.arb +0 -2
  35. egse/aeu/arbdata/ccdRead31_25.arb +0 -2
  36. egse/aeu/arbdata/ccdRead31_25_150ms.arb +0 -2
  37. egse/aeu/arbdata/ccdRead37_50.arb +0 -2
  38. egse/aeu/arbdata/ccdRead37_50_150ms.arb +0 -2
  39. egse/aeu/arbdata/ccdRead43_75.arb +0 -2
  40. egse/aeu/arbdata/ccdRead43_75_150ms.arb +0 -2
  41. egse/aeu/arbdata/ccdRead50.arb +0 -2
  42. egse/aeu/arbdata/ccdRead50_150ms.arb +0 -2
  43. egse/alert/__init__.py +0 -1049
  44. egse/alert/alertman.yaml +0 -37
  45. egse/alert/alertman_cs.py +0 -234
  46. egse/alert/alertman_ui.py +0 -603
  47. egse/alert/gsm/beaglebone.py +0 -138
  48. egse/alert/gsm/beaglebone.yaml +0 -51
  49. egse/alert/gsm/beaglebone_cs.py +0 -108
  50. egse/alert/gsm/beaglebone_devif.py +0 -130
  51. egse/alert/gsm/beaglebone_protocol.py +0 -48
  52. egse/bits.py +0 -318
  53. egse/camera.py +0 -44
  54. egse/collimator/__init__.py +0 -0
  55. egse/collimator/fcul/__init__.py +0 -0
  56. egse/collimator/fcul/ogse.py +0 -1077
  57. egse/collimator/fcul/ogse.yaml +0 -14
  58. egse/collimator/fcul/ogse_cs.py +0 -154
  59. egse/collimator/fcul/ogse_devif.py +0 -358
  60. egse/collimator/fcul/ogse_protocol.py +0 -129
  61. egse/collimator/fcul/ogse_sim.py +0 -431
  62. egse/collimator/fcul/ogse_ui.py +0 -1108
  63. egse/command.py +0 -699
  64. egse/config.py +0 -410
  65. egse/confman/__init__.py +0 -1015
  66. egse/confman/confman.yaml +0 -67
  67. egse/confman/confman_cs.py +0 -239
  68. egse/confman/confman_ui.py +0 -381
  69. egse/confman/setup_ui.py +0 -565
  70. egse/control.py +0 -442
  71. egse/coordinates/__init__.py +0 -531
  72. egse/coordinates/avoidance.py +0 -103
  73. egse/coordinates/cslmodel.py +0 -127
  74. egse/coordinates/laser_tracker_to_dict.py +0 -120
  75. egse/coordinates/point.py +0 -707
  76. egse/coordinates/pyplot.py +0 -195
  77. egse/coordinates/referenceFrame.py +0 -1279
  78. egse/coordinates/refmodel.py +0 -737
  79. egse/coordinates/rotationMatrix.py +0 -85
  80. egse/coordinates/transform3d_addon.py +0 -419
  81. egse/csl/__init__.py +0 -50
  82. egse/csl/commanding.py +0 -78
  83. egse/csl/icons/hexapod-connected-selected.svg +0 -30
  84. egse/csl/icons/hexapod-connected.svg +0 -30
  85. egse/csl/icons/hexapod-homing-selected.svg +0 -68
  86. egse/csl/icons/hexapod-homing.svg +0 -68
  87. egse/csl/icons/hexapod-retract-selected.svg +0 -56
  88. egse/csl/icons/hexapod-retract.svg +0 -51
  89. egse/csl/icons/hexapod-zero-selected.svg +0 -56
  90. egse/csl/icons/hexapod-zero.svg +0 -56
  91. egse/csl/icons/logo-puna.svg +0 -92
  92. egse/csl/icons/stop.svg +0 -1
  93. egse/csl/initialisation.py +0 -102
  94. egse/csl/mech_pos_settings.yaml +0 -18
  95. egse/das.py +0 -1247
  96. egse/das.yaml +0 -7
  97. egse/data/conf/SETUP_CSL_00000_170620_150000.yaml +0 -5
  98. egse/data/conf/SETUP_CSL_00001_170620_151010.yaml +0 -69
  99. egse/data/conf/SETUP_CSL_00002_170620_151020.yaml +0 -69
  100. egse/data/conf/SETUP_CSL_00003_170620_151030.yaml +0 -69
  101. egse/data/conf/SETUP_CSL_00004_170620_151040.yaml +0 -69
  102. egse/data/conf/SETUP_CSL_00005_170620_151050.yaml +0 -69
  103. egse/data/conf/SETUP_CSL_00006_170620_151060.yaml +0 -69
  104. egse/data/conf/SETUP_CSL_00007_170620_151070.yaml +0 -69
  105. egse/data/conf/SETUP_CSL_00008_170620_151080.yaml +0 -75
  106. egse/data/conf/SETUP_CSL_00010_210308_083016.yaml +0 -138
  107. egse/data/conf/SETUP_INTA_00000_170620_150000.yaml +0 -4
  108. egse/data/conf/SETUP_SRON_00000_170620_150000.yaml +0 -4
  109. egse/decorators.py +0 -415
  110. egse/device.py +0 -269
  111. egse/dpu/__init__.py +0 -2681
  112. egse/dpu/ccd_ui.py +0 -508
  113. egse/dpu/dpu.py +0 -786
  114. egse/dpu/dpu.yaml +0 -153
  115. egse/dpu/dpu_cs.py +0 -272
  116. egse/dpu/dpu_ui.py +0 -668
  117. egse/dpu/fitsgen.py +0 -2077
  118. egse/dpu/fitsgen_test.py +0 -752
  119. egse/dpu/fitsgen_ui.py +0 -399
  120. egse/dpu/hdf5_model.py +0 -332
  121. egse/dpu/hdf5_ui.py +0 -277
  122. egse/dpu/hdf5_viewer.py +0 -506
  123. egse/dpu/hk_ui.py +0 -468
  124. egse/dpu_commands.py +0 -81
  125. egse/dsi/constants.py +0 -220
  126. egse/dsi/esl.py +0 -870
  127. egse/dsi/rmap.py +0 -1042
  128. egse/dsi/rmapci.py +0 -37
  129. egse/dsi/spw.py +0 -154
  130. egse/dsi/spw_state.py +0 -29
  131. egse/dummy.py +0 -258
  132. egse/dyndummy.py +0 -179
  133. egse/env.py +0 -278
  134. egse/exceptions.py +0 -88
  135. egse/fdir/__init__.py +0 -28
  136. egse/fdir/fdir_manager.py +0 -85
  137. egse/fdir/fdir_manager.yaml +0 -51
  138. egse/fdir/fdir_manager_controller.py +0 -228
  139. egse/fdir/fdir_manager_cs.py +0 -164
  140. egse/fdir/fdir_manager_interface.py +0 -25
  141. egse/fdir/fdir_remote.py +0 -73
  142. egse/fdir/fdir_remote.yaml +0 -37
  143. egse/fdir/fdir_remote_controller.py +0 -50
  144. egse/fdir/fdir_remote_cs.py +0 -97
  145. egse/fdir/fdir_remote_interface.py +0 -14
  146. egse/fdir/fdir_remote_popup.py +0 -31
  147. egse/fee/__init__.py +0 -114
  148. egse/fee/f_fee_register.yaml +0 -43
  149. egse/fee/fee.py +0 -631
  150. egse/fee/feesim.py +0 -750
  151. egse/fee/n_fee_hk.py +0 -761
  152. egse/fee/nfee.py +0 -187
  153. egse/filterwheel/__init__.py +0 -4
  154. egse/filterwheel/eksma/__init__.py +0 -24
  155. egse/filterwheel/eksma/fw8smc4.py +0 -661
  156. egse/filterwheel/eksma/fw8smc4.yaml +0 -121
  157. egse/filterwheel/eksma/fw8smc4_cs.py +0 -144
  158. egse/filterwheel/eksma/fw8smc4_devif.py +0 -473
  159. egse/filterwheel/eksma/fw8smc4_protocol.py +0 -81
  160. egse/filterwheel/eksma/fw8smc4_ui.py +0 -940
  161. egse/filterwheel/eksma/fw8smc5.py +0 -111
  162. egse/filterwheel/eksma/fw8smc5.yaml +0 -105
  163. egse/filterwheel/eksma/fw8smc5_controller.py +0 -307
  164. egse/filterwheel/eksma/fw8smc5_cs.py +0 -141
  165. egse/filterwheel/eksma/fw8smc5_interface.py +0 -65
  166. egse/filterwheel/eksma/fw8smc5_simulator.py +0 -29
  167. egse/filterwheel/eksma/fw8smc5_ui.py +0 -1068
  168. egse/filterwheel/eksma/testpythonfw.py +0 -215
  169. egse/fov/__init__.py +0 -65
  170. egse/fov/fov_hk.py +0 -712
  171. egse/fov/fov_ui.py +0 -861
  172. egse/fov/fov_ui_controller.py +0 -140
  173. egse/fov/fov_ui_model.py +0 -200
  174. egse/fov/fov_ui_view.py +0 -345
  175. egse/gimbal/__init__.py +0 -32
  176. egse/gimbal/symetrie/__init__.py +0 -26
  177. egse/gimbal/symetrie/alpha.py +0 -586
  178. egse/gimbal/symetrie/generic_gimbal_ui.py +0 -1521
  179. egse/gimbal/symetrie/gimbal.py +0 -877
  180. egse/gimbal/symetrie/gimbal.yaml +0 -168
  181. egse/gimbal/symetrie/gimbal_cs.py +0 -183
  182. egse/gimbal/symetrie/gimbal_protocol.py +0 -135
  183. egse/gimbal/symetrie/gimbal_ui.py +0 -361
  184. egse/gimbal/symetrie/pmac.py +0 -1006
  185. egse/gimbal/symetrie/pmac_regex.py +0 -83
  186. egse/graph.py +0 -132
  187. egse/gui/__init__.py +0 -47
  188. egse/gui/buttons.py +0 -378
  189. egse/gui/focalplane.py +0 -1281
  190. egse/gui/formatter.py +0 -10
  191. egse/gui/led.py +0 -162
  192. egse/gui/limitswitch.py +0 -143
  193. egse/gui/mechanisms.py +0 -588
  194. egse/gui/states.py +0 -148
  195. egse/gui/stripchart.py +0 -729
  196. egse/gui/switch.py +0 -112
  197. egse/h5.py +0 -274
  198. egse/help/__init__.py +0 -0
  199. egse/help/help_ui.py +0 -126
  200. egse/hexapod/__init__.py +0 -32
  201. egse/hexapod/symetrie/__init__.py +0 -138
  202. egse/hexapod/symetrie/alpha.py +0 -874
  203. egse/hexapod/symetrie/dynalpha.py +0 -1387
  204. egse/hexapod/symetrie/hexapod_ui.py +0 -1516
  205. egse/hexapod/symetrie/pmac.py +0 -1010
  206. egse/hexapod/symetrie/pmac_regex.py +0 -83
  207. egse/hexapod/symetrie/puna.py +0 -1167
  208. egse/hexapod/symetrie/puna.yaml +0 -193
  209. egse/hexapod/symetrie/puna_cs.py +0 -196
  210. egse/hexapod/symetrie/puna_protocol.py +0 -131
  211. egse/hexapod/symetrie/puna_ui.py +0 -434
  212. egse/hexapod/symetrie/punaplus.py +0 -107
  213. egse/hexapod/symetrie/zonda.py +0 -872
  214. egse/hexapod/symetrie/zonda.yaml +0 -337
  215. egse/hexapod/symetrie/zonda_cs.py +0 -172
  216. egse/hexapod/symetrie/zonda_devif.py +0 -415
  217. egse/hexapod/symetrie/zonda_protocol.py +0 -119
  218. egse/hexapod/symetrie/zonda_ui.py +0 -449
  219. egse/hk.py +0 -765
  220. egse/icons/aeu-cs-start.svg +0 -117
  221. egse/icons/aeu-cs-stop.svg +0 -118
  222. egse/icons/aeu-cs.svg +0 -107
  223. egse/icons/aeu_cs-started.svg +0 -112
  224. egse/icons/aeu_cs-stopped.svg +0 -112
  225. egse/icons/aeu_cs.svg +0 -55
  226. egse/icons/alert.svg +0 -1
  227. egse/icons/arrow-double-left.png +0 -0
  228. egse/icons/arrow-double-right.png +0 -0
  229. egse/icons/arrow-up.svg +0 -11
  230. egse/icons/backward.svg +0 -1
  231. egse/icons/busy.svg +0 -1
  232. egse/icons/cleaning.svg +0 -115
  233. egse/icons/color-scheme.svg +0 -1
  234. egse/icons/cs-connected-alert.svg +0 -91
  235. egse/icons/cs-connected-disabled.svg +0 -43
  236. egse/icons/cs-connected.svg +0 -89
  237. egse/icons/cs-not-connected.svg +0 -44
  238. egse/icons/double-left-arrow.svg +0 -1
  239. egse/icons/double-right-arrow.svg +0 -1
  240. egse/icons/erase-disabled.svg +0 -19
  241. egse/icons/erase.svg +0 -59
  242. egse/icons/fitsgen-start.svg +0 -47
  243. egse/icons/fitsgen-stop.svg +0 -48
  244. egse/icons/fitsgen.svg +0 -1
  245. egse/icons/forward.svg +0 -1
  246. egse/icons/fov-hk-start.svg +0 -33
  247. egse/icons/fov-hk-stop.svg +0 -37
  248. egse/icons/fov-hk.svg +0 -1
  249. egse/icons/front-desk.svg +0 -1
  250. egse/icons/home-actioned.svg +0 -15
  251. egse/icons/home-disabled.svg +0 -15
  252. egse/icons/home.svg +0 -13
  253. egse/icons/info.svg +0 -1
  254. egse/icons/invalid.png +0 -0
  255. egse/icons/led-green.svg +0 -20
  256. egse/icons/led-grey.svg +0 -20
  257. egse/icons/led-orange.svg +0 -20
  258. egse/icons/led-red.svg +0 -20
  259. egse/icons/led-square-green.svg +0 -134
  260. egse/icons/led-square-grey.svg +0 -134
  261. egse/icons/led-square-orange.svg +0 -134
  262. egse/icons/led-square-red.svg +0 -134
  263. egse/icons/limit-switch-all-green.svg +0 -115
  264. egse/icons/limit-switch-all-red.svg +0 -117
  265. egse/icons/limit-switch-el+.svg +0 -116
  266. egse/icons/limit-switch-el-.svg +0 -117
  267. egse/icons/location-marker.svg +0 -1
  268. egse/icons/logo-dpu.svg +0 -48
  269. egse/icons/logo-gimbal.svg +0 -112
  270. egse/icons/logo-huber.svg +0 -23
  271. egse/icons/logo-ogse.svg +0 -31
  272. egse/icons/logo-puna.svg +0 -92
  273. egse/icons/logo-tcs.svg +0 -29
  274. egse/icons/logo-zonda.svg +0 -66
  275. egse/icons/maximize.svg +0 -1
  276. egse/icons/meter.svg +0 -1
  277. egse/icons/more.svg +0 -45
  278. egse/icons/n-fee-hk-start.svg +0 -24
  279. egse/icons/n-fee-hk-stop.svg +0 -25
  280. egse/icons/n-fee-hk.svg +0 -83
  281. egse/icons/observing-off.svg +0 -46
  282. egse/icons/observing-on.svg +0 -46
  283. egse/icons/open-document-hdf5.png +0 -0
  284. egse/icons/open-document-hdf5.svg +0 -21
  285. egse/icons/ops-mode.svg +0 -1
  286. egse/icons/play-green.svg +0 -17
  287. egse/icons/plugged-disabled.svg +0 -27
  288. egse/icons/plugged.svg +0 -21
  289. egse/icons/pm_ui.svg +0 -1
  290. egse/icons/power-button-green.svg +0 -27
  291. egse/icons/power-button-red.svg +0 -27
  292. egse/icons/power-button.svg +0 -27
  293. egse/icons/radar.svg +0 -1
  294. egse/icons/radioactive.svg +0 -2
  295. egse/icons/reload.svg +0 -1
  296. egse/icons/remote-control-off.svg +0 -28
  297. egse/icons/remote-control-on.svg +0 -28
  298. egse/icons/repeat-blue.svg +0 -15
  299. egse/icons/repeat.svg +0 -1
  300. egse/icons/settings.svg +0 -1
  301. egse/icons/shrink.svg +0 -1
  302. egse/icons/shutter.svg +0 -1
  303. egse/icons/sign-off.svg +0 -1
  304. egse/icons/sign-on.svg +0 -1
  305. egse/icons/sim-mode.svg +0 -1
  306. egse/icons/small-buttons-go.svg +0 -20
  307. egse/icons/small-buttons-minus.svg +0 -51
  308. egse/icons/small-buttons-plus.svg +0 -51
  309. egse/icons/sponge.svg +0 -220
  310. egse/icons/start-button-disabled.svg +0 -84
  311. egse/icons/start-button.svg +0 -50
  312. egse/icons/stop-button-disabled.svg +0 -84
  313. egse/icons/stop-button.svg +0 -50
  314. egse/icons/stop-red.svg +0 -17
  315. egse/icons/stop.svg +0 -1
  316. egse/icons/switch-disabled-square.svg +0 -87
  317. egse/icons/switch-disabled.svg +0 -15
  318. egse/icons/switch-off-square.svg +0 -87
  319. egse/icons/switch-off.svg +0 -72
  320. egse/icons/switch-on-square.svg +0 -87
  321. egse/icons/switch-on.svg +0 -61
  322. egse/icons/temperature-control.svg +0 -44
  323. egse/icons/th_ui_logo.svg +0 -1
  324. egse/icons/unplugged.svg +0 -23
  325. egse/icons/unvalid.png +0 -0
  326. egse/icons/user-interface.svg +0 -1
  327. egse/icons/vacuum.svg +0 -1
  328. egse/icons/valid.png +0 -0
  329. egse/icons/zoom-to-pixel-dark.svg +0 -64
  330. egse/icons/zoom-to-pixel-white.svg +0 -36
  331. egse/images/big-rotation-stage.png +0 -0
  332. egse/images/connected-100.png +0 -0
  333. egse/images/cross.svg +0 -6
  334. egse/images/disconnected-100.png +0 -0
  335. egse/images/gui-icon.png +0 -0
  336. egse/images/home.svg +0 -6
  337. egse/images/info-icon.png +0 -0
  338. egse/images/led-black.svg +0 -89
  339. egse/images/led-green.svg +0 -85
  340. egse/images/led-orange.svg +0 -85
  341. egse/images/led-red.svg +0 -85
  342. egse/images/load-icon.png +0 -0
  343. egse/images/load-setup.png +0 -0
  344. egse/images/load.png +0 -0
  345. egse/images/pause.png +0 -0
  346. egse/images/play-button.svg +0 -8
  347. egse/images/play.png +0 -0
  348. egse/images/process-status.png +0 -0
  349. egse/images/restart.png +0 -0
  350. egse/images/search.png +0 -0
  351. egse/images/sma.png +0 -0
  352. egse/images/start.png +0 -0
  353. egse/images/stop-button.svg +0 -8
  354. egse/images/stop.png +0 -0
  355. egse/images/switch-off.svg +0 -48
  356. egse/images/switch-on.svg +0 -48
  357. egse/images/undo.png +0 -0
  358. egse/images/update-button.svg +0 -11
  359. egse/imageviewer/exposureselection.py +0 -475
  360. egse/imageviewer/imageviewer.py +0 -198
  361. egse/imageviewer/matchfocalplane.py +0 -179
  362. egse/imageviewer/subfieldposition.py +0 -133
  363. egse/lampcontrol/__init__.py +0 -4
  364. egse/lampcontrol/beaglebone/beaglebone.py +0 -178
  365. egse/lampcontrol/beaglebone/beaglebone.yaml +0 -62
  366. egse/lampcontrol/beaglebone/beaglebone_cs.py +0 -106
  367. egse/lampcontrol/beaglebone/beaglebone_devif.py +0 -150
  368. egse/lampcontrol/beaglebone/beaglebone_protocol.py +0 -73
  369. egse/lampcontrol/energetiq/__init__.py +0 -22
  370. egse/lampcontrol/energetiq/eq99.yaml +0 -98
  371. egse/lampcontrol/energetiq/lampEQ99.py +0 -283
  372. egse/lampcontrol/energetiq/lampEQ99_cs.py +0 -128
  373. egse/lampcontrol/energetiq/lampEQ99_devif.py +0 -158
  374. egse/lampcontrol/energetiq/lampEQ99_encode_decode_errors.py +0 -73
  375. egse/lampcontrol/energetiq/lampEQ99_protocol.py +0 -69
  376. egse/lampcontrol/energetiq/lampEQ99_ui.py +0 -465
  377. egse/lib/CentOS-7/EtherSpaceLink_v34_86.dylib +0 -0
  378. egse/lib/CentOS-8/ESL-RMAP_v34_86.dylib +0 -0
  379. egse/lib/CentOS-8/EtherSpaceLink_v34_86.dylib +0 -0
  380. egse/lib/Debian/ESL-RMAP_v34_86.dylib +0 -0
  381. egse/lib/Debian/EtherSpaceLink_v34_86.dylib +0 -0
  382. egse/lib/Debian/libetherspacelink_v35_21.dylib +0 -0
  383. egse/lib/Linux/ESL-RMAP_v34_86.dylib +0 -0
  384. egse/lib/Linux/EtherSpaceLink_v34_86.dylib +0 -0
  385. egse/lib/Ubuntu-20/ESL-RMAP_v34_86.dylib +0 -0
  386. egse/lib/Ubuntu-20/EtherSpaceLink_v34_86.dylib +0 -0
  387. egse/lib/gssw/python3-gssw_2.2.3+31f63c9f-1_all.deb +0 -0
  388. egse/lib/macOS/ESL-RMAP_v34_86.dylib +0 -0
  389. egse/lib/macOS/EtherSpaceLink_v34_86.dylib +0 -0
  390. egse/lib/ximc/__pycache__/pyximc.cpython-38 2.pyc +0 -0
  391. egse/lib/ximc/__pycache__/pyximc.cpython-38.pyc +0 -0
  392. egse/lib/ximc/libximc.framework/Frameworks/libbindy.dylib +0 -0
  393. egse/lib/ximc/libximc.framework/Frameworks/libxiwrapper.dylib +0 -0
  394. egse/lib/ximc/libximc.framework/Headers/ximc.h +0 -5510
  395. egse/lib/ximc/libximc.framework/Resources/Info.plist +0 -42
  396. egse/lib/ximc/libximc.framework/Resources/keyfile.sqlite +0 -0
  397. egse/lib/ximc/libximc.framework/libbindy.so +0 -0
  398. egse/lib/ximc/libximc.framework/libximc +0 -0
  399. egse/lib/ximc/libximc.framework/libximc.so +0 -0
  400. egse/lib/ximc/libximc.framework/libximc.so.7.0.0 +0 -0
  401. egse/lib/ximc/libximc.framework/libxiwrapper.so +0 -0
  402. egse/lib/ximc/pyximc.py +0 -922
  403. egse/listener.py +0 -73
  404. egse/logger/__init__.py +0 -243
  405. egse/logger/log_cs.py +0 -321
  406. egse/metrics.py +0 -98
  407. egse/mixin.py +0 -464
  408. egse/monitoring.py +0 -95
  409. egse/ni/alarms/__init__.py +0 -26
  410. egse/ni/alarms/cdaq9375.py +0 -300
  411. egse/ni/alarms/cdaq9375.yaml +0 -89
  412. egse/ni/alarms/cdaq9375_cs.py +0 -130
  413. egse/ni/alarms/cdaq9375_devif.py +0 -183
  414. egse/ni/alarms/cdaq9375_protocol.py +0 -48
  415. egse/obs_inspection.py +0 -163
  416. egse/observer.py +0 -41
  417. egse/obsid.py +0 -163
  418. egse/powermeter/__init__.py +0 -0
  419. egse/powermeter/ni/__init__.py +0 -38
  420. egse/powermeter/ni/cdaq9184.py +0 -224
  421. egse/powermeter/ni/cdaq9184.yaml +0 -73
  422. egse/powermeter/ni/cdaq9184_cs.py +0 -130
  423. egse/powermeter/ni/cdaq9184_devif.py +0 -201
  424. egse/powermeter/ni/cdaq9184_protocol.py +0 -48
  425. egse/powermeter/ni/cdaq9184_ui.py +0 -544
  426. egse/powermeter/thorlabs/__init__.py +0 -25
  427. egse/powermeter/thorlabs/pm100a.py +0 -380
  428. egse/powermeter/thorlabs/pm100a.yaml +0 -132
  429. egse/powermeter/thorlabs/pm100a_cs.py +0 -136
  430. egse/powermeter/thorlabs/pm100a_devif.py +0 -127
  431. egse/powermeter/thorlabs/pm100a_protocol.py +0 -80
  432. egse/powermeter/thorlabs/pm100a_ui.py +0 -725
  433. egse/process.py +0 -451
  434. egse/procman/__init__.py +0 -811
  435. egse/procman/cannot_start_process_popup.py +0 -43
  436. egse/procman/procman.yaml +0 -49
  437. egse/procman/procman_cs.py +0 -201
  438. egse/procman/procman_ui.py +0 -2081
  439. egse/protocol.py +0 -603
  440. egse/proxy.py +0 -522
  441. egse/randomwalk.py +0 -140
  442. egse/reg.py +0 -585
  443. egse/reload.py +0 -122
  444. egse/reprocess.py +0 -675
  445. egse/resource.py +0 -333
  446. egse/rst.py +0 -135
  447. egse/search.py +0 -182
  448. egse/serialdevice.py +0 -190
  449. egse/services.py +0 -212
  450. egse/services.yaml +0 -51
  451. egse/settings.py +0 -379
  452. egse/settings.yaml +0 -980
  453. egse/setup.py +0 -1180
  454. egse/shutter/__init__.py +0 -0
  455. egse/shutter/thorlabs/__init__.py +0 -19
  456. egse/shutter/thorlabs/ksc101.py +0 -205
  457. egse/shutter/thorlabs/ksc101.yaml +0 -105
  458. egse/shutter/thorlabs/ksc101_cs.py +0 -136
  459. egse/shutter/thorlabs/ksc101_devif.py +0 -201
  460. egse/shutter/thorlabs/ksc101_protocol.py +0 -69
  461. egse/shutter/thorlabs/ksc101_ui.py +0 -548
  462. egse/shutter/thorlabs/sc10.py +0 -82
  463. egse/shutter/thorlabs/sc10.yaml +0 -52
  464. egse/shutter/thorlabs/sc10_controller.py +0 -81
  465. egse/shutter/thorlabs/sc10_cs.py +0 -108
  466. egse/shutter/thorlabs/sc10_interface.py +0 -25
  467. egse/shutter/thorlabs/sc10_simulator.py +0 -30
  468. egse/simulator.py +0 -41
  469. egse/slack.py +0 -61
  470. egse/socketdevice.py +0 -218
  471. egse/sockets.py +0 -218
  472. egse/spw.py +0 -1479
  473. egse/stages/__init__.py +0 -12
  474. egse/stages/aerotech/ensemble.py +0 -247
  475. egse/stages/aerotech/ensemble.yaml +0 -205
  476. egse/stages/aerotech/ensemble_controller.py +0 -275
  477. egse/stages/aerotech/ensemble_cs.py +0 -110
  478. egse/stages/aerotech/ensemble_interface.py +0 -132
  479. egse/stages/aerotech/ensemble_parameters.py +0 -433
  480. egse/stages/aerotech/ensemble_simulator.py +0 -27
  481. egse/stages/aerotech/mgse_sim.py +0 -193
  482. egse/stages/arun/smd3.py +0 -111
  483. egse/stages/arun/smd3.yaml +0 -68
  484. egse/stages/arun/smd3_controller.py +0 -472
  485. egse/stages/arun/smd3_cs.py +0 -112
  486. egse/stages/arun/smd3_interface.py +0 -53
  487. egse/stages/arun/smd3_simulator.py +0 -27
  488. egse/stages/arun/smd3_stop.py +0 -16
  489. egse/stages/huber/__init__.py +0 -49
  490. egse/stages/huber/smc9300.py +0 -904
  491. egse/stages/huber/smc9300.yaml +0 -63
  492. egse/stages/huber/smc9300_cs.py +0 -178
  493. egse/stages/huber/smc9300_devif.py +0 -345
  494. egse/stages/huber/smc9300_protocol.py +0 -111
  495. egse/stages/huber/smc9300_sim.py +0 -547
  496. egse/stages/huber/smc9300_ui.py +0 -973
  497. egse/state.py +0 -173
  498. egse/statemachine.py +0 -274
  499. egse/storage/__init__.py +0 -1004
  500. egse/storage/persistence.py +0 -2295
  501. egse/storage/storage.yaml +0 -72
  502. egse/storage/storage_cs.py +0 -214
  503. egse/styles/dark.qss +0 -343
  504. egse/styles/default.qss +0 -48
  505. egse/synoptics/__init__.py +0 -412
  506. egse/synoptics/syn.yaml +0 -9
  507. egse/synoptics/syn_cs.py +0 -195
  508. egse/system.py +0 -1408
  509. egse/tcs/__init__.py +0 -14
  510. egse/tcs/tcs.py +0 -874
  511. egse/tcs/tcs.yaml +0 -14
  512. egse/tcs/tcs_cs.py +0 -202
  513. egse/tcs/tcs_devif.py +0 -292
  514. egse/tcs/tcs_protocol.py +0 -177
  515. egse/tcs/tcs_sim.py +0 -177
  516. egse/tcs/tcs_ui.py +0 -543
  517. egse/tdms.py +0 -171
  518. egse/tempcontrol/__init__.py +0 -23
  519. egse/tempcontrol/agilent/agilent34970.py +0 -109
  520. egse/tempcontrol/agilent/agilent34970.yaml +0 -44
  521. egse/tempcontrol/agilent/agilent34970_cs.py +0 -116
  522. egse/tempcontrol/agilent/agilent34970_devif.py +0 -182
  523. egse/tempcontrol/agilent/agilent34970_protocol.py +0 -99
  524. egse/tempcontrol/agilent/agilent34972.py +0 -111
  525. egse/tempcontrol/agilent/agilent34972.yaml +0 -44
  526. egse/tempcontrol/agilent/agilent34972_cs.py +0 -117
  527. egse/tempcontrol/agilent/agilent34972_devif.py +0 -189
  528. egse/tempcontrol/agilent/agilent34972_protocol.py +0 -101
  529. egse/tempcontrol/beaglebone/beaglebone.py +0 -342
  530. egse/tempcontrol/beaglebone/beaglebone.yaml +0 -110
  531. egse/tempcontrol/beaglebone/beaglebone_cs.py +0 -117
  532. egse/tempcontrol/beaglebone/beaglebone_protocol.py +0 -135
  533. egse/tempcontrol/beaglebone/beaglebone_ui.py +0 -681
  534. egse/tempcontrol/digalox/digalox.py +0 -107
  535. egse/tempcontrol/digalox/digalox.yaml +0 -36
  536. egse/tempcontrol/digalox/digalox_cs.py +0 -112
  537. egse/tempcontrol/digalox/digalox_protocol.py +0 -55
  538. egse/tempcontrol/keithley/__init__.py +0 -33
  539. egse/tempcontrol/keithley/daq6510.py +0 -662
  540. egse/tempcontrol/keithley/daq6510.yaml +0 -105
  541. egse/tempcontrol/keithley/daq6510_cs.py +0 -163
  542. egse/tempcontrol/keithley/daq6510_devif.py +0 -343
  543. egse/tempcontrol/keithley/daq6510_protocol.py +0 -78
  544. egse/tempcontrol/keithley/daq6510_sim.py +0 -186
  545. egse/tempcontrol/lakeshore/__init__.py +0 -33
  546. egse/tempcontrol/lakeshore/lsci.py +0 -361
  547. egse/tempcontrol/lakeshore/lsci.yaml +0 -162
  548. egse/tempcontrol/lakeshore/lsci_cs.py +0 -174
  549. egse/tempcontrol/lakeshore/lsci_devif.py +0 -292
  550. egse/tempcontrol/lakeshore/lsci_protocol.py +0 -73
  551. egse/tempcontrol/lakeshore/lsci_ui.py +0 -389
  552. egse/tempcontrol/ni/__init__.py +0 -0
  553. egse/tempcontrol/spid/spid.py +0 -109
  554. egse/tempcontrol/spid/spid.yaml +0 -81
  555. egse/tempcontrol/spid/spid_controller.py +0 -279
  556. egse/tempcontrol/spid/spid_cs.py +0 -136
  557. egse/tempcontrol/spid/spid_protocol.py +0 -107
  558. egse/tempcontrol/spid/spid_ui.py +0 -727
  559. egse/tempcontrol/srs/__init__.py +0 -22
  560. egse/tempcontrol/srs/ptc10.py +0 -875
  561. egse/tempcontrol/srs/ptc10.yaml +0 -227
  562. egse/tempcontrol/srs/ptc10_cs.py +0 -128
  563. egse/tempcontrol/srs/ptc10_devif.py +0 -118
  564. egse/tempcontrol/srs/ptc10_protocol.py +0 -42
  565. egse/tempcontrol/srs/ptc10_ui.py +0 -906
  566. egse/ups/apc/apc.py +0 -236
  567. egse/ups/apc/apc.yaml +0 -45
  568. egse/ups/apc/apc_cs.py +0 -101
  569. egse/ups/apc/apc_protocol.py +0 -125
  570. egse/user.yaml +0 -7
  571. egse/vacuum/beaglebone/beaglebone.py +0 -149
  572. egse/vacuum/beaglebone/beaglebone.yaml +0 -44
  573. egse/vacuum/beaglebone/beaglebone_cs.py +0 -108
  574. egse/vacuum/beaglebone/beaglebone_devif.py +0 -164
  575. egse/vacuum/beaglebone/beaglebone_protocol.py +0 -193
  576. egse/vacuum/beaglebone/beaglebone_ui.py +0 -638
  577. egse/vacuum/instrutech/igm402.py +0 -92
  578. egse/vacuum/instrutech/igm402.yaml +0 -90
  579. egse/vacuum/instrutech/igm402_controller.py +0 -128
  580. egse/vacuum/instrutech/igm402_cs.py +0 -108
  581. egse/vacuum/instrutech/igm402_interface.py +0 -49
  582. egse/vacuum/instrutech/igm402_simulator.py +0 -36
  583. egse/vacuum/keller/kellerBus.py +0 -256
  584. egse/vacuum/keller/leo3.py +0 -102
  585. egse/vacuum/keller/leo3.yaml +0 -38
  586. egse/vacuum/keller/leo3_controller.py +0 -83
  587. egse/vacuum/keller/leo3_cs.py +0 -101
  588. egse/vacuum/keller/leo3_interface.py +0 -33
  589. egse/vacuum/mks/evision.py +0 -86
  590. egse/vacuum/mks/evision.yaml +0 -75
  591. egse/vacuum/mks/evision_cs.py +0 -101
  592. egse/vacuum/mks/evision_devif.py +0 -316
  593. egse/vacuum/mks/evision_interface.py +0 -60
  594. egse/vacuum/mks/evision_simulator.py +0 -24
  595. egse/vacuum/mks/evision_ui.py +0 -704
  596. egse/vacuum/pfeiffer/acp40.py +0 -87
  597. egse/vacuum/pfeiffer/acp40.yaml +0 -60
  598. egse/vacuum/pfeiffer/acp40_controller.py +0 -117
  599. egse/vacuum/pfeiffer/acp40_cs.py +0 -109
  600. egse/vacuum/pfeiffer/acp40_interface.py +0 -40
  601. egse/vacuum/pfeiffer/acp40_simulator.py +0 -39
  602. egse/vacuum/pfeiffer/tc400.py +0 -113
  603. egse/vacuum/pfeiffer/tc400.yaml +0 -83
  604. egse/vacuum/pfeiffer/tc400_controller.py +0 -140
  605. egse/vacuum/pfeiffer/tc400_cs.py +0 -109
  606. egse/vacuum/pfeiffer/tc400_interface.py +0 -70
  607. egse/vacuum/pfeiffer/tc400_simulator.py +0 -24
  608. egse/vacuum/pfeiffer/tpg261.py +0 -81
  609. egse/vacuum/pfeiffer/tpg261.yaml +0 -66
  610. egse/vacuum/pfeiffer/tpg261_controller.py +0 -150
  611. egse/vacuum/pfeiffer/tpg261_cs.py +0 -109
  612. egse/vacuum/pfeiffer/tpg261_interface.py +0 -60
  613. egse/vacuum/pfeiffer/tpg261_simulator.py +0 -24
  614. egse/version.py +0 -174
  615. egse/visitedpositions.py +0 -398
  616. egse/windowing.py +0 -213
  617. egse/zmq/__init__.py +0 -28
  618. egse/zmq/spw.py +0 -160
  619. egse/zmq_ser.py +0 -41
  620. scripts/alerts/cold.yaml +0 -278
  621. scripts/alerts/example_alerts.yaml +0 -54
  622. scripts/alerts/transition.yaml +0 -14
  623. scripts/alerts/warm.yaml +0 -49
  624. scripts/analyse_n_fee_hk_data.py +0 -44
  625. scripts/check_hdf5_files.py +0 -192
  626. scripts/check_register_sync.py +0 -47
  627. scripts/create_hdf5_report.py +0 -295
  628. scripts/csl_model.py +0 -436
  629. scripts/csl_restore_setup.py +0 -230
  630. scripts/export-grafana-dashboards.py +0 -50
  631. scripts/fdir/cs_recovery/fdir_cs_recovery.py +0 -59
  632. scripts/fdir/fdir_table.yaml +0 -70
  633. scripts/fdir/fdir_test_recovery.py +0 -11
  634. scripts/fdir/hw_recovery/fdir_agilent_hw_recovery.py +0 -73
  635. scripts/fdir/limit_recovery/fdir_agilent_limit.py +0 -64
  636. scripts/fdir/limit_recovery/fdir_bb_heater_limit.py +0 -61
  637. scripts/fdir/limit_recovery/fdir_ensemble_limit.py +0 -33
  638. scripts/fdir/limit_recovery/fdir_pressure_limit_recovery.py +0 -71
  639. scripts/fix_csv.py +0 -80
  640. scripts/n_fee_supply_voltage_calculation.py +0 -92
  641. scripts/playground.py +0 -30
  642. scripts/print_hdf5_hk_data.py +0 -68
  643. scripts/print_register_map.py +0 -43
  644. scripts/sron/commanding/control_heaters.py +0 -44
  645. scripts/sron/commanding/pumpdown.py +0 -46
  646. scripts/sron/commanding/set_pid_setpoint.py +0 -19
  647. scripts/sron/commanding/shutdown_bbb_heaters.py +0 -10
  648. scripts/sron/commanding/shutdown_pumps.py +0 -33
  649. scripts/sron/tm_gen/tm_gen_agilent.py +0 -38
  650. scripts/sron/tm_gen/tm_gen_heaters.py +0 -4
  651. scripts/sron/tm_gen/tm_gen_spid.py +0 -13
  652. scripts/update_operational_cgse.py +0 -268
  653. scripts/update_operational_cgse_old.py +0 -273
egse/protocol.py DELETED
@@ -1,603 +0,0 @@
1
- """
2
- CommandProtocol is a base class for communicating commands with the hardware or
3
- the control server. This class implements methods to send command messages and
4
- receive responses.
5
-
6
- The protocol also knows how to load the commands from the YAML file that contains
7
- command definitions.
8
-
9
- """
10
- import abc
11
- import inspect
12
- import logging
13
- import pickle
14
-
15
- from prometheus_client import Counter
16
- from prometheus_client import Summary
17
-
18
- from egse.command import Command
19
- from egse.command import CommandExecution
20
- from egse.control import ControlServer
21
- from egse.control import Failure
22
- from egse.device import DeviceConnectionObserver
23
- from egse.system import format_datetime
24
- from egse.zmq_ser import bind_address
25
-
26
- logger = logging.getLogger(__name__)
27
-
28
- COMMAND_REQUESTS = Counter("cs_command_requests_count", "Count the number of commands", ["target"])
29
- EXECUTION_TIME = Summary("cs_command_execution_time_seconds", "Time spent executing a command")
30
-
31
-
32
- def get_method(parent_obj, method_name: str):
33
- """
34
- Returns a bound method from a given class instance.
35
-
36
- Args:
37
- parent_obj: the class instance that provides the method
38
- method_name: name of the method that is requested
39
-
40
- Returns:
41
- the method [type: method].
42
-
43
- .. note::
44
- The method returned is an bound class instance method and therefore
45
- this method *does not* expects as its first argument the class
46
- instance, i.e. self, when you call this as a function.
47
-
48
- """
49
- if method_name is None or method_name == "None":
50
- return None
51
-
52
- if hasattr(parent_obj, method_name):
53
- method = getattr(parent_obj, method_name)
54
- if inspect.ismethod(method) or hasattr(method, "__method_wrapper"):
55
- return method
56
- logger.warning(f"{method_name} is not a method, type={type(method)}")
57
- else:
58
- logger.warning(f"{parent_obj!r} has no method called {method_name}")
59
-
60
- return None
61
-
62
-
63
- def get_function(parent_class, method_name: str):
64
- """
65
- Returns a function (unbound method) from a given class.
66
-
67
- Args:
68
- parent_class: the class that provides the method
69
- method_name: name of the method that is requested
70
-
71
- Returns:
72
- the method [type: function].
73
-
74
- .. note::
75
- The function returned is an unbound class instance method and
76
- therefore this function expects as its first argument the class
77
- instance, i.e. self, when you call it as a function.
78
-
79
- """
80
- if method_name is None or method_name == "None":
81
- return None
82
-
83
- if hasattr(parent_class, method_name):
84
- func = getattr(parent_class, method_name)
85
- if inspect.isfunction(func):
86
- return func
87
- logger.warning(f"{method_name} is not a function, type={type(func)}")
88
- else:
89
- logger.warning(
90
- f"{parent_class.__module__}.{parent_class.__name__} has no method called {method_name}"
91
- )
92
-
93
- return None
94
-
95
-
96
- class BaseCommandProtocol(DeviceConnectionObserver):
97
-
98
- def __init__(self, control_server: ControlServer):
99
- super().__init__()
100
- self.__socket = None
101
- self.__control_server = control_server
102
-
103
- def bind(self, socket):
104
- """Bind to a socket to listen for commands."""
105
- self.__socket = socket
106
- self.__socket.bind(self.get_bind_address())
107
-
108
- def get_bind_address(self):
109
- """
110
- Returns a string with the bind address, the endpoint, for accepting connections
111
- and bind a socket to.
112
-
113
- This method should be implemented by the sub-class since it contains the protocol
114
- and port number that are specific for the sub-class.
115
-
116
- Returns:
117
- a string with the protocol and port to bind a socket to.
118
- """
119
- return bind_address(
120
- self.__control_server.get_communication_protocol(),
121
- self.__control_server.get_commanding_port(),
122
- )
123
-
124
- def is_alive(self) -> bool:
125
- """
126
- This method can be overridden by a sub-class to check whether any Thread or sub-process
127
- that was started is still alive.
128
- """
129
- return True
130
-
131
- def get_control_server(self):
132
- return self.__control_server
133
-
134
- def get_status(self):
135
- """
136
- Returns a dictionary with status information for the control server, enhanced by the
137
- sub-class with device specific status information.
138
-
139
- This method should be implemented/overridden by the sub-class. The sub-class specific
140
- method should update the dictionary returned by this super-class method with device
141
- specific status values.
142
-
143
- The dict returned by this method includes the following keywords:
144
-
145
- * timestamp (str): a string representation of the current datetime
146
- * PID (int): the Process ID for the control server
147
- * Up (float): the uptime of the control server [s]
148
- * UUID (uuid1): a UUID for the control server
149
- * RSS (int): the 'Resident Set Size', this is the non-swapped physical memory a process
150
- has used [byte]
151
- * USS (int): the 'Unique Set Size', this is the memory which is unique to a process [byte]
152
- * CPU User (float): time spent in user mode [s]
153
- * CPU System (float): time spent in kernel mode [s]
154
- * CPU% (float): the process CPU utilization as a percentage [%]
155
-
156
- Check the documentation for `psutil.Process` for more in-depth information about the
157
- dict keys.
158
-
159
- Returns:
160
- a dictionary with status information.
161
- """
162
- status = {
163
- "timestamp": format_datetime(),
164
- "delay": self.__control_server.delay,
165
- }
166
- status.update(self.__control_server.get_process_status())
167
- return status
168
-
169
- def get_housekeeping(self) -> dict:
170
- """Returns a dictionary with housekeeping information about the device."""
171
- raise NotImplementedError(
172
- f"The get_housekeeping() method shall be implemented for {self.__class__.__name__}."
173
- )
174
-
175
- def get_device(self):
176
- """Returns the device object for the device that is controlled by this protocol."""
177
- raise NotImplementedError
178
-
179
- def send(self, data):
180
- """
181
- Send a message to the ControlServer. The message shall be fully populated
182
- and is only serialized before sending over the ZeroMQ socket.
183
-
184
- FIXME: We need to add error handling here, e.g. what if the send() fails? Do we need
185
- to implement retries as with Proxy?
186
- """
187
- pickle_string = pickle.dumps(data)
188
- self.__socket.send(pickle_string)
189
-
190
- def receive(self):
191
- """
192
- Receive a serialized message from the ControlServer. The message will not
193
- be decoded/de-serialized, but is returned as it was sent. Decoding shall
194
- be handled by the calling method.
195
- """
196
- pickle_string = self.__socket.recv()
197
- return pickle.loads(pickle_string)
198
-
199
- # FIXME:
200
- # We might want to reconsider how commands are send over the ZeroMQ sockets.
201
- # it can be very useful to use multipart messages here with the type and
202
- # origin etc. to ease the if..else.. constructs.
203
-
204
- @EXECUTION_TIME.time()
205
- def execute(self):
206
- cs = self.get_control_server()
207
- data = self.receive()
208
- cmd = None
209
- if isinstance(data, CommandExecution):
210
- cmd = data.get_cmd()
211
- cmd_name = cmd.get_name()
212
- args = data.get_args()
213
- kwargs = data.get_kwargs()
214
- elif isinstance(data, dict):
215
- cmd_name = data.get("cmd")
216
- args = data.get("args")
217
- kwargs = data.get("kwargs")
218
- elif isinstance(data, str):
219
- cmd_name = data
220
- else:
221
- cmd_name = None
222
-
223
- logger.log(0, f"cmd_name = {cmd_name}")
224
-
225
- # Server availability request - Ping-Pong
226
-
227
- if cmd_name == "Ping":
228
- COMMAND_REQUESTS.labels(target="ping").inc()
229
- self.send("Pong")
230
- elif cmd_name == "send_commands":
231
- logger.warning("send_commands was commanded for a DynamicCommandProtocol!")
232
- elif cmd_name == "get_service_port":
233
- self.send(self.__control_server.get_service_port())
234
- elif cmd_name == "get_monitoring_port":
235
- self.send(self.__control_server.get_monitoring_port())
236
- elif cmd_name == "get_commanding_port":
237
- self.send(self.__control_server.get_commanding_port())
238
- elif cmd_name == "get_ip_address":
239
- self.send(self.__control_server.get_ip_address())
240
- elif cmd:
241
- COMMAND_REQUESTS.labels(target="device").inc()
242
- cmd.server_call(self, *args, **kwargs)
243
- else:
244
- COMMAND_REQUESTS.labels(target="invalid").inc()
245
- logger.warning(f"Invalid command received: {cmd_name}")
246
- self.send(Failure(f"Invalid command: {cmd_name}"))
247
-
248
- def quit(self):
249
- """
250
- This method can be overridden by a sub-class to cleanup and stop threads that it
251
- started.
252
- """
253
-
254
- logger.info("quit() method called on Protocol base class.")
255
-
256
-
257
-
258
- class DynamicCommandProtocol(BaseCommandProtocol, metaclass=abc.ABCMeta):
259
- def __init__(self, control_server: ControlServer):
260
- super().__init__(control_server)
261
-
262
- def handle_device_method(self, cmd: Command, *args, **kwargs):
263
- """
264
- Call the device method with the given arguments.
265
-
266
- Args:
267
- cmd: the devices command class that knows which device command shall be called
268
- args: the arguments that will be passed on to the device command
269
- kwargs: the keyword arguments that will be passed on to the device command
270
- """
271
- # The lookup table contains object (bound) methods, so we do not have to
272
- # provide the 'self' argument anymore.
273
-
274
- method_name = cmd.get_device_method_name()
275
- method = get_method(self.get_device(), method_name)
276
-
277
- # We treat the get_response function special as it needs to send the ``cmd`` string
278
- # to the device we need to pass the processed cmd string into the method.
279
-
280
- try:
281
- if method_name == "get_response":
282
- device_cmd_string = cmd.get_cmd_string(*args, *kwargs)
283
- logger.log(5, f"Executing method {method.__name__}({device_cmd_string})")
284
- response = method(device_cmd_string)
285
- else:
286
- logger.log(5, f"Executing method {method.__name__}({args}, {kwargs})")
287
- response = method(*args, **kwargs)
288
- except Exception as exc:
289
- logger.exception(f"Executing {method_name} failed.")
290
- # Pass the exception on to the client as a Failure message
291
- response = Failure(f"Executing {method_name} failed: ", exc)
292
-
293
- # Enable the following message only when debugging, because this log message can become
294
- # very long for data storage commands.
295
- # logger.debug(f"handle_device_method: {device_name}({args}, {kwargs}) -> {response!s}")
296
-
297
- self.send(response)
298
-
299
-
300
- # TODO (rik): The CommandProtocol shall also inherit from the BaseCommandProtocol
301
-
302
- class CommandProtocol(DeviceConnectionObserver, metaclass=abc.ABCMeta):
303
- """
304
- This class is the glue between the control servers and the hardware
305
- controllers on one side, and between the control server and the connected
306
- proxy classes on the other side.
307
-
308
- The connection with the hardware controllers is when the ``execute()`` method
309
- calls the ``server_call()`` method of the command class.
310
-
311
- The connection with the proxy classes is when the ``client_call()`` method is added to the
312
- interface of the Proxy subclass (by the ``_add_commands()`` method).
313
-
314
- FIXME: Protocol is not used at the client side, i.e. the Proxy class.
315
- """
316
-
317
- def __init__(self):
318
- super().__init__()
319
- self.__socket = None
320
- self._commands = {} # variable is used by sub classes
321
- self.control_server: ControlServer | None = None # variable set by the sub-class
322
- self._method_lookup = {} # lookup table for device methods
323
-
324
- def bind(self, socket):
325
- """Bind to a socket to listen for commands."""
326
- self.__socket = socket
327
-
328
- bind_address = self.get_bind_address()
329
- logger.info(f"Binding to {bind_address}")
330
-
331
- self.__socket.bind(bind_address)
332
-
333
- # FIXME:
334
- # We might want to reconsider how commands are send over the ZeroMQ sockets.
335
- # it can be very useful to use multipart messages here with the type and
336
- # origin etc. to ease the if..else.. constructs.
337
-
338
- @EXECUTION_TIME.time()
339
- def execute(self):
340
- data = self.receive()
341
- cmd = None
342
- if isinstance(data, CommandExecution):
343
- cmd = data.get_cmd()
344
- cmd_name = cmd.get_name()
345
- args = data.get_args()
346
- kwargs = data.get_kwargs()
347
- elif isinstance(data, dict):
348
- cmd_name = data.get("cmd")
349
- args = data.get("args")
350
- kwargs = data.get("kwargs")
351
- elif isinstance(data, str):
352
- cmd_name = data
353
- else:
354
- cmd_name = None
355
-
356
- logger.log(0, f"cmd_name = {cmd_name}")
357
-
358
- # Server availability request - Ping-Pong
359
-
360
- if cmd_name == "Ping":
361
- COMMAND_REQUESTS.labels(target="ping").inc()
362
- self.send("Pong")
363
- elif cmd_name == "send_commands":
364
- self.send_commands()
365
- elif cmd_name == "get_service_port":
366
- self.send(self.control_server.get_service_port())
367
- elif cmd_name == "get_monitoring_port":
368
- self.send(self.control_server.get_monitoring_port())
369
- elif cmd_name == "get_commanding_port":
370
- self.send(self.control_server.get_commanding_port())
371
- elif cmd_name == "get_ip_address":
372
- self.send(self.control_server.get_ip_address())
373
- elif cmd_name == "get_storage_mnemonic":
374
- self.send(self.control_server.get_storage_mnemonic())
375
- elif cmd:
376
- COMMAND_REQUESTS.labels(target="device").inc()
377
- cmd.server_call(self, *args, **kwargs)
378
- else:
379
- COMMAND_REQUESTS.labels(target="invalid").inc()
380
- logger.warning(f"Invalid command received: {cmd_name}")
381
- self.send(Failure(f"Invalid command: {cmd_name}"))
382
-
383
- def quit(self):
384
- """
385
- This method can be overridden by a sub-class to cleanup and stop threads that it
386
- started.
387
- """
388
-
389
- logger.info("quit() method called on Protocol base class.")
390
-
391
- def is_alive(self) -> bool:
392
- """
393
- This method can be overridden by a sub-class to check whether any Thread or sub-process
394
- that was started is still alive.
395
- """
396
- return True
397
-
398
- @abc.abstractmethod
399
- def get_bind_address(self):
400
- """
401
- Returns a string with the bind address, the endpoint, for accepting connections
402
- and bind a socket to.
403
-
404
- This method should be implemented by the sub-class since it contains the protocol
405
- and port number that are specific for the sub-class.
406
-
407
- Returns:
408
- a string with the protocol and port to bind a socket to.
409
- """
410
- pass
411
-
412
- @abc.abstractmethod
413
- def get_status(self):
414
- """
415
- Returns a dictionary with status information for the control server, enhanced by the
416
- sub-class with device specific status information.
417
-
418
- This method should be implemented/overridden by the sub-class. The sub-class specific
419
- method should update the dictionary returned by this super-class method with device
420
- specific status values.
421
-
422
- The dict returned by this method includes the following keywords:
423
-
424
- * timestamp (str): a string representation of the current datetime
425
- * PID (int): the Process ID for the control server
426
- * Up (float): the uptime of the control server [s]
427
- * UUID (uuid1): a UUID for the control server
428
- * RSS (int): the 'Resident Set Size', this is the non-swapped physical memory a process
429
- has used [byte]
430
- * USS (int): the 'Unique Set Size', this is the memory which is unique to a process [byte]
431
- * CPU User (float): time spent in user mode [s]
432
- * CPU System (float): time spent in kernel mode [s]
433
- * CPU% (float): the process CPU utilization as a percentage [%]
434
-
435
- Check the documentation for `psutil.Process` for more in-depth information about the
436
- dict keys.
437
-
438
- Returns:
439
- a dictionary with status information.
440
- """
441
- status = {
442
- "timestamp": format_datetime(),
443
- "delay": self.control_server.delay,
444
- }
445
- status.update(self.control_server.get_process_status())
446
- return status
447
-
448
- def get_housekeeping(self) -> dict:
449
- """Returns a dictionary with housekeeping information about the device."""
450
- raise NotImplementedError(
451
- f"The get_housekeeping() method shall be implemented for {self.__class__.__name__}."
452
- )
453
-
454
- def send(self, data):
455
- """
456
- Send a message to the ControlServer. The message shall be fully populated
457
- and is only serialized before sending over the ZeroMQ socket.
458
-
459
- FIXME: We need to add error handling here, e.g. what if the send() fails? Do we need
460
- to implement retries as with Proxy?
461
- """
462
- pickle_string = pickle.dumps(data)
463
- self.__socket.send(pickle_string)
464
-
465
- def receive(self):
466
- """
467
- Receive a serialized message from the ControlServer. The message will not
468
- be decoded/de-serialized, but is returned as it was sent. Decoding shall
469
- be handled by the calling method.
470
- """
471
- pickle_string = self.__socket.recv()
472
- data = pickle.loads(pickle_string)
473
- return data
474
-
475
- def send_commands(self):
476
- """
477
- Send the command definitions that were loaded for the specific device.
478
- """
479
- self.send(self._commands)
480
-
481
- def load_commands(self, command_settings, command_class, device_class):
482
- """
483
- Loads the command definitions from the given ``command_settings`` and builds an internal
484
- dictionary containing the command names as keys and the corresponding ``Command`` class
485
- objects as values.
486
-
487
- The ``command_settings`` is usually loaded from a YAML configuration file containing the
488
- command definitions for the device.
489
-
490
- Args:
491
- command_settings: a dictionary containing the command definitions for this device
492
- command_class: the type of command to create, a subclass of Command
493
- device_class: the type of the base device class from which the methods are loaded
494
- """
495
- for name in command_settings:
496
- command_settings_name = command_settings[name]
497
- if "cmd" in command_settings_name:
498
- cmd = command_settings_name["cmd"]
499
- else:
500
- cmd = ""
501
-
502
- if "description" in command_settings_name:
503
- description = command_settings_name["description"]
504
- else:
505
- description = None
506
-
507
- # The response field is the name of a function from the CommandProtocol class or a
508
- # sub-class. This function shall send a response back to the client (Proxy). That's
509
- # why this field is called response.
510
- # By convention we like that this method name would start with `handle_` so the we
511
- # can make a distinction between response commands and normal methods in Protocol.
512
- # Remember that response methods should send a reply back to the client (which will
513
- # be waiting for it..).
514
- # If no response field is given, then the `handle_device_method` will be called.
515
-
516
- if "response" in command_settings_name:
517
- response_method = get_function(self.__class__, command_settings_name["response"])
518
- else:
519
- response_method = get_function(self.__class__, "handle_device_method")
520
-
521
- # The device_method field is used in the `handle_device_method` to call the method on
522
- # the device class. That is the class that implements the DeviceInterface and is
523
- # usually called a Controller or a Simulator.
524
- #
525
- # If no device_name field is given, the name from the command_settings is used.
526
-
527
- if "device_method" in command_settings_name:
528
- device_method_name = command_settings_name["device_method"]
529
- else:
530
- device_method_name = name
531
-
532
- # check if the device_method exists in the device base class
533
-
534
- if device_method_name == "None":
535
- device_method = None
536
- else:
537
- device_method = get_function(device_class, device_method_name)
538
-
539
- logger.log(
540
- 0,
541
- f"Creating {command_class.__module__}.{command_class.__name__}(name='{name}', "
542
- f"cmd='{cmd}', "
543
- f"response={response_method}, device_method={device_method})",
544
- )
545
- logger.debug(f"Creating {command_class.__name__} command with {name=}, {cmd=}, {device_method=}")
546
-
547
- self._commands[name] = command_class(
548
- name=name,
549
- cmd=cmd,
550
- response=response_method,
551
- description=description,
552
- device_method=device_method,
553
- )
554
-
555
- def build_device_method_lookup_table(self, device_obj):
556
- """
557
- Fill the lookup table with device command methods that are bound to the device object.
558
-
559
- Args:
560
- device_obj: instance of a device command class
561
- """
562
- for cmd in self._commands.values():
563
- method_name = cmd.get_device_method_name()
564
- method = get_method(device_obj, method_name)
565
- if method is not None:
566
- self._method_lookup[method_name] = method
567
-
568
- def handle_device_method(self, cmd: Command, *args, **kwargs):
569
- """
570
- Call the device method with the given arguments.
571
-
572
- Args:
573
- cmd: the devices command class that knows which device command shall be called
574
- args: the arguments that will be passed on to the device command
575
- kwargs: the keyword arguments that will be passed on to the device command
576
- """
577
- # The lookup table contains object (bound) methods, so we do not have to
578
- # provide the 'self' argument anymore.
579
-
580
- device_name = cmd.get_device_method_name()
581
- method = self._method_lookup[device_name]
582
-
583
- # We treat the get_response function special as it needs to send the ``cmd`` string
584
- # to the device we need to pass the processed cmd string into the method.
585
-
586
- try:
587
- if device_name == "get_response":
588
- device_cmd_string = cmd.get_cmd_string(*args, *kwargs)
589
- logger.log(5, f"Executing method {method.__name__}({device_cmd_string})")
590
- response = method(device_cmd_string)
591
- else:
592
- logger.log(5, f"Executing method {method.__name__}({args}, {kwargs})")
593
- response = method(*args, **kwargs)
594
- except Exception as exc:
595
- logger.exception(f"Executing {device_name} failed.")
596
- # Pass the exception on to the client as a Failure message
597
- response = Failure(f"Executing {device_name} failed: ", exc)
598
-
599
- # Enable the following message only when debugging, because this log message can become
600
- # very long for data storage commands.
601
- # logger.debug(f"handle_device_method: {device_name}({args}, {kwargs}) -> {response!s}")
602
-
603
- self.send(response)