cgse 2024.7.0__py3-none-any.whl → 2025.0.2__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.2.dist-info/METADATA +38 -0
  4. cgse-2025.0.2.dist-info/RECORD +5 -0
  5. {cgse-2024.7.0.dist-info → cgse-2025.0.2.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/alert/__init__.py DELETED
@@ -1,1049 +0,0 @@
1
- import importlib
2
- import logging
3
- import pickle
4
- import ssl
5
- import threading
6
- from collections import namedtuple
7
- from smtplib import SMTP
8
-
9
- import numpy as np
10
- import time
11
- import zmq
12
-
13
- from egse.alert.gsm.beaglebone import BeagleboneProxy
14
- from egse.command import ClientServerCommand
15
- from egse.control import ControlServer, is_control_server_active
16
- from egse.decorators import dynamic_interface
17
- from egse.device import DeviceInterface
18
- from egse.fdir.fdir_manager import FdirManagerProxy
19
- from egse.hk import get_housekeeping, HKError
20
- from egse.procman import ProcessManagerProxy
21
- from egse.protocol import CommandProtocol
22
- from egse.proxy import Proxy
23
- from egse.settings import Settings
24
- from egse.setup import load_setup
25
- from egse.system import format_datetime, EPOCH_1958_1970
26
- from egse.system import replace_environment_variable, find_class
27
- from egse.zmq_ser import connect_address, bind_address
28
-
29
- logger = logging.getLogger(__name__)
30
-
31
- CTRL_SETTINGS = Settings.load("Alert Manager Control Server")
32
- DEVICE_SETTINGS = Settings.load(filename='alertman.yaml')
33
-
34
- CS_STATUS = ['Offline', 'Controller disconnected', 'Online']
35
-
36
- LIMIT = 0
37
- SETPOINT = 1
38
- RATEOFCHANGE = 2
39
- MASK = 3
40
-
41
- class AlertException(BaseException):
42
- """
43
- Base class for all expcetion related to the Alerts
44
- """
45
- pass
46
-
47
-
48
-
49
- class AlertManagerCommand(ClientServerCommand):
50
- """
51
- Command (client-server) for process management.
52
- """
53
- pass
54
-
55
-
56
-
57
- def load_alert_configuration(phase):
58
- """ Load the alert configuration for the configration file.
59
-
60
- Returns all the alerts defined in the given file. Uses defaults for missing fields.
61
- """
62
- AlertConfiguration = namedtuple('AlertConfiguration', ['name',
63
- 'metric',
64
- 'timeout',
65
- 'unit',
66
- 'actions',
67
- 'maximum_age',
68
- 'fdir_code',
69
- 'sample_rate',
70
- 'alert_type',
71
- 'limits'])
72
-
73
- try:
74
- configuration = load_setup().gse.alert_manager.configuration[phase]
75
- except AttributeError as ex:
76
- raise AlertException("Could not find a configuration for the {} phase in the setup".format(phase)) from ex
77
- else:
78
- alerts = {}
79
-
80
- for name, config in configuration['alerts'].items():
81
- try:
82
- if 'operational_limit' in config:
83
- alert_type = LIMIT
84
-
85
- limits = {
86
- 'min' : config['operational_limit'].get('min', None),
87
- 'max' : config['operational_limit'].get('max', None)
88
- }
89
-
90
- elif 'rate_of_change' in config:
91
- alert_type = RATEOFCHANGE
92
-
93
- limits = {
94
- 'limit' : config['rate_of_change'].get('limit', None),
95
- 'time_window' : config['rate_of_change'].get('time_window', 60)
96
- }
97
-
98
- if limits['limit'] is None:
99
- raise AlertException(\
100
- "Invalid configuration: No limit defined in the configuration")
101
-
102
- elif 'mask' in config:
103
- alert_type = MASK
104
- limits = {'mask': config['mask']}
105
-
106
- elif 'setpoint' in config:
107
- alert_type = SETPOINT
108
-
109
- limits = {
110
- 'metric' : None,
111
- 'value' : None,
112
- 'offset' : config['setpoint'].get('offset', 2)
113
- }
114
-
115
- if 'setpoint' in config:
116
- if 'sensor' not in config['setpoint'] and \
117
- 'value' not in config['setpoint']:
118
- raise AlertException(\
119
- "Invalid configuration: No metric or value defined in configuration")
120
- elif 'sensor' in config['setpoint'] and \
121
- 'value' in config['setpoint']:
122
- raise AlertException(\
123
- "Invalid configuration: Can not have both a metric and value in configuration")
124
- else:
125
- if 'sensor' in config['setpoint']:
126
- limits['metric'] = config['setpoint']['sensor']
127
- if 'value' in config['setpoint']:
128
- limits['value'] = config['setpoint']['value']
129
- else:
130
- raise AlertException("Invalid configuration: No alert type was defined in the configuration")
131
-
132
- alerts[name] = AlertConfiguration(
133
- name,
134
- config['sensor'],
135
- config.get('timeout', 5),
136
- config.get('unit', ''),
137
- config.get('actions', []),
138
- config.get('maxAge', 30),
139
- config.get('fdir', ''),
140
- config.get('sampleRate', 5),
141
- alert_type,
142
- limits
143
- )
144
-
145
- except Exception as ex:
146
- logger.warning("Could not create configuration for alert '%s' : %s" % (name, ex))
147
-
148
- return configuration, alerts
149
-
150
-
151
-
152
- class Alert:
153
- """
154
- Alert class representing a single alert
155
-
156
- An alert has one main function self.update() which retrieves data from housekeeping and compares
157
- it to the limits defined in the configuration. The Alert Manager Controller runs this method in
158
- a thread. Sanity checks have been added to prevent false positives.
159
- """
160
- def __init__(self, configuration, notification_worker):
161
- """ Initialization of a new alert, to be used by the Alert Manager
162
-
163
-
164
- Args:
165
- configuration (_type_): _description_
166
- notification_worker (_type_): _description_
167
- """
168
- self.configuration = configuration
169
- self.notification_worker = notification_worker
170
- self.value = 0.0
171
- self.triggered = False
172
- self.notified = False
173
- self.active = False
174
- self.timed_out = 0.0
175
- self.cycle_check = False
176
- self.cycle_timeout = 0.0
177
- self.lastTimestamp = 0.0
178
- self.last_update = 0.0
179
- self.thread = threading.Thread()
180
-
181
- def metric_is_available(self):
182
- """ Check whether a metric is available in the HK files """
183
- try:
184
- timestamp, _ = get_housekeeping(hk_name=self.configuration.metric)
185
- if time.time() - (float(timestamp) - EPOCH_1958_1970) > self.configuration.maximum_age:
186
- raise
187
- except:
188
- logger.warning(f"Alert '{self.configuration.name}' : " \
189
- f"Metric: '{self.configuration.metric}' is currently unavailable." \
190
- f"Please restart this alert when metric is available.")
191
- return False
192
- else:
193
- return True
194
-
195
- def daily_cycle_check(self):
196
- """ When a metric is not found in housekeeping, keeping trying for 60 seconds.
197
- This makes sure that alerts don't crash during the daily cycle of housekeeping files
198
-
199
- Returns:
200
- bool: Whether the file has cycled or not
201
- """
202
- try:
203
- _, _ = get_housekeeping(hk_name=self.configuration.metric)
204
- except HKError as ex:
205
- if not self.cycle_check:
206
- self.cycle_check = True
207
- self.cycle_timeout = time.time()
208
-
209
- if (time.time() - self.cycle_timeout) > 300:
210
- logger.info(f"Alert '{self.configuration.name}' : No daily file found after 5 minutes. Housekeeping is unavailable: {ex}")
211
- self.active = False
212
- self.triggered = False
213
- self.timed_out = 0.0
214
-
215
- return False
216
- else:
217
- logger.debug(f"Alert '{self.configuration.name}' : Daily file has been cycled")
218
- self.cycle_check = False
219
- return True
220
-
221
-
222
- def generate_alert_message(self):
223
- """ Generates the email body for the alert based on the alert type
224
- """
225
- body = ""
226
-
227
- fdir_code = self.configuration.fdir_code
228
- actions = self.configuration.actions
229
-
230
- if self.configuration.alert_type == LIMIT:
231
- if self.configuration.limits['min'] != None and self.configuration.limits['max'] != None:
232
- body += f"- {self.configuration.name} exceeded its operational limits of {float(self.configuration.limits['min']):.3e} & {float(self.configuration.limits['max']):.3e}. Current value: {float(self.value):.3e}\n"
233
- elif self.configuration.limits['min'] != None:
234
- body += f"- {self.configuration.name} exceeded its operatinoal limits of {float(self.configuration.limits['min']):.3e}. Current value: {float(self.value):.3e}\n"
235
- elif self.configuration.limits['max'] != None:
236
- body += f"- {self.configuration.name} exceeded its operational limits of {float(self.configuration.limits['max']):.3e}. Current value: {float(self.value):.3e}\n"
237
- elif self.configuration.alert_type is SETPOINT:
238
- body += f"- {self.configuration.name} setpoint offset was larger than {float(self.configuration.limits['offset']):.3e}. Current value: {float(self.value):.3e}\n"
239
- elif self.configuration.alert_type is MASK:
240
- body += f"- {self.configuration.name} is {bool(self.value)}"
241
- elif self.configuration.alert_type is RATEOFCHANGE:
242
- body += f"- {self.configuration.name} has exceeded its RoC limit of {float(self.configuration.limits['limit']):.3e}. Current value: {float(self.value):.3e}\n"
243
-
244
- if fdir_code:
245
- body += f"\t- FDIR code: '{self.configuration.fdir_code}' was fired\n"
246
-
247
- if actions:
248
- body += f"\t- Corrective actions:\n"
249
- for action in actions:
250
- body += f"\t\t\t- {action}\n"
251
-
252
- return body
253
-
254
- def trigger_alert(self):
255
- """ Send a notifcation and FDIR signal, also ensures that alerts are only triggered once
256
- """
257
- if not self.notified:
258
- self.triggered = True
259
- logger.critical(f"Alert '{self.configuration.name}' : triggered!")
260
-
261
- self.notification_worker.add_notification(0, self.generate_alert_message())
262
-
263
- self.signal_fdir()
264
-
265
- self.notified = True
266
-
267
- def clear_alert(self):
268
- """ Resets the alert after it being triggered
269
- """
270
- if self.triggered:
271
- logger.info(f"Alert '{self.configuration.name}' : Alert status has been reset.")
272
- self.triggered = False
273
- self.timed_out = 0.0
274
- self.notified = False
275
-
276
- def signal_fdir(self):
277
- """ Signal the FDIR manager, if a fidr code was configured
278
- """
279
- if self.configuration.fdir_code:
280
- try:
281
- logger.info(f"Signaled fdir: '{self.configuration.fdir_code}'")
282
- with FdirManagerProxy() as fdir:
283
- fdir.signal_fdir(self.configuration.fdir_code, self.configuration.metric)
284
- except Exception as e:
285
- logger.critical(f"Alert '{self.configuration.name}' : Could not signal FDIR manager. {e}")
286
- else:
287
- logger.info(f"Alert '{self.configuration.name}' Does not have a configured FDIR code.")
288
-
289
- def get_housekeeping(self, metric, time_window=None):
290
- """ Retrieves housekeeping from the HK files. Checks whether the value actually exists,
291
- and if the age is still appropiate.
292
- """
293
- try:
294
- timestamps, values = get_housekeeping(hk_name=metric, time_window=time_window)
295
- except HKError:
296
- if self.daily_cycle_check():
297
- timestamps, values = get_housekeeping(hk_name=metric, time_window=time_window)
298
- else:
299
- raise AlertException(f"{self.configuration.name} : Daily file cycle is being checked")
300
-
301
-
302
- most_recent_timestamp = float(timestamps[-1]) if type(timestamps) == np.ndarray else timestamps
303
-
304
- self.lastTimestamp = most_recent_timestamp
305
-
306
- is_alive = time.time() - (most_recent_timestamp - EPOCH_1958_1970) > self.configuration.maximum_age
307
- value_exists = values[-1] if type(values) == np.ndarray else values
308
-
309
- if is_alive:
310
- self.active = False
311
- logger.critical(f"Alert '{self.configuration.name}' : Last housekeeping value was older than {self.configuration.maximum_age} seconds")
312
- raise Exception(f"Alert '{self.configuration.name}' : Last housekeeping value was older than {self.configuration.maximum_age} seconds")
313
-
314
- if not value_exists:
315
- logger.critical(f"Alert '{self.configuration.name}' : Housekeeping returned an emptry string as latest value")
316
- self.active = False
317
- raise Exception(f"Alert '{self.configuration.name}' : Housekeeping returned an emptry string as latest value")
318
-
319
- return timestamps, values
320
-
321
- def limit_check(self, value):
322
- """ Compares a given value to the configured upper and lower limits
323
- """
324
- min_triggered = False
325
- max_triggered = False
326
-
327
- self.value = value
328
-
329
- if self.configuration.limits['max']:
330
- if float(value) > self.configuration.limits['max']:
331
- self.timed_out = time.time() if self.timed_out == 0.0 else self.timed_out
332
- if (time.time() - self.timed_out) > self.configuration.timeout:
333
- max_triggered = True
334
-
335
- if self.configuration.limits['min']:
336
- if float(value) < self.configuration.limits['min']:
337
- self.timed_out = time.time() if self.timed_out == 0.0 else self.timed_out
338
- if (time.time() - self.timed_out) > self.configuration.timeout:
339
- min_triggered = True
340
-
341
- if min_triggered or max_triggered:
342
- self.trigger_alert()
343
- else:
344
- self.clear_alert()
345
-
346
- def setpoint_check(self, value):
347
- """ Compares a given value to the configured setpoint (dynamic/static setpoint)
348
- """
349
- self.value = value
350
-
351
- if self.configuration.limits['metric']:
352
- try:
353
- _, sp_value = self.get_housekeeping(self.configuration.limits['metric'], None)
354
- except Exception as ex:
355
- logger.warning(ex, exc_info=True)
356
- self.active = False
357
- return
358
-
359
- else:
360
- sp_value = self.configuration.limits['value']
361
-
362
- if not (float(sp_value) - self.configuration.limits['offset']) <= float(value) <= (float(sp_value) + self.configuration.limits['offset']):
363
- self.timed_out = time.time() if self.timed_out == 0.0 else self.timed_out
364
- if (time.time() - self.timed_out) > self.configuration.timeout:
365
- self.trigger_alert()
366
- else:
367
- self.clear_alert()
368
-
369
-
370
- def rate_of_change_check(self, timestamps, values):
371
- """ Compares a given value to the configured rate-of-change setpoint
372
- """
373
- deltaY = float(values[-1]) - float(values[0])
374
- deltaX = float(timestamps[-1]) - float(timestamps[0])
375
-
376
- self.value = deltaY / deltaX
377
-
378
- if self.configuration.limits['limit'] < 0:
379
- if self.value < self.configuration.limits['limit']:
380
- self.timed_out = time.time() if self.timed_out == 0.0 else self.timed_out
381
- if (time.time() - self.timed_out) > self.configuration.timeout:
382
- self.trigger_alert()
383
- else:
384
- self.clear_alert()
385
- else:
386
- if self.value > self.configuration.limits['limit']:
387
- self.timed_out = time.time() if self.timed_out == 0.0 else self.timed_out
388
- if (time.time() - self.timed_out) > self.configuration.timeout:
389
- self.trigger_alert()
390
- else:
391
- self.clear_alert()
392
-
393
-
394
- def mask_check(self, value):
395
- """ Masks a bitmap and check the state of a single bit
396
- """
397
- if isinstance(value, str):
398
- value = eval(value)
399
-
400
- self.value = int(value)
401
-
402
- if (self.value >> self.configuration.limits['mask']) & 0b1:
403
- self.timed_out = time.time() if self.timed_out == 0.0 else self.timed_out
404
- if (time.time() - self.timed_out) > self.configuration.maximum_age:
405
- self.trigger_alert()
406
- else:
407
- self.clear_alert()
408
-
409
- def update(self):
410
- """ Update current alert status
411
- """
412
- if self.active:
413
- if (time.time() - self.last_update) > self.configuration.sample_rate:
414
- try:
415
- timestamps, values = self.get_housekeeping(self.configuration.metric,
416
- self.configuration.limits.get('time_window', None))
417
- except Exception as ex:
418
- logger.warning(ex)
419
- self.active = False
420
- return
421
- except AlertException as a_ex:
422
- logger.warning(a_ex)
423
- self.last_update = (time.time() + 10) # Increase delay before next check
424
- return
425
- else:
426
- self.cycle_check = False
427
-
428
- try:
429
- if self.configuration.alert_type == LIMIT:
430
- self.limit_check(values)
431
- elif self.configuration.alert_type is SETPOINT:
432
- self.setpoint_check(values)
433
- elif self.configuration.alert_type is MASK:
434
- self.mask_check(values)
435
- elif self.configuration.alert_type is RATEOFCHANGE:
436
- self.rate_of_change_check(timestamps, values)
437
- except Exception as ex:
438
- logger.warning(ex)
439
- raise AlertException(ex)
440
-
441
- self.last_update = time.time()
442
-
443
-
444
-
445
- class ControlServerWorker:
446
- """ Worker to monitor all running control servers
447
- """
448
- def __init__(self, notifyer):
449
- self.pm_proxy = ProcessManagerProxy()
450
- self.devices = self.pm_proxy.get_devices()
451
- self.notifyer = notifyer
452
- self.thread = threading.Thread()
453
- self.active = False
454
- self.monitors = {}
455
-
456
- for process_name, process_info in self.devices.items():
457
-
458
- if ('alert' in process_name.lower()) or ('fdir' in process_name.lower()) or ('dpu' in process_name.lower()):
459
- continue
460
-
461
- self.monitors[process_name] = {'name' : process_name,
462
- 'type' : process_info[0],
463
- 'type_as_type' : find_class(process_info[0][7:]),
464
- 'device_args' : process_info[1],
465
- 'socket' : None,
466
- 'timeout' : 0.5,
467
- 'connected' : False,
468
- 'address' : None,
469
- 'prev_status' : False}
470
-
471
- self.fdir_codes = {}
472
-
473
- self.connect_to_sockets()
474
-
475
-
476
- def load_fdir_codes(self, configuration):
477
- if 'hardware' in configuration:
478
- self.fdir_codes = configuration['hardware']
479
- else:
480
- self.fdir_codes = []
481
-
482
- def connect_to_sockets(self):
483
- """ Connect to commanding/monitoring socket of control servers
484
- """
485
- for monitor in self.monitors.values():
486
- try:
487
- module_name = monitor['type'][7:].rsplit(".", 1)[0]
488
- module = importlib.import_module(module_name)
489
-
490
- ctrl_settings = module.CTRL_SETTINGS
491
- transport = ctrl_settings.PROTOCOL
492
- hostname = ctrl_settings.HOSTNAME
493
-
494
- if module_name == "egse.aeu.aeu" or "egse.tempcontrol.agilent.agilent3497" in module_name:
495
- name = monitor['name'].split(" ")[1].upper()
496
-
497
- commanding_port = ctrl_settings[name]['COMMANDING_PORT']
498
- monitoring_port = ctrl_settings[name]['MONITORING_PORT']
499
- else:
500
- commanding_port = ctrl_settings.COMMANDING_PORT
501
- monitoring_port = ctrl_settings.MONITORING_PORT
502
-
503
- monitoring_address = connect_address(transport, hostname, monitoring_port)
504
-
505
- monitor['socket'] = zmq.Context().socket(zmq.SUB)
506
- monitor['socket'].connect(monitoring_address)
507
- monitor['socket'].setsockopt_string(zmq.SUBSCRIBE, "")
508
-
509
- monitor['address'] = connect_address(transport, hostname, commanding_port)
510
- monitor['connected'] = True
511
-
512
- except AttributeError as attr_er:
513
- logger.exception(attr_er)
514
- monitor['connected'] = False
515
-
516
- # break
517
-
518
- def get_device_status(self, monitor):
519
- """ Retreive the current status of a control server
520
- """
521
- status = 0
522
-
523
- try:
524
- with find_class(monitor['type'])(*monitor['device_args']) as proxy:
525
- if isinstance(proxy, DeviceInterface):
526
- try:
527
- if proxy.is_connected():
528
- # CS connected to controller
529
- status = 2
530
- else:
531
- # CS not connected to controller
532
- status = 1
533
- except AttributeError:
534
- status = 1
535
-
536
- else:
537
- status = 2
538
-
539
- except ConnectionError:
540
- pass
541
-
542
- return status
543
-
544
- def start(self):
545
- """ Start Control server monitor worker
546
- """
547
- logger.info("Starting control server monitoring loop")
548
-
549
- if self.thread != None:
550
- if self.thread.is_alive():
551
- return
552
-
553
- self.thread = threading.Thread(target=self.run)
554
- self.thread.start()
555
-
556
-
557
- def stop(self):
558
- """ Stop Control server monitor worker
559
- """
560
- self.active = False
561
- if self.thread != None:
562
- if self.thread.is_alive():
563
- self.thread.join()
564
- return
565
- logger.info("Control server monitor has been stopped")
566
-
567
- def run(self):
568
- """ Control server monitoring loop
569
- """
570
- self.active = True
571
-
572
- while self.active:
573
- for monitor in self.monitors.values():
574
- if monitor['connected']:
575
- status = 0
576
- try:
577
- socket_list, _, _ = zmq.select([monitor['socket']], [], [], timeout=monitor['timeout'])
578
-
579
- if monitor['socket'] in socket_list:
580
- pickle_string = monitor['socket'].recv()
581
- monitoring_info = pickle.loads(pickle_string)
582
-
583
- monitor['timeout'] = monitoring_info['delay'] / 1000 + 0.5 # [s]
584
-
585
- status = self.get_device_status(monitor)
586
-
587
- except zmq.ZMQError:
588
- pass
589
-
590
- # Timeout occured
591
- else:
592
- if not is_control_server_active(endpoint=monitor['address'], timeout=30):
593
- status = 0
594
- else:
595
- monitor['timeout'] += 0.5
596
-
597
- if monitor['prev_status'] != status:
598
-
599
- if monitor['prev_status'] >= 1 and status == 0:
600
-
601
- msg = f"- {monitor['name']} Contol Server has gone offline."
602
-
603
- self.notifyer.add_notification(-1, msg)
604
-
605
- logger.warning(f"{monitor['name']} has stopped. Signalling FDIR to attempt a restart")
606
-
607
- try:
608
- with FdirManagerProxy() as fdir:
609
- fdir.signal_fdir('FDIR_CS_STOPPED', [monitor['name']])
610
- except Exception as ex:
611
- logger.critical(f"Could not send FDIR signal: {ex}")
612
-
613
- elif monitor['prev_status'] == 2 and status == 1:
614
-
615
- msg = f"- {monitor['name']} Control Server has lost connection to the device controller"
616
-
617
- self.notifyer.add_notification(-1, msg)
618
-
619
- if self.fdir_codes != {}:
620
- if monitor['name'] in self.fdir_codes:
621
- fdir_code = self.fdir_codes[monitor['name']]
622
- try:
623
- with FdirManagerProxy() as fdir:
624
- fdir.signal_fdir(f"FDIR_{fdir_code['code']}_HW", [f"{fdir_code['arg']}"])
625
- except Exception as ex:
626
- logger.critical(f"Could not send FDIR signal: {ex}")
627
-
628
- logger.warning(f"{monitor['name']} has lost connection to the device. Signalling FDIR to recover")
629
-
630
- monitor['prev_status'] = status
631
-
632
- logger.debug(f"{monitor['name']} Control Server status: {CS_STATUS[monitor['prev_status']]}")
633
- time.sleep(60)
634
-
635
-
636
-
637
- class AlertNotificationWorker:
638
- """
639
- Worker for handling the pushing of notification
640
- """
641
- def __init__(self):
642
- self.sender = replace_environment_variable(CTRL_SETTINGS.EMAIL_SENDER)
643
- self.recipients = replace_environment_variable(CTRL_SETTINGS.EMAIL_CS_RECIPIENTS)
644
-
645
- self._server = replace_environment_variable(CTRL_SETTINGS.EMAIL_SERVER)
646
- self._port = 25
647
- self._context = ssl.create_default_context()
648
-
649
- self.gsm_module = BeagleboneProxy()
650
-
651
- self.alert_messages = []
652
- self.cs_messages = []
653
-
654
- self.thread = threading.Thread()
655
-
656
- def notify(self):
657
- '''
658
- Threaded method that waits 30 seconds for new messages before pushing the notification
659
- '''
660
- # Check if message buffer is empty
661
- n = len(self.alert_messages) + len(self.cs_messages)
662
-
663
- if n != 0:
664
- start_time = time.time()
665
-
666
- # Wait until no new messages are added to the buffer in the last 30 seconds
667
- while time.time() - start_time < 60:
668
- n2 = len(self.alert_messages) + len(self.cs_messages)
669
-
670
- if n2 is not n:
671
- start_time = time.time()
672
- n = n2
673
- time.sleep(1)
674
-
675
- self.send_mail()
676
- self.send_sms()
677
-
678
- self.clear_notifications()
679
-
680
-
681
- def add_notification(self, nType, msg: str):
682
- """
683
- Adds a message to the notification queue and start the threaded method, if it is not running
684
- """
685
-
686
- # Add notification to message buffer
687
- if nType in [LIMIT, SETPOINT, RATEOFCHANGE, MASK]:
688
- self.alert_messages.append(msg)
689
- else:
690
- self.cs_messages.append(msg)
691
- # See if notification thread has already been started
692
- if not self.thread.is_alive():
693
- self.thread.__init__(target=self.notify, daemon=True)
694
- self.thread.start()
695
-
696
- def clear_notifications(self):
697
- """
698
- Clears the notification queue's
699
- """
700
- self.alert_messages = []
701
- self.cs_messages = []
702
-
703
- def send_mail(self):
704
- """
705
- """
706
- message = """\
707
- Subject: Plato Common-EGSE Notification
708
-
709
- """
710
- # message = """Subject: Plato Common-EGSE Notification"""
711
- if len(self.alert_messages) > 0:
712
- message += ''.join(self.alert_messages)
713
-
714
- if len(self.cs_messages) > 0:
715
- message += ''.join(self.cs_messages)
716
-
717
- message += "\n\nKind regards,\n" \
718
- "Your friendly PLATO notification bot\n\n" \
719
- "If I am not working as expected, please contact: s.n.gomashie@sron.nl"
720
- try:
721
- with SMTP(self._server, self._port) as server:
722
- server.sendmail(self.sender, self.recipients.split(','), message)
723
- except Exception as ex:
724
- logger.critical(f"Notification worker could not send an email: {ex}")
725
-
726
- logger.debug(f"Email send: {self.sender} {self.recipients}")
727
-
728
- def send_sms(self):
729
- self.gsm_module.set_alert(1)
730
-
731
- class AlertManagerInterface:
732
- """
733
- Interface for dynamic loading of the command for Alert Management.
734
- """
735
- @dynamic_interface
736
- def load_configuration(self, phase = 'none'):
737
- """ Load a new phase into the alert manager. """
738
-
739
- raise NotImplementedError
740
-
741
- @dynamic_interface
742
- def status(self, name = None) -> dict:
743
- """ Returns the status of all configured alerts """
744
-
745
- raise NotImplementedError
746
-
747
- @dynamic_interface
748
- def stop_cs_monitor(self):
749
- """ Stops the control server monitor """
750
-
751
- raise NotImplementedError
752
-
753
- @dynamic_interface
754
- def start_cs_monitor(self):
755
- """ Starts the control server monitor """
756
-
757
- raise NotImplementedError
758
-
759
- @dynamic_interface
760
- def start_alert(self, name = None):
761
- """ Start the alert monitor """
762
-
763
- raise NotImplementedError
764
-
765
- @dynamic_interface
766
- def stop_alert(self, name = None):
767
- """ Stop the alert monitor """
768
-
769
- raise NotImplementedError
770
-
771
-
772
-
773
- class AlertManagerController(AlertManagerInterface):
774
-
775
- """
776
- Controller for Alert Management
777
- """
778
-
779
- def __init__(self, phase='none'):
780
- """ Initialization for the Alert Manager Controller.
781
-
782
- Args:
783
- phase (str, optional): _description_. Defaults to 'none'.
784
- """
785
- self.phase = phase
786
- self.active = False
787
- self.alerts = {}
788
- self.notification_worker = AlertNotificationWorker()
789
- self.control_server_worker = ControlServerWorker(self.notification_worker)
790
-
791
- self.thread = threading.Thread()
792
-
793
- self.load_configuration(phase)
794
- self.control_server_worker.start()
795
-
796
- def load_configuration(self, phase = 'none'):
797
- """ Load a new phase into the alert manager. """
798
- if phase != 'none':
799
- self.stop_alert()
800
-
801
- try:
802
- self.configuration, alerts = load_alert_configuration(phase)
803
- self.alerts = self.initialize_alerts(alerts)
804
-
805
- self.control_server_worker.load_fdir_codes(self.configuration)
806
- except Exception as ex:
807
- logger.warning(f"Could not load {phase} phase: {ex}")
808
- else:
809
- self.phase = phase
810
-
811
- else:
812
- self.stop_alert()
813
- self.alerts = {}
814
- self.phase = phase
815
-
816
- logger.info(f"Alert manager has been configured for the {self.phase} phase")
817
-
818
- def initialize_alerts(self, alerts):
819
- """ Initialize the alerts
820
- """
821
- for key, configuration in alerts.items():
822
- try:
823
- alerts[key] = Alert(configuration, self.notification_worker)
824
- except Exception as ex:
825
- logger.warning(f"Could not create alert '{key}': {ex}")
826
- return alerts
827
-
828
- def status(self, name = None) -> dict:
829
- """ Returns the status of all configured alerts """
830
- status = {}
831
- if self.alerts:
832
- for key, alert in self.alerts.items():
833
- status[key] = {
834
- 'active' : alert.active,
835
- 'triggered' : alert.triggered,
836
- 'value' : alert.value,
837
- 'lastTimestamp' : alert.lastTimestamp
838
- }
839
- if name and (name in status):
840
- return status[name]
841
- else:
842
- return status
843
-
844
- def stop_cs_monitor(self):
845
- """ Stops the control server monitor
846
- """
847
- self.control_server_worker.stop()
848
-
849
- def start_cs_monitor(self):
850
- """ Starts the control server monitor
851
- """
852
- self.control_server_worker.start()
853
-
854
- def start_alert(self, name = None):
855
- """ Start one or all alert monitors
856
- """
857
- if name != None:
858
- self.alerts[name].active = True
859
- else:
860
- for alert in self.alerts.values():
861
- alert.active = True
862
-
863
- if self.thread.is_alive():
864
- return
865
-
866
- logger.info(f"Starting alert monitor in {self.phase} phase")
867
-
868
- self.thread.__init__(target=self.run)
869
- self.thread.setDaemon(True)
870
- self.thread.start()
871
-
872
- def stop_alert(self, name = None):
873
- """ Stop one or all alert monitors
874
- """
875
- if name != None:
876
- self.alerts[name].active = False
877
- else:
878
-
879
- self.active = False
880
- for alert in self.alerts.values():
881
- alert.active = False
882
-
883
- if self.thread.is_alive():
884
- self.thread.join()
885
-
886
- def run(self):
887
- """ The alert monitoring loop
888
- """
889
- self.active = True
890
-
891
- while self.active:
892
- for alert in self.alerts.values():
893
- if alert.active:
894
- try:
895
- alert.update()
896
- except Exception as ex:
897
- logger.info(ex)
898
- time.sleep(0.1)
899
-
900
-
901
-
902
-
903
- class AlertManagerProxy(Proxy, AlertManagerInterface):
904
- """
905
- Proxy for Alert Management, used to connect to the Alert Manager
906
- Control Server and send commands remotely.
907
- """
908
-
909
- def __init__(
910
- self,
911
- protocol=CTRL_SETTINGS.PROTOCOL,
912
- hostname=CTRL_SETTINGS.HOSTNAME,
913
- port=CTRL_SETTINGS.COMMANDING_PORT,
914
- ):
915
- """Initialisation of a new Proxy for Alert Management.
916
-
917
- If no connection details (transport protocol, hostname, and port) are
918
- not provided, these are taken from the settings file.
919
-
920
- Args:
921
- - protocol: Transport protocol [default is taken from settings
922
- file].
923
- - hostname: Location of the control server (IP address) [default
924
- is taken from settings file].
925
- - port: TCP port on which the Control Server is listening for
926
- commands [default is taken from settings file].
927
- """
928
-
929
- super().__init__(connect_address(protocol, hostname, port))
930
-
931
-
932
-
933
- class AlertManagerProtocol(CommandProtocol):
934
- """
935
- Command Protocol for Alert Management.
936
- """
937
- def __init__(self, control_server: ControlServer, phase):
938
- """Initialisation of a new Protocol for Alert Management.
939
-
940
- The initialisation of this Protocol consists of the following steps:
941
-
942
- - create a Controller to which the given Control Server should send commands;
943
- - load the commands;
944
- - build a look-up table for the commands.
945
-
946
- Args:
947
- - control_server: Control Server via which commands should be sent
948
- to the Controller.
949
- """
950
-
951
- super().__init__()
952
-
953
- # Control Server for Alert Management
954
-
955
- self.control_server = control_server
956
-
957
- # Create a new Controller for Alert Management
958
-
959
- self.controller = AlertManagerController(phase)
960
-
961
- # Load the commands (for commanding of the AM Controller) from the
962
- # YAML file into a dictionary, stored in the AM Protocol
963
-
964
- self.load_commands(
965
- DEVICE_SETTINGS.Commands, AlertManagerCommand, AlertManagerController
966
- )
967
-
968
- # Build a look-up table for the methods
969
-
970
- self.build_device_method_lookup_table(self.controller)
971
-
972
-
973
- def get_bind_address(self):
974
- """Returns the address to bind a socket to.
975
-
976
- This bind address is a properly formatted URL, based on the
977
- communication protocol and the commanding port.
978
-
979
- Returns:
980
- - Properly formatted URL to bind a socket to.
981
- """
982
-
983
- return bind_address(
984
- self.control_server.get_communication_protocol(),
985
- self.control_server.get_commanding_port(),
986
- )
987
-
988
- def get_status(self) -> dict:
989
- """Returns the status information for the Control Server.
990
-
991
- This status information is returned in the form of a dictionary that
992
- contains the following information about the Control Server for
993
- Alert Management:
994
-
995
- - timestamp (str): string representation of the current datetime;
996
- - PID (int): process ID for the Control Server;
997
- - Up (float): uptime of the Control Server [s];
998
- - UUID (uuid1): Universally Unique Identifier for the Control
999
- Server;
1000
- - RSS (int): 'Resident Set Size', this is the non-swapped physical
1001
- memory a process has used [byte];
1002
- - USS (int): 'Unique Set Size', this is the memory which is unique
1003
- to a process [byte];
1004
- - CPU User (float): time spent in user mode [s];
1005
- - CPU System (float): time spent in kernel mode [s];
1006
- - CPU count: number of CPU cores in use by the process;
1007
- - CPU% (float): process CPU utilization as a percentage [%].
1008
- - Alert Status (dict): The status of all active alerts
1009
- - Alert Phase (int): The enumerate value of the current phase
1010
-
1011
- Returns:
1012
- - Dictionary with status information for the Control Server for
1013
- Alert Management.
1014
- """
1015
-
1016
- status = super().get_status()
1017
-
1018
- status['alert_status'] = self.controller.status()
1019
- status['phase'] = self.controller.phase
1020
-
1021
- return status
1022
-
1023
- def get_housekeeping(self) -> dict:
1024
- """Returns the housekeeping data for the Control Server.
1025
-
1026
- This housekeeping data is returns in the form of a dictionary that
1027
- contains the following information about the Control Server for
1028
- Alert Management:
1029
-
1030
- - timestamp (str): string representation of the current datetime.
1031
-
1032
- Returns:
1033
- - Dictionary with housekeeping data for the Control Server for
1034
- Alert Management.
1035
- """
1036
-
1037
- return {"timestamp": format_datetime()}
1038
-
1039
- def quit(self):
1040
- self.controller.quit()
1041
-
1042
-
1043
-
1044
- def main():
1045
-
1046
- dev = AlertManagerController()
1047
-
1048
- if __name__ == "__main__":
1049
- main()