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/tcs/tcs.py DELETED
@@ -1,879 +0,0 @@
1
- """
2
- This module defines the device classes to be used to connect to and control the TCS EGSE.
3
- """
4
- import datetime
5
- import itertools
6
- import logging
7
- import multiprocessing
8
- import operator
9
- from collections import namedtuple
10
- from enum import IntEnum
11
- from pathlib import Path
12
- from typing import Any
13
- from typing import Dict
14
- from typing import List
15
- from typing import Optional
16
- from typing import Union
17
-
18
- from prometheus_client import start_http_server
19
- from rich.console import Console
20
- from rich.table import Table
21
-
22
- from egse.control import Failure
23
- from egse.control import is_control_server_active
24
- from egse.device import DeviceConnectionState
25
- from egse.device import DeviceInterface
26
- from egse.hk import read_conversion_dict
27
- from egse.metrics import define_metrics
28
- from egse.mixin import DynamicCommandMixin
29
- from egse.mixin import ETX
30
- from egse.mixin import add_etx
31
- from egse.mixin import dynamic_command
32
- from egse.proxy import DynamicProxy
33
- from egse.settings import Settings
34
- from egse.setup import load_setup, Setup
35
- from egse.sockets import SocketInterface
36
- from egse.storage import StorageProxy
37
- from egse.storage import register_to_storage_manager
38
- from egse.storage import unregister_from_storage_manager
39
- from egse.storage.persistence import CSV
40
- from egse.synoptics import SynopticsManagerProxy
41
- from egse.system import SignalCatcher
42
- from egse.system import format_datetime
43
- from egse.tcs.tcs_devif import TCSEthernetInterface
44
- from egse.zmq_ser import connect_address
45
-
46
- logger = logging.getLogger(__name__)
47
-
48
- TCS_SETTINGS = Settings.load("TCS Controller")
49
- CTRL_SETTINGS = Settings.load("TCS Control Server")
50
- DEVICE_SETTINGS = Settings.load(filename="tcs.yaml")
51
-
52
- THIS_FILE_LOCATION = Path(__file__).parent
53
-
54
- ORIGIN = "TCS-HK"
55
-
56
- REMOTE_MODE_NOT_ACTIVE = 'not_allowed_remote_mode_not_active'
57
-
58
- REQUEST_TIMEOUT = 10_000 # 10s timeout should be more than enough
59
-
60
- HousekeepingValue = namedtuple("HousekeepingValue", "timestamp name value")
61
-
62
-
63
- def decode_response(response: bytes) -> str:
64
- """Decodes the bytes object. Doesn't perform any other conversions on the response."""
65
- return response.decode()
66
-
67
-
68
- def decode_and_strip_response(response: bytes) -> str:
69
- """Decodes the bytes object and strips off the last ETX character."""
70
- return response.decode().rstrip(ETX)
71
-
72
-
73
- def process_request_remote_response(response: bytes) -> bool:
74
- """Returns True when the remote operation is active or could be activated."""
75
- return b"acknowledge_remote_operation" in response
76
-
77
-
78
- def process_quit_remote_response(response: bytes) -> bool:
79
- """Returns True when the remote operation is inactive or could be inactivated."""
80
- return b"acknowledge_quit_remote_operation" in response
81
-
82
-
83
- def process_is_remote_operation_response(response: bytes) -> bool:
84
- """Returns True when the remote operation is active or could be activated."""
85
- return b"remote_mode_active" in response
86
-
87
-
88
- def process_is_task_running_response(response: bytes) -> Union[bool, Failure]:
89
- """
90
- Returns True when a task is running on the TCS EGSE.
91
-
92
- When an error occurred, a warning messages is logged and a Failure is returned.
93
- """
94
- is_task_running = process_get_housekeeping_value(response)
95
- if isinstance(is_task_running, Failure):
96
- return is_task_running
97
- elif is_task_running.value == 'tbd':
98
- return Failure("Cannot determine if task is running, TBD was returned.")
99
- else:
100
- return bool(int(is_task_running.value))
101
-
102
-
103
- def process_commit_response(response: bytes):
104
- if b"erroneous" in response:
105
- logger.warning("The TCS EGSE commit has automatically corrected some of the "
106
- "configuration parameters that were provided.")
107
- return decode_and_strip_response(response)
108
-
109
-
110
- def expand_kwargs(kv_pairs: Dict):
111
- return " ".join(f"{k} {v}" for k, v in kv_pairs.items())
112
-
113
-
114
- def process_error_response(response: bytes) -> List:
115
-
116
- data = response.decode().split('\x03')
117
-
118
- # Remove empty items from the list
119
-
120
- data = [x for x in data if x]
121
-
122
- data = [x for y in data for x in y.split('\r\n') if x]
123
-
124
- return data
125
-
126
-
127
- def process_all_housekeeping(response: bytes) -> Union[List[List], Failure]:
128
- """
129
- Process the response from the TCS EGSE remote command `get_TM_All`.
130
-
131
- Args:
132
- response (bytes): contains the response from the TCS EGSE to the get_TM_All command.
133
-
134
- Returns:
135
- A nested list where the inner lists contain the name, timestamp and value of a housekeeping
136
- parameter. The list is sorted on the name.
137
-
138
- A Failure object is returned when the remote control mode is de-activated or when a
139
- communication error occurred.
140
- """
141
-
142
- data = response.decode().split('\x03')
143
-
144
- if "not_allowed_remote_mode_not_active" in data:
145
- msg = "Remote Control not active, no housekeeping values received."
146
- return Failure(msg)
147
-
148
- if len(data) != 2:
149
- msg = "Format error: no housekeeping value received."
150
- logger.error(msg)
151
- return Failure(msg)
152
-
153
- data = data[0].split('\r\n')
154
- data = [x.split('\t') for x in data]
155
- data = sorted(data, key=operator.itemgetter(0))
156
-
157
- return data
158
-
159
-
160
- def process_get_housekeeping_value(response: bytes) -> Union[HousekeepingValue, Failure]:
161
-
162
- if b"remote_mode_not_active" in response:
163
- logger.warning("Requesting housekeeping value invalid, not in remote control mode.")
164
- return Failure("Requesting housekeeping value invalid, not in remote control mode.")
165
- if b"invalid_tm_item_id" in response:
166
- logger.warning("Invalid housekeeping parameter name.")
167
- return Failure("Invalid housekeeping parameter name.")
168
- data = response.decode().split('\x03')
169
- data = data[0].split('\t')
170
- return HousekeepingValue(*data)
171
-
172
-
173
- def process_configuration(response: bytes) -> Union[Dict, Failure]:
174
- """
175
- Process the response to the get_configuration command. Processing in this context means to
176
- disentangle the returned string and split it up in parameter name, value pairs.
177
-
178
- Args:
179
- response (str): a string containing the response from the TCS EGSE
180
-
181
- Returns:
182
- A dictionary with the parameters names and their current values.
183
- """
184
-
185
- # The rest of this function parse and organise the configuration parameters
186
- # in a list of lists with [[name, value], [...]]
187
-
188
- tcs_conf = response.decode().split('\x03')
189
-
190
- # Remove empty items from the list. The list ends always with an empty item because
191
- # the response ends with the '\x03' EOT character.
192
-
193
- tcs_conf = [x for x in tcs_conf if x]
194
-
195
- if not tcs_conf:
196
- logger.info('No response received from the get_Conf command to TCS EGSE.')
197
- return {}
198
-
199
- # The last line is what we need, we assume there is only one line, but just to be sure...
200
-
201
- tcs_conf = tcs_conf[-1]
202
-
203
- if REMOTE_MODE_NOT_ACTIVE in tcs_conf:
204
- return Failure(
205
- "Remote mode is not active on the TCS EGSE, commanding is not possible, "
206
- "you can still retrieve telemetry."
207
- )
208
-
209
- # All configuration parameters are separated by '\r\n', and the name and value
210
- # are separated by a '\t', so let's split them into a list of [value, name].
211
-
212
- tcs_conf = tcs_conf.split('\r\n')
213
-
214
- # The configuration parameters also contain the RTD parameters which is organised
215
- # differently, so we threat them differently and append them to the configuration
216
- # parameters again later.
217
-
218
- rtd_list = [x for x in tcs_conf if x.startswith("rtd_list") or x.startswith('\t')]
219
- rtd_list = [x.split('\t') for x in rtd_list]
220
- rtd_list = [x[1:] for x in rtd_list]
221
- rtd_list = [[f"{x[0]}_A_B_C_D_E_F", f"{x[1]}, {x[2]}, {x[3]}, {x[4]}, {x[5]}, {x[6]}"] for x in
222
- rtd_list]
223
-
224
- # Now process each of the 'normal' configuration parameters, and
225
-
226
- tcs_conf = [x for x in tcs_conf if not (x.startswith("rtd_list") or x.startswith('\t'))]
227
- tcs_conf = [x.split('\t') for x in tcs_conf if x]
228
-
229
- # add the RTD list back into the configuration parameters.
230
-
231
- tcs_conf.extend(rtd_list)
232
-
233
- # We want to sort the final list by name, this order will be preserved when we convert it
234
- # into a dictionary.
235
-
236
- tcs_conf = sorted(tcs_conf, key=operator.itemgetter(0))
237
-
238
- # Now convert the above list into a dictionary.
239
-
240
- tcs_conf = {k: v for k, v in tcs_conf}
241
-
242
- return tcs_conf
243
-
244
-
245
- class OperatingMode(IntEnum):
246
- NORMAL = 0
247
- SAFE = 1
248
- DECONTAMINATION = 2
249
- CALIBRATION = 3
250
- EMC = 4
251
- SELF_TEST = 5
252
- EXTENDED = 6
253
-
254
-
255
- class ClosedLoopMode(IntEnum):
256
- PI_ALG_1 = 0
257
- PI_ALG_2 = 1
258
- BANG_BANG_ALG_1 = 2
259
- BANG_BANG_ALG_2 = 3
260
-
261
-
262
- class TCSError(Exception):
263
- """A TCS EGSE specific error."""
264
-
265
- pass
266
-
267
-
268
- def is_tcs_cs_active(timeout: float = 2.0):
269
- """Check if the TCS Control Server is running.
270
-
271
- Args:
272
- timeout (float): timeout when waiting for a reply [seconds, default=0.5]
273
- Returns:
274
- True if the Storage Manager is running and replied with the expected answer.
275
- """
276
-
277
- endpoint = connect_address(
278
- CTRL_SETTINGS.PROTOCOL, CTRL_SETTINGS.HOSTNAME, CTRL_SETTINGS.COMMANDING_PORT
279
- )
280
-
281
- return is_control_server_active(endpoint, timeout)
282
-
283
-
284
- def print_configuration():
285
- """
286
- This convenience function prints the configuration parameters in a table on your console.
287
- This function is for use in a REPL or Jupyter Notebook.
288
- """
289
- with TCSProxy() as tcs:
290
- tcs_conf = tcs.get_configuration()
291
-
292
- table = Table(title="Configuration")
293
-
294
- table.add_column("Name", justify="left", style="cyan", no_wrap=True)
295
- table.add_column("Value", justify="right", style="green")
296
-
297
- for name, value in sorted(tcs_conf.items()):
298
- table.add_row(name, value)
299
-
300
- console = Console()
301
- console.print(table)
302
-
303
-
304
- def print_all_housekeeping():
305
- """
306
- This convenience function prints the last sampled housekeeping parameters in a table in your
307
- console. This function is for use in a REPL or Jupyter Notebook.
308
-
309
- Note that when no task is running most of the housekeeping parameters are not sampled and the
310
- values will be out-of-date. To make sure you have up-to-date values, run the task.
311
- """
312
- with TCSProxy() as tcs:
313
- hk = tcs.get_all_housekeeping()
314
-
315
- table = Table(title="All Telemetry")
316
-
317
- table.add_column("Date Time", justify="center", style="cyan", no_wrap=True)
318
- table.add_column("Name", style="magenta")
319
- table.add_column("Value", justify="right", style="green")
320
-
321
- for name, date, value in hk:
322
- table.add_row(name, date, value)
323
-
324
- console = Console()
325
- console.print(table)
326
-
327
-
328
- def set_operating_mode(mode: str):
329
-
330
- with TCSProxy() as tcs:
331
- tcs.set_operating_mode(mode)
332
- tcs.commit()
333
-
334
-
335
- class TCSInterface(DeviceInterface):
336
- """
337
- Interface definition for the TCSController, the TCSProxy and the TCSSimulator.
338
- """
339
-
340
- @dynamic_command(cmd_type="query", cmd_string="request_remote_operation",
341
- process_cmd_string=add_etx, process_response=process_request_remote_response)
342
- def request_remote_operation(self) -> bool:
343
- """
344
- Activate the Remote Control Mode in the TCS EGSE.
345
-
346
- Returns:
347
- True, if Remote Control could be activated, False otherwise.
348
- """
349
- raise NotImplementedError
350
-
351
- @dynamic_command(cmd_type="query", cmd_string="quit_remote_operation",
352
- process_cmd_string=add_etx, process_response=process_quit_remote_response)
353
- def quit_remote_operation(self) -> bool:
354
- """
355
- Command the TCS EGSE to end the Remote Control Mode. The TCS EGSE will continue with the
356
- ongoing task in Local Control Mode.
357
-
358
- Returns:
359
- True, if Remote Control mode could be de-activated, False otherwise.
360
- """
361
- raise NotImplementedError
362
-
363
- @dynamic_command(cmd_type="query", cmd_string="get_remote_status",
364
- process_cmd_string=add_etx, process_response=process_is_remote_operation_response)
365
- def is_remote_operation_active(self) -> bool:
366
- """
367
- Check if Remote Control Mode is active in the TCS EGSE.
368
-
369
- Returns:
370
- True, if the command was acknowledged, False otherwise.
371
- """
372
- raise NotImplementedError
373
-
374
- @dynamic_command(cmd_type="query", cmd_string="get_tm task_is_running",
375
- process_cmd_string=add_etx, process_response=process_is_task_running_response)
376
- def is_task_running(self) -> bool:
377
- """
378
- Query the TCS EGSE if a task is currently running.
379
-
380
- Returns:
381
- True, if a task is running, False otherwise.
382
- """
383
- raise NotImplementedError
384
-
385
- @dynamic_command(cmd_type="query", cmd_string="clear_popup",
386
- process_cmd_string=add_etx, process_response=decode_and_strip_response)
387
- def clear_popup(self) -> str:
388
- """Clears the error/warning popup on the MMI. Returns an acknowledgement message."""
389
- raise NotImplementedError
390
-
391
- @dynamic_command(cmd_type="query", cmd_string="get_conf_all",
392
- process_cmd_string=add_etx, process_response=process_configuration)
393
- def get_configuration(self) -> Dict:
394
- """Returns the last valid configuration as a dictionary."""
395
- raise NotImplementedError
396
-
397
- @dynamic_command(cmd_type="query", cmd_string="get_tm_all",
398
- process_cmd_string=add_etx, process_response=process_all_housekeeping)
399
- def get_all_housekeeping(self) -> List[List[str]]:
400
- """
401
- Returns the last housekeeping data sampled by the TCS EGSE.
402
-
403
- Returns:
404
- A list of lists, where the inner list contains the parameter name, the sampling time,
405
- and the parameter value, in that order. All these values are returned as strings.
406
- """
407
- raise NotImplementedError
408
-
409
- @dynamic_command(cmd_type="query", cmd_string="get_tm ${hk_id}",
410
- process_cmd_string=add_etx, process_response=process_get_housekeeping_value)
411
- def get_housekeeping_value(self, hk_id: str) -> Optional[HousekeepingValue]:
412
- """Returns the last updated value for the housekeeping parameter.
413
-
414
- The returned value is a namedtuple with the following fields:
415
-
416
- * timestamp: a string in the format 'YYYY/mm/dd HH:MM:SS.%f UTC'
417
- * name: the name of the requested housekeeping value
418
- * value: as a string
419
-
420
- """
421
- raise NotImplementedError
422
-
423
- @dynamic_command(cmd_type="query", cmd_string="set_parameter operation_mode ${mode}",
424
- process_cmd_string=add_etx, process_response=decode_and_strip_response)
425
- def set_operating_mode(self, mode: Union[str, int]) -> str:
426
- """
427
- Define the operating mode of the task to be started.
428
-
429
- The mode can either be an integer or the text value of:
430
- 0. normal
431
- 1. safe
432
- 2. decontamination|de-icing
433
- 3. calibration
434
- 4. EMC
435
- 5. self-test
436
- 6. extended
437
-
438
- Args:
439
- mode (str, int): the required mode as defined above.
440
-
441
- Returns:
442
- No response.
443
- """
444
- raise NotImplementedError
445
-
446
- @dynamic_command(cmd_type="query", cmd_string="set_parameter ${name} ${value}",
447
- process_cmd_string=add_etx, process_response=decode_and_strip_response)
448
- def set_parameter(self, name: str, value: Any) -> str:
449
- """
450
- Set the parameter in the configuration.
451
-
452
- These parameters will be stored on a temporary buffer and will only be applied
453
- after the commit command is sent.
454
- """
455
- raise NotImplementedError
456
-
457
- @dynamic_command(cmd_type="query", cmd_string="set_parameter ${kwargs}",
458
- process_cmd_string=add_etx, process_kwargs=expand_kwargs,
459
- process_response=decode_and_strip_response)
460
- def set_parameters(self, **kwargs) -> str:
461
- """
462
- Set the given parameters in the configuration.
463
-
464
- The parameters are provided as a dictionary
465
-
466
- These parameters will be stored on a temporary buffer and will only be applied
467
- after the commit command is sent.
468
- """
469
- raise NotImplementedError
470
-
471
- @dynamic_command(cmd_type="query", cmd_string="commit_config",
472
- process_cmd_string=add_etx, process_response=process_commit_response)
473
- def commit(self) -> str:
474
- """
475
- Enables all received configuration parameters, check the parameters and
476
- saves them to the TCS local configuration file. If a parameter that was
477
- previously sent and is not in accordance to the range and resolution allowed,
478
- it will be coerced to the nearest value allowed. In this latter case, a Warning
479
- message will be logged.
480
- """
481
- raise NotImplementedError
482
-
483
- @dynamic_command(cmd_type="query", cmd_string="discard_config",
484
- process_cmd_string=add_etx, process_response=decode_and_strip_response)
485
- def discard(self) -> str:
486
- """
487
- Discards all received configuration parameters sent without commit command.
488
- """
489
- raise NotImplementedError
490
-
491
- @dynamic_command(
492
- cmd_type="query",
493
- cmd_string="upload_rtd_parameters ${sn}_${r0}_${a}_${b}_${c}",
494
- process_cmd_string=add_etx)
495
- def upload_callendar_vandusen_rtd_parameters(self, sn, r0, a, b, c):
496
- """
497
- Uploads the Callendar Van-Dussen parameters.
498
-
499
- The 'sn' argument is the 'serial number' or identifier for this set of parameters.
500
-
501
- Values of R0, A, B and C constants for 3th order polynomial equation
502
- calibration curves.
503
-
504
- These parameters are required for 2 wire RTDs.
505
- """
506
- raise NotImplementedError
507
-
508
- @dynamic_command(
509
- cmd_type="query",
510
- cmd_string="upload_rtd_parameters ${sn}_${a}_${b}_${c}_${d}_${e}_${f}",
511
- process_cmd_string=add_etx)
512
- def upload_polynomial_rtd_parameters(self, sn, a, b, c, d, e, f):
513
- """
514
- Uploads the 5th order polynomial calibration parameters.
515
-
516
- The 'sn' argument is the 'serial number' or identifier for this set of parameters.
517
-
518
- Values of A, B, C, D, E, F constants for 5th order polynomial equation
519
- calibration curves.
520
-
521
- These parameters are required for 4 wire RTDs.
522
- """
523
- raise NotImplementedError
524
-
525
- @dynamic_command(cmd_type="query", cmd_string="set_rtd_parameters ${rtd_id} ${sn}",
526
- process_cmd_string=add_etx)
527
- def set_rtd_parameters(self, rtd_id, sn):
528
- """
529
- Applies a previously uploaded set of RTD parameters, identified by the
530
- serial number (<sn>), to a given RTD channel (<rtd_id>).
531
-
532
- RTD parameters can be uploaded with the functions:
533
-
534
- * upload_callendar_vandusen_rtd_parameters()
535
- * upload_polynomial_rtd_parameters()
536
-
537
- """
538
- raise NotImplementedError
539
-
540
- @dynamic_command(cmd_type="query", cmd_string="run_task",
541
- process_cmd_string=add_etx, process_response=decode_and_strip_response)
542
- def run_task(self) -> str:
543
- """
544
- Stores previously sent configuration parameters and runs the task.
545
- """
546
- raise NotImplementedError
547
-
548
- @dynamic_command(cmd_type="query", cmd_string="stop_task",
549
- process_cmd_string=add_etx, process_response=decode_and_strip_response)
550
- def stop_task(self) -> str:
551
- """
552
- Stops the current running task (test, self-test, etc).
553
- """
554
- raise NotImplementedError
555
-
556
- @dynamic_command(cmd_type="query", cmd_string="get_error",
557
- process_cmd_string=add_etx, process_response=process_error_response)
558
- def get_error(self) -> List:
559
- """
560
- Returns a list of the active occurred errors.
561
- """
562
- raise NotImplementedError
563
-
564
-
565
- class TCSParameterNaming:
566
- """Defines the mapping between TCS EGSE device names and the CGSE names."""
567
- def __init__(self, origin: str, setup: Setup):
568
-
569
- # The hk_names_mapping is a dictionary that maps the original device telemetry parameter
570
- # names to the correct device names as defined in the CGSE. The keys in the mapping are
571
- # the original device name, the values are the CGSE corrected names.
572
-
573
- self.hk_names_mapping = read_conversion_dict(storage_mnemonic=origin, use_site=False, setup=setup)
574
-
575
- def get_all_cgse_names(self):
576
- """Returns the correct CGSE housekeeping parameter names."""
577
- return self.hk_names_mapping.values()
578
-
579
- def get_cgse_name(self, device_name: str):
580
- """
581
- Returns the CGSE name corresponding to the given device name.
582
- If no CGSE name exists, the device name is returned.
583
- """
584
- return self.hk_names_mapping.get(device_name, device_name)
585
-
586
-
587
- class TCSTelemetryMetrics:
588
- """Defines the metrics for the TCS EGSE that are maintained by the TCSTelemetry class."""
589
- def __init__(self, setup: Setup):
590
-
591
- self.hk_metrics = define_metrics(origin=ORIGIN, dashboard="TCS_TRP_MON", setup=setup)
592
- self.hk_metrics_status = define_metrics(origin=ORIGIN, dashboard="TCS_STATUS_MON", setup=setup)
593
-
594
- def update_metrics(self, name: str, value: str):
595
- """Update the TCS metric parameter with the given value."""
596
- if name in self.hk_metrics:
597
- self.hk_metrics[name].set(float(value))
598
- elif name in self.hk_metrics_status:
599
- if '.' in value:
600
- self.hk_metrics_status[name].set(float(value))
601
- else:
602
- self.hk_metrics_status[name].set(int(value))
603
-
604
-
605
- class TCSTelemetry(multiprocessing.Process):
606
- def __init__(self,
607
- command_queue: multiprocessing.Queue,
608
- response_queue: multiprocessing.Queue):
609
- super().__init__()
610
- self._command_q = command_queue
611
- self._response_q = response_queue
612
- self._column_names = None
613
- self._housekeeping = {}
614
- self._killer = None
615
- self._metrics: TCSTelemetryMetrics = None
616
- self._hk_names: TCSParameterNaming = None
617
- self.setup = load_setup()
618
-
619
- def run(self):
620
-
621
- # The TCS Telemetry runs in a different process and since ZeroMQ Sockets are not
622
- # thread/process safe, we have to recreate the ZeroMQHandler attached to the egse.logger
623
- # in this process.
624
-
625
- import egse.logger
626
- egse.logger.replace_zmq_handler()
627
-
628
- self._killer = SignalCatcher()
629
-
630
-
631
- # Device naming is different from CGSE naming, the conversion is handled in the
632
- # TCSParameterNaming class.
633
-
634
- self._hk_names = TCSParameterNaming(origin=ORIGIN, setup=self.setup)
635
-
636
- # Metrics must be defined only when the sub-process already exists, otherwise a TypeError
637
- # will be raised, indicating that a thread lock cannot be pickled
638
-
639
- self._metrics = TCSTelemetryMetrics(self.setup)
640
-
641
- # These are only 61 of the telemetry parameters of the TCS EGSE. It is not possible to know
642
- # all the parameters beforehand since they are only published when changed. So, eventually,
643
- # more HK parameters will be available, but the CSV implementation does not allow to add
644
- # them after the file creation. In the TCS CSV the corrected CGSE naming is used.
645
-
646
- self._column_names = list(self._hk_names.get_all_cgse_names())
647
-
648
- # Prepend the timestamp (_ts) for each of the telemetry parameters.
649
- # Start with a reference timestamp
650
-
651
- self._column_names = list(
652
- (
653
- "timestamp",
654
- *itertools.chain.from_iterable((f"{x}_ts", x) for x in self._column_names)
655
- )
656
- )
657
-
658
- # The HTTP server is usually started from the control server, only in this case, the
659
- # metrics will be updated from this process and the server has to be started from here.
660
-
661
- start_http_server(CTRL_SETTINGS.METRICS_PORT)
662
-
663
- with SocketInterface(hostname=TCS_SETTINGS.HOSTNAME, port=TCS_SETTINGS.TELEMETRY_PORT) \
664
- as socket, StorageProxy() as storage:
665
-
666
- register_to_storage_manager(origin=CTRL_SETTINGS.STORAGE_TM_MNEMONIC,
667
- persistence_class=CSV,
668
- prep={"mode": "a", "column_names": self._column_names})
669
- while True:
670
- data = socket.read()
671
- self.process_telemetry(data)
672
- # logger.info(f"{data=}")
673
- # logger.info(f"{self._housekeeping=}")
674
- # logger.info(f"{self._housekeeping.keys()}")
675
- storage.save(
676
- {
677
- "origin": CTRL_SETTINGS.STORAGE_TM_MNEMONIC,
678
- "data": self._housekeeping,
679
- }
680
- )
681
-
682
- with SynopticsManagerProxy() as synoptics:
683
- synoptics.store_common_synoptics(
684
- dict(**self._housekeeping))
685
-
686
- if self._killer.term_signal_received:
687
- break
688
-
689
- unregister_from_storage_manager(origin=CTRL_SETTINGS.STORAGE_TM_MNEMONIC)
690
-
691
- def process_telemetry(self, data: bytes):
692
- """
693
- Process the housekeeping telemetry that was received from the TCS EGSE.
694
-
695
- Args:
696
- data (str): a string containing the telemetry from the TCS EGSE.
697
-
698
- Returns:
699
- A dictionary where the key is the housekeeping parameter name and the value is a list
700
- containing the timestamp, name, and value of the housekeeping parameter. Only the last
701
- sample in kept in the dictionary.
702
- """
703
-
704
- self._housekeeping["timestamp"] = format_datetime()
705
-
706
- data = data.decode().split('\x03')
707
- data = [x for x in data if x]
708
- if not data:
709
- logger.warning("Format error: no new housekeeping values received.")
710
- return
711
- data = data[0].split('\r\n')
712
- data = [x.split('\t') for x in data]
713
-
714
- # We do not need to sort by timestamp since the data is already sorted by time.
715
- # data = sorted(data, key=operator.itemgetter(0)) # sort by date
716
-
717
- # We noticed that the telemetry transmission is not fail safe and sometimes data is missing
718
- # from the stream. That will result in an IndexError because one or more fields are missing.
719
- # When the date is incomplete due to transmission errors and cannot be parsed, a ValueError
720
- # is raised and silently ignored here.
721
-
722
- name = value = "not-yet-set"
723
- for x in data:
724
- try:
725
- date = convert_date(x[0])
726
- # replace name with the CGSE mnemonic from the TM dictionary
727
- name = self._hk_names.get_cgse_name(x[1])
728
- value = x[2] # extract_value(x[1], x[2])
729
- self._housekeeping[f"{name}_ts"] = date
730
- self._housekeeping[name] = value
731
- self._metrics.update_metrics(name, value)
732
- except (IndexError, ValueError) as exc:
733
- logger.info(f"{exc=!s} for {name=}, {value=}")
734
-
735
-
736
- def convert_date(date: str):
737
- """
738
- Convert the datetime string that is sent by the TCS EGSE to a format that is used by the CGSE.
739
-
740
- Args:
741
- date (str): datetime as "YYYY/MM/DD HH:MM:SS.ms UTC"
742
-
743
- Returns:
744
- A date string in the format "DD.MM.YYYY HH:MM:SS"
745
-
746
- Raises:
747
- ValueError when the date string can not be parsed.
748
- """
749
- try:
750
- dt = datetime.datetime.strptime(date, "%Y/%m/%d %H:%M:%S.%f utc")
751
- return format_datetime(dt)
752
- except ValueError as exc:
753
- logger.warning(f"Could not process date: {date=}, {exc=}")
754
- raise
755
-
756
-
757
- class TCSController(TCSInterface, DynamicCommandMixin):
758
- """
759
- The constructor sets the connection parameters but there is no attempt to connect
760
- to the controller. Make sure that you explicitly use the `tcs.connect()` and `tcs.disconnect()`
761
- commands before communicating with the TCS EGSE.
762
-
763
- The controller can also be used as a context manager, in which case the `connect()`
764
- and `disconnect()` methods should not be called:
765
-
766
- with TCSController() as tcs:
767
- tcs.get_configuration()
768
-
769
- """
770
-
771
- def __init__(self, hostname=TCS_SETTINGS.HOSTNAME, port=TCS_SETTINGS.COMMANDING_PORT):
772
- """
773
- Opens a TCP/IP socket connection with the TCS EGSE Hardware Controller.
774
-
775
- Args:
776
- hostname (str): the IP address or fully qualified hostname of the TCS hardware
777
- controller. The default is defined in the `settings.yaml` configuration file.
778
-
779
- port (int): the IP port number to connect to, by default set in the `settings.yaml`
780
- configuration file.
781
-
782
- Raises:
783
- TCSError: when the connection could not be established for some reason.
784
- """
785
-
786
- super().__init__()
787
-
788
- logger.debug(f"Initializing TCSController with hostname={hostname} on port={port}")
789
-
790
- try:
791
- self.transport = self.tcs = TCSEthernetInterface(hostname, port)
792
- except TCSError as exc:
793
- logger.warning(
794
- f"TCSError: Couldn't establish connection with the TCS EGSE Hardware "
795
- f"Controller: ({exc})"
796
- )
797
-
798
- def is_simulator(self):
799
- return False
800
-
801
- def is_connected(self):
802
- return self.tcs.is_connected()
803
-
804
- def connect(self):
805
- try:
806
- self.tcs.connect()
807
- except TCSError as exc:
808
- logger.warning(f"TCSError caught: Couldn't establish connection ({exc})")
809
- raise ConnectionError("Couldn't establish a connection with the TCS EGSE.") from exc
810
-
811
- self.notify_observers(DeviceConnectionState.DEVICE_CONNECTED)
812
-
813
- def disconnect(self):
814
- try:
815
- self.tcs.disconnect()
816
- except TCSError as exc:
817
- raise ConnectionError("Couldn't disconnect from TCS EGSE.") from exc
818
-
819
- self.notify_observers(DeviceConnectionState.DEVICE_NOT_CONNECTED)
820
-
821
- def reconnect(self):
822
- if self.is_connected():
823
- self.disconnect()
824
- self.connect()
825
-
826
-
827
- class TCSSimulator(TCSInterface):
828
- """
829
- Use the tcs_sim process instead of this class!
830
- """
831
- pass
832
-
833
-
834
- class TCSProxy(DynamicProxy, TCSInterface):
835
- """
836
- The TCSProxy class is used to connect to the control server and send commands to the
837
- TCS EGSE remotely.
838
- """
839
-
840
- def __init__(
841
- self,
842
- protocol=CTRL_SETTINGS.PROTOCOL,
843
- hostname=CTRL_SETTINGS.HOSTNAME,
844
- port=CTRL_SETTINGS.COMMANDING_PORT,
845
- timeout=REQUEST_TIMEOUT
846
- ):
847
- """
848
- Args:
849
- protocol: the transport protocol [default is taken from settings file]
850
- hostname: location of the control server (IP address) [default is taken from settings
851
- file]
852
- port: TCP port on which the control server is listening for commands [default is
853
- taken from settings file]
854
- """
855
- super().__init__(connect_address(protocol, hostname, port), timeout=timeout)
856
-
857
-
858
- def process_data(data: str) -> List[List[str]]:
859
- """
860
- Process the response from the TCS EGSE periodic telemetry.
861
-
862
- The output list may contains the same parameter several times.
863
- Args:
864
- data (str): a string containg the periodic telemetry sent out by the TCS EGSE.
865
-
866
- Returns:
867
- A nested list where the inner lists contain the name, timestamp and value of a telemetry
868
- parameter. The list is not sorted, but left in the order that they had in the response
869
- string.
870
- """
871
-
872
- data = data.split('\x03')
873
- data = [x for y in data for x in y.split('\r\n') if x]
874
- data = [x.split('\t') for x in data]
875
-
876
- # make sure the nested list is [name, date, value]
877
- data = [[x[1], x[0], x[2]] for x in data]
878
-
879
- return data