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