cgse 2024.7.0__py3-none-any.whl → 2025.0.1__py3-none-any.whl
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- README.md +27 -0
- bump.py +85 -0
- cgse-2025.0.1.dist-info/METADATA +38 -0
- cgse-2025.0.1.dist-info/RECORD +5 -0
- {cgse-2024.7.0.dist-info → cgse-2025.0.1.dist-info}/WHEEL +1 -2
- cgse-2024.7.0.dist-info/COPYING +0 -674
- cgse-2024.7.0.dist-info/COPYING.LESSER +0 -165
- cgse-2024.7.0.dist-info/METADATA +0 -144
- cgse-2024.7.0.dist-info/RECORD +0 -660
- cgse-2024.7.0.dist-info/entry_points.txt +0 -75
- cgse-2024.7.0.dist-info/top_level.txt +0 -2
- egse/__init__.py +0 -12
- egse/__main__.py +0 -32
- egse/aeu/aeu.py +0 -5238
- egse/aeu/aeu_awg.yaml +0 -265
- egse/aeu/aeu_crio.yaml +0 -273
- egse/aeu/aeu_cs.py +0 -627
- egse/aeu/aeu_devif.py +0 -321
- egse/aeu/aeu_main_ui.py +0 -903
- egse/aeu/aeu_metrics.py +0 -131
- egse/aeu/aeu_protocol.py +0 -463
- egse/aeu/aeu_psu.yaml +0 -204
- egse/aeu/aeu_ui.py +0 -873
- egse/aeu/arbdata/FccdRead.arb +0 -2
- egse/aeu/arbdata/FccdRead_min_points.arb +0 -2
- egse/aeu/arbdata/HeaterSync_FccdRead.arb +0 -2
- egse/aeu/arbdata/HeaterSync_ccdRead25.arb +0 -2
- egse/aeu/arbdata/HeaterSync_ccdRead31_25.arb +0 -2
- egse/aeu/arbdata/HeaterSync_ccdRead37_50.arb +0 -2
- egse/aeu/arbdata/HeaterSync_ccdRead43_75.arb +0 -2
- egse/aeu/arbdata/HeaterSync_ccdRead50.arb +0 -2
- egse/aeu/arbdata/Heater_FccdRead_min_points.arb +0 -2
- egse/aeu/arbdata/ccdRead25.arb +0 -2
- egse/aeu/arbdata/ccdRead25_150ms.arb +0 -2
- egse/aeu/arbdata/ccdRead31_25.arb +0 -2
- egse/aeu/arbdata/ccdRead31_25_150ms.arb +0 -2
- egse/aeu/arbdata/ccdRead37_50.arb +0 -2
- egse/aeu/arbdata/ccdRead37_50_150ms.arb +0 -2
- egse/aeu/arbdata/ccdRead43_75.arb +0 -2
- egse/aeu/arbdata/ccdRead43_75_150ms.arb +0 -2
- egse/aeu/arbdata/ccdRead50.arb +0 -2
- egse/aeu/arbdata/ccdRead50_150ms.arb +0 -2
- egse/alert/__init__.py +0 -1049
- egse/alert/alertman.yaml +0 -37
- egse/alert/alertman_cs.py +0 -233
- egse/alert/alertman_ui.py +0 -600
- egse/alert/gsm/beaglebone.py +0 -138
- egse/alert/gsm/beaglebone.yaml +0 -51
- egse/alert/gsm/beaglebone_cs.py +0 -108
- egse/alert/gsm/beaglebone_devif.py +0 -122
- egse/alert/gsm/beaglebone_protocol.py +0 -46
- egse/bits.py +0 -318
- egse/camera.py +0 -44
- egse/collimator/__init__.py +0 -0
- egse/collimator/fcul/__init__.py +0 -0
- egse/collimator/fcul/ogse.py +0 -1077
- egse/collimator/fcul/ogse.yaml +0 -14
- egse/collimator/fcul/ogse_cs.py +0 -154
- egse/collimator/fcul/ogse_devif.py +0 -358
- egse/collimator/fcul/ogse_protocol.py +0 -132
- egse/collimator/fcul/ogse_sim.py +0 -431
- egse/collimator/fcul/ogse_ui.py +0 -1108
- egse/command.py +0 -699
- egse/config.py +0 -410
- egse/confman/__init__.py +0 -1058
- egse/confman/confman.yaml +0 -70
- egse/confman/confman_cs.py +0 -240
- egse/confman/confman_ui.py +0 -381
- egse/confman/setup_ui.py +0 -565
- egse/control.py +0 -632
- egse/coordinates/__init__.py +0 -534
- egse/coordinates/avoidance.py +0 -100
- egse/coordinates/cslmodel.py +0 -127
- egse/coordinates/laser_tracker_to_dict.py +0 -122
- egse/coordinates/point.py +0 -707
- egse/coordinates/pyplot.py +0 -194
- egse/coordinates/referenceFrame.py +0 -1279
- egse/coordinates/refmodel.py +0 -737
- egse/coordinates/rotationMatrix.py +0 -85
- egse/coordinates/transform3d_addon.py +0 -419
- egse/csl/__init__.py +0 -50
- egse/csl/commanding.py +0 -78
- egse/csl/icons/hexapod-connected-selected.svg +0 -30
- egse/csl/icons/hexapod-connected.svg +0 -30
- egse/csl/icons/hexapod-homing-selected.svg +0 -68
- egse/csl/icons/hexapod-homing.svg +0 -68
- egse/csl/icons/hexapod-retract-selected.svg +0 -56
- egse/csl/icons/hexapod-retract.svg +0 -51
- egse/csl/icons/hexapod-zero-selected.svg +0 -56
- egse/csl/icons/hexapod-zero.svg +0 -56
- egse/csl/icons/logo-puna.svg +0 -92
- egse/csl/icons/stop.svg +0 -1
- egse/csl/initialisation.py +0 -102
- egse/csl/mech_pos_settings.yaml +0 -18
- egse/das.py +0 -1240
- egse/das.yaml +0 -7
- egse/data/conf/SETUP_CSL_00000_170620_150000.yaml +0 -5
- egse/data/conf/SETUP_CSL_00001_170620_151010.yaml +0 -69
- egse/data/conf/SETUP_CSL_00002_170620_151020.yaml +0 -69
- egse/data/conf/SETUP_CSL_00003_170620_151030.yaml +0 -69
- egse/data/conf/SETUP_CSL_00004_170620_151040.yaml +0 -69
- egse/data/conf/SETUP_CSL_00005_170620_151050.yaml +0 -69
- egse/data/conf/SETUP_CSL_00006_170620_151060.yaml +0 -69
- egse/data/conf/SETUP_CSL_00007_170620_151070.yaml +0 -69
- egse/data/conf/SETUP_CSL_00008_170620_151080.yaml +0 -75
- egse/data/conf/SETUP_CSL_00010_210308_083016.yaml +0 -138
- egse/data/conf/SETUP_INTA_00000_170620_150000.yaml +0 -4
- egse/data/conf/SETUP_SRON_00000_170620_150000.yaml +0 -4
- egse/decorators.py +0 -514
- egse/device.py +0 -269
- egse/dpu/__init__.py +0 -2698
- egse/dpu/ccd_ui.py +0 -514
- egse/dpu/dpu.py +0 -783
- egse/dpu/dpu.yaml +0 -153
- egse/dpu/dpu_cs.py +0 -272
- egse/dpu/dpu_ui.py +0 -671
- egse/dpu/fitsgen.py +0 -2096
- egse/dpu/fitsgen_ui.py +0 -399
- egse/dpu/hdf5_model.py +0 -332
- egse/dpu/hdf5_ui.py +0 -277
- egse/dpu/hdf5_viewer.py +0 -506
- egse/dpu/hk_ui.py +0 -468
- egse/dpu_commands.py +0 -81
- egse/dsi/__init__.py +0 -33
- egse/dsi/_libesl.py +0 -232
- egse/dsi/constants.py +0 -296
- egse/dsi/esl.py +0 -630
- egse/dsi/rmap.py +0 -444
- egse/dsi/rmapci.py +0 -39
- egse/dsi/spw.py +0 -335
- egse/dsi/spw_state.py +0 -29
- egse/dummy.py +0 -318
- egse/dyndummy.py +0 -179
- egse/env.py +0 -278
- egse/exceptions.py +0 -88
- egse/fdir/__init__.py +0 -26
- egse/fdir/fdir_manager.py +0 -85
- egse/fdir/fdir_manager.yaml +0 -37
- egse/fdir/fdir_manager_controller.py +0 -136
- egse/fdir/fdir_manager_cs.py +0 -164
- egse/fdir/fdir_manager_interface.py +0 -15
- egse/fdir/fdir_remote.py +0 -73
- egse/fdir/fdir_remote.yaml +0 -30
- egse/fdir/fdir_remote_controller.py +0 -30
- egse/fdir/fdir_remote_cs.py +0 -94
- egse/fdir/fdir_remote_interface.py +0 -9
- egse/fdir/fdir_remote_popup.py +0 -26
- egse/fee/__init__.py +0 -106
- egse/fee/f_fee_register.yaml +0 -43
- egse/fee/feesim.py +0 -914
- egse/fee/n_fee_hk.py +0 -768
- egse/fee/nfee.py +0 -188
- egse/filterwheel/__init__.py +0 -4
- egse/filterwheel/eksma/__init__.py +0 -49
- egse/filterwheel/eksma/fw8smc4.py +0 -657
- egse/filterwheel/eksma/fw8smc4.yaml +0 -121
- egse/filterwheel/eksma/fw8smc4_cs.py +0 -144
- egse/filterwheel/eksma/fw8smc4_devif.py +0 -473
- egse/filterwheel/eksma/fw8smc4_protocol.py +0 -82
- egse/filterwheel/eksma/fw8smc4_ui.py +0 -940
- egse/filterwheel/eksma/fw8smc5.py +0 -115
- egse/filterwheel/eksma/fw8smc5.yaml +0 -105
- egse/filterwheel/eksma/fw8smc5_controller.py +0 -307
- egse/filterwheel/eksma/fw8smc5_cs.py +0 -141
- egse/filterwheel/eksma/fw8smc5_interface.py +0 -65
- egse/filterwheel/eksma/fw8smc5_simulator.py +0 -29
- egse/filterwheel/eksma/fw8smc5_ui.py +0 -1065
- egse/filterwheel/eksma/testpythonfw.py +0 -215
- egse/fov/__init__.py +0 -65
- egse/fov/fov_hk.py +0 -710
- egse/fov/fov_ui.py +0 -859
- egse/fov/fov_ui_controller.py +0 -140
- egse/fov/fov_ui_model.py +0 -200
- egse/fov/fov_ui_view.py +0 -345
- egse/gimbal/__init__.py +0 -32
- egse/gimbal/symetrie/__init__.py +0 -26
- egse/gimbal/symetrie/alpha.py +0 -586
- egse/gimbal/symetrie/generic_gimbal_ui.py +0 -1521
- egse/gimbal/symetrie/gimbal.py +0 -877
- egse/gimbal/symetrie/gimbal.yaml +0 -168
- egse/gimbal/symetrie/gimbal_cs.py +0 -183
- egse/gimbal/symetrie/gimbal_protocol.py +0 -138
- egse/gimbal/symetrie/gimbal_ui.py +0 -361
- egse/gimbal/symetrie/pmac.py +0 -1006
- egse/gimbal/symetrie/pmac_regex.py +0 -83
- egse/graph.py +0 -132
- egse/gui/__init__.py +0 -47
- egse/gui/buttons.py +0 -378
- egse/gui/focalplane.py +0 -1285
- egse/gui/formatter.py +0 -10
- egse/gui/led.py +0 -162
- egse/gui/limitswitch.py +0 -143
- egse/gui/mechanisms.py +0 -587
- egse/gui/states.py +0 -148
- egse/gui/stripchart.py +0 -729
- egse/gui/styles.qss +0 -48
- egse/gui/switch.py +0 -112
- egse/h5.py +0 -274
- egse/help/__init__.py +0 -0
- egse/help/help_ui.py +0 -126
- egse/hexapod/__init__.py +0 -32
- egse/hexapod/symetrie/__init__.py +0 -137
- egse/hexapod/symetrie/alpha.py +0 -874
- egse/hexapod/symetrie/dynalpha.py +0 -1387
- egse/hexapod/symetrie/hexapod_ui.py +0 -1516
- egse/hexapod/symetrie/pmac.py +0 -1010
- egse/hexapod/symetrie/pmac_regex.py +0 -83
- egse/hexapod/symetrie/puna.py +0 -1167
- egse/hexapod/symetrie/puna.yaml +0 -193
- egse/hexapod/symetrie/puna_cs.py +0 -195
- egse/hexapod/symetrie/puna_protocol.py +0 -134
- egse/hexapod/symetrie/puna_ui.py +0 -433
- egse/hexapod/symetrie/punaplus.py +0 -107
- egse/hexapod/symetrie/zonda.py +0 -872
- egse/hexapod/symetrie/zonda.yaml +0 -337
- egse/hexapod/symetrie/zonda_cs.py +0 -172
- egse/hexapod/symetrie/zonda_devif.py +0 -414
- egse/hexapod/symetrie/zonda_protocol.py +0 -123
- egse/hexapod/symetrie/zonda_ui.py +0 -449
- egse/hk.py +0 -791
- egse/icons/aeu-cs-start.svg +0 -117
- egse/icons/aeu-cs-stop.svg +0 -118
- egse/icons/aeu-cs.svg +0 -107
- egse/icons/aeu_cs-started.svg +0 -112
- egse/icons/aeu_cs-stopped.svg +0 -112
- egse/icons/aeu_cs.svg +0 -55
- egse/icons/alert.svg +0 -1
- egse/icons/arrow-double-left.png +0 -0
- egse/icons/arrow-double-right.png +0 -0
- egse/icons/arrow-up.svg +0 -11
- egse/icons/backward.svg +0 -1
- egse/icons/busy.svg +0 -1
- egse/icons/cleaning.svg +0 -115
- egse/icons/color-scheme.svg +0 -1
- egse/icons/cs-connected-alert.svg +0 -91
- egse/icons/cs-connected-disabled.svg +0 -43
- egse/icons/cs-connected.svg +0 -89
- egse/icons/cs-not-connected.svg +0 -44
- egse/icons/double-left-arrow.svg +0 -1
- egse/icons/double-right-arrow.svg +0 -1
- egse/icons/erase-disabled.svg +0 -19
- egse/icons/erase.svg +0 -59
- egse/icons/fitsgen-start.svg +0 -47
- egse/icons/fitsgen-stop.svg +0 -48
- egse/icons/fitsgen.svg +0 -1
- egse/icons/forward.svg +0 -1
- egse/icons/fov-hk-start.svg +0 -33
- egse/icons/fov-hk-stop.svg +0 -37
- egse/icons/fov-hk.svg +0 -1
- egse/icons/front-desk.svg +0 -1
- egse/icons/home-actioned.svg +0 -15
- egse/icons/home-disabled.svg +0 -15
- egse/icons/home.svg +0 -13
- egse/icons/info.svg +0 -1
- egse/icons/invalid.png +0 -0
- egse/icons/led-green.svg +0 -20
- egse/icons/led-grey.svg +0 -20
- egse/icons/led-orange.svg +0 -20
- egse/icons/led-red.svg +0 -20
- egse/icons/led-square-green.svg +0 -134
- egse/icons/led-square-grey.svg +0 -134
- egse/icons/led-square-orange.svg +0 -134
- egse/icons/led-square-red.svg +0 -134
- egse/icons/limit-switch-all-green.svg +0 -115
- egse/icons/limit-switch-all-red.svg +0 -117
- egse/icons/limit-switch-el+.svg +0 -116
- egse/icons/limit-switch-el-.svg +0 -117
- egse/icons/location-marker.svg +0 -1
- egse/icons/logo-dpu.svg +0 -48
- egse/icons/logo-gimbal.svg +0 -112
- egse/icons/logo-huber.svg +0 -23
- egse/icons/logo-ogse.svg +0 -31
- egse/icons/logo-puna.svg +0 -92
- egse/icons/logo-tcs.svg +0 -29
- egse/icons/logo-zonda.svg +0 -66
- egse/icons/maximize.svg +0 -1
- egse/icons/meter.svg +0 -1
- egse/icons/more.svg +0 -45
- egse/icons/n-fee-hk-start.svg +0 -24
- egse/icons/n-fee-hk-stop.svg +0 -25
- egse/icons/n-fee-hk.svg +0 -83
- egse/icons/observing-off.svg +0 -46
- egse/icons/observing-on.svg +0 -46
- egse/icons/open-document-hdf5.png +0 -0
- egse/icons/open-document-hdf5.svg +0 -21
- egse/icons/ops-mode.svg +0 -1
- egse/icons/play-green.svg +0 -17
- egse/icons/plugged-disabled.svg +0 -27
- egse/icons/plugged.svg +0 -21
- egse/icons/pm_ui.svg +0 -1
- egse/icons/power-button-green.svg +0 -27
- egse/icons/power-button-red.svg +0 -27
- egse/icons/power-button.svg +0 -27
- egse/icons/radar.svg +0 -1
- egse/icons/radioactive.svg +0 -2
- egse/icons/reload.svg +0 -1
- egse/icons/remote-control-off.svg +0 -28
- egse/icons/remote-control-on.svg +0 -28
- egse/icons/repeat-blue.svg +0 -15
- egse/icons/repeat.svg +0 -1
- egse/icons/settings.svg +0 -1
- egse/icons/shrink.svg +0 -1
- egse/icons/shutter.svg +0 -1
- egse/icons/sign-off.svg +0 -1
- egse/icons/sign-on.svg +0 -1
- egse/icons/sim-mode.svg +0 -1
- egse/icons/small-buttons-go.svg +0 -20
- egse/icons/small-buttons-minus.svg +0 -51
- egse/icons/small-buttons-plus.svg +0 -51
- egse/icons/sponge.svg +0 -220
- egse/icons/start-button-disabled.svg +0 -84
- egse/icons/start-button.svg +0 -50
- egse/icons/stop-button-disabled.svg +0 -84
- egse/icons/stop-button.svg +0 -50
- egse/icons/stop-red.svg +0 -17
- egse/icons/stop.svg +0 -1
- egse/icons/switch-disabled-square.svg +0 -87
- egse/icons/switch-disabled.svg +0 -15
- egse/icons/switch-off-square.svg +0 -87
- egse/icons/switch-off.svg +0 -72
- egse/icons/switch-on-square.svg +0 -87
- egse/icons/switch-on.svg +0 -61
- egse/icons/temperature-control.svg +0 -44
- egse/icons/th_ui_logo.svg +0 -1
- egse/icons/unplugged.svg +0 -23
- egse/icons/unvalid.png +0 -0
- egse/icons/user-interface.svg +0 -1
- egse/icons/vacuum.svg +0 -1
- egse/icons/valid.png +0 -0
- egse/icons/zoom-to-pixel-dark.svg +0 -64
- egse/icons/zoom-to-pixel-white.svg +0 -36
- egse/images/big-rotation-stage.png +0 -0
- egse/images/connected-100.png +0 -0
- egse/images/cross.svg +0 -6
- egse/images/disconnected-100.png +0 -0
- egse/images/gui-icon.png +0 -0
- egse/images/home.svg +0 -6
- egse/images/info-icon.png +0 -0
- egse/images/led-black.svg +0 -89
- egse/images/led-green.svg +0 -85
- egse/images/led-orange.svg +0 -85
- egse/images/led-red.svg +0 -85
- egse/images/load-icon.png +0 -0
- egse/images/load-setup.png +0 -0
- egse/images/load.png +0 -0
- egse/images/pause.png +0 -0
- egse/images/play-button.svg +0 -8
- egse/images/play.png +0 -0
- egse/images/process-status.png +0 -0
- egse/images/restart.png +0 -0
- egse/images/search.png +0 -0
- egse/images/sma.png +0 -0
- egse/images/start.png +0 -0
- egse/images/stop-button.svg +0 -8
- egse/images/stop.png +0 -0
- egse/images/switch-off.svg +0 -48
- egse/images/switch-on.svg +0 -48
- egse/images/undo.png +0 -0
- egse/images/update-button.svg +0 -11
- egse/imageviewer/exposureselection.py +0 -475
- egse/imageviewer/imageviewer.py +0 -198
- egse/imageviewer/matchfocalplane.py +0 -179
- egse/imageviewer/subfieldposition.py +0 -133
- egse/lampcontrol/__init__.py +0 -4
- egse/lampcontrol/beaglebone/beaglebone.py +0 -178
- egse/lampcontrol/beaglebone/beaglebone.yaml +0 -62
- egse/lampcontrol/beaglebone/beaglebone_cs.py +0 -106
- egse/lampcontrol/beaglebone/beaglebone_devif.py +0 -150
- egse/lampcontrol/beaglebone/beaglebone_protocol.py +0 -73
- egse/lampcontrol/energetiq/__init__.py +0 -22
- egse/lampcontrol/energetiq/eq99.yaml +0 -98
- egse/lampcontrol/energetiq/lampEQ99.py +0 -283
- egse/lampcontrol/energetiq/lampEQ99_cs.py +0 -128
- egse/lampcontrol/energetiq/lampEQ99_devif.py +0 -158
- egse/lampcontrol/energetiq/lampEQ99_encode_decode_errors.py +0 -73
- egse/lampcontrol/energetiq/lampEQ99_protocol.py +0 -71
- egse/lampcontrol/energetiq/lampEQ99_ui.py +0 -465
- egse/lib/CentOS-7/EtherSpaceLink_v34_86.dylib +0 -0
- egse/lib/CentOS-8/ESL-RMAP_v34_86.dylib +0 -0
- egse/lib/CentOS-8/EtherSpaceLink_v34_86.dylib +0 -0
- egse/lib/Debian/ESL-RMAP_v34_86.dylib +0 -0
- egse/lib/Debian/EtherSpaceLink_v34_86.dylib +0 -0
- egse/lib/Debian/libetherspacelink_v35_21.dylib +0 -0
- egse/lib/Linux/ESL-RMAP_v34_86.dylib +0 -0
- egse/lib/Linux/EtherSpaceLink_v34_86.dylib +0 -0
- egse/lib/Ubuntu-20/ESL-RMAP_v34_86.dylib +0 -0
- egse/lib/Ubuntu-20/EtherSpaceLink_v34_86.dylib +0 -0
- egse/lib/gssw/python3-gssw_2.2.3+31f63c9f-1_all.deb +0 -0
- egse/lib/ximc/__pycache__/pyximc.cpython-38 2.pyc +0 -0
- egse/lib/ximc/__pycache__/pyximc.cpython-38.pyc +0 -0
- egse/lib/ximc/libximc.framework/Frameworks/libbindy.dylib +0 -0
- egse/lib/ximc/libximc.framework/Frameworks/libxiwrapper.dylib +0 -0
- egse/lib/ximc/libximc.framework/Headers/ximc.h +0 -5510
- egse/lib/ximc/libximc.framework/Resources/Info.plist +0 -42
- egse/lib/ximc/libximc.framework/Resources/keyfile.sqlite +0 -0
- egse/lib/ximc/libximc.framework/libbindy.so +0 -0
- egse/lib/ximc/libximc.framework/libximc +0 -0
- egse/lib/ximc/libximc.framework/libximc.so +0 -0
- egse/lib/ximc/libximc.framework/libximc.so.7.0.0 +0 -0
- egse/lib/ximc/libximc.framework/libxiwrapper.so +0 -0
- egse/lib/ximc/pyximc.py +0 -922
- egse/listener.py +0 -179
- egse/logger/__init__.py +0 -243
- egse/logger/log_cs.py +0 -321
- egse/metrics.py +0 -102
- egse/mixin.py +0 -464
- egse/monitoring.py +0 -95
- egse/ni/alarms/__init__.py +0 -26
- egse/ni/alarms/cdaq9375.py +0 -300
- egse/ni/alarms/cdaq9375.yaml +0 -89
- egse/ni/alarms/cdaq9375_cs.py +0 -130
- egse/ni/alarms/cdaq9375_devif.py +0 -183
- egse/ni/alarms/cdaq9375_protocol.py +0 -48
- egse/obs_inspection.py +0 -165
- egse/observer.py +0 -41
- egse/obsid.py +0 -163
- egse/powermeter/__init__.py +0 -0
- egse/powermeter/ni/__init__.py +0 -38
- egse/powermeter/ni/cdaq9184.py +0 -224
- egse/powermeter/ni/cdaq9184.yaml +0 -73
- egse/powermeter/ni/cdaq9184_cs.py +0 -130
- egse/powermeter/ni/cdaq9184_devif.py +0 -201
- egse/powermeter/ni/cdaq9184_protocol.py +0 -48
- egse/powermeter/ni/cdaq9184_ui.py +0 -544
- egse/powermeter/thorlabs/__init__.py +0 -25
- egse/powermeter/thorlabs/pm100a.py +0 -380
- egse/powermeter/thorlabs/pm100a.yaml +0 -132
- egse/powermeter/thorlabs/pm100a_cs.py +0 -136
- egse/powermeter/thorlabs/pm100a_devif.py +0 -127
- egse/powermeter/thorlabs/pm100a_protocol.py +0 -80
- egse/powermeter/thorlabs/pm100a_ui.py +0 -725
- egse/process.py +0 -451
- egse/procman/__init__.py +0 -834
- egse/procman/cannot_start_process_popup.py +0 -43
- egse/procman/procman.yaml +0 -49
- egse/procman/procman_cs.py +0 -201
- egse/procman/procman_ui.py +0 -2081
- egse/protocol.py +0 -605
- egse/proxy.py +0 -531
- egse/randomwalk.py +0 -140
- egse/reg.py +0 -585
- egse/reload.py +0 -122
- egse/reprocess.py +0 -693
- egse/resource.py +0 -333
- egse/rmap.py +0 -406
- egse/rst.py +0 -135
- egse/search.py +0 -182
- egse/serialdevice.py +0 -190
- egse/services.py +0 -247
- egse/services.yaml +0 -68
- egse/settings.py +0 -379
- egse/settings.yaml +0 -980
- egse/setup.py +0 -1181
- egse/shutter/__init__.py +0 -0
- egse/shutter/thorlabs/__init__.py +0 -19
- egse/shutter/thorlabs/ksc101.py +0 -205
- egse/shutter/thorlabs/ksc101.yaml +0 -105
- egse/shutter/thorlabs/ksc101_cs.py +0 -136
- egse/shutter/thorlabs/ksc101_devif.py +0 -201
- egse/shutter/thorlabs/ksc101_protocol.py +0 -71
- egse/shutter/thorlabs/ksc101_ui.py +0 -548
- egse/shutter/thorlabs/sc10.py +0 -82
- egse/shutter/thorlabs/sc10.yaml +0 -52
- egse/shutter/thorlabs/sc10_controller.py +0 -81
- egse/shutter/thorlabs/sc10_cs.py +0 -108
- egse/shutter/thorlabs/sc10_interface.py +0 -25
- egse/shutter/thorlabs/sc10_simulator.py +0 -30
- egse/simulator.py +0 -41
- egse/slack.py +0 -61
- egse/socketdevice.py +0 -218
- egse/sockets.py +0 -218
- egse/spw.py +0 -1401
- egse/stages/__init__.py +0 -12
- egse/stages/aerotech/ensemble.py +0 -245
- egse/stages/aerotech/ensemble.yaml +0 -205
- egse/stages/aerotech/ensemble_controller.py +0 -275
- egse/stages/aerotech/ensemble_cs.py +0 -110
- egse/stages/aerotech/ensemble_interface.py +0 -132
- egse/stages/aerotech/ensemble_parameters.py +0 -433
- egse/stages/aerotech/ensemble_simulator.py +0 -27
- egse/stages/aerotech/mgse_sim.py +0 -188
- egse/stages/arun/smd3.py +0 -110
- egse/stages/arun/smd3.yaml +0 -68
- egse/stages/arun/smd3_controller.py +0 -470
- egse/stages/arun/smd3_cs.py +0 -112
- egse/stages/arun/smd3_interface.py +0 -53
- egse/stages/arun/smd3_simulator.py +0 -27
- egse/stages/arun/smd3_stop.py +0 -16
- egse/stages/huber/__init__.py +0 -49
- egse/stages/huber/smc9300.py +0 -920
- egse/stages/huber/smc9300.yaml +0 -63
- egse/stages/huber/smc9300_cs.py +0 -178
- egse/stages/huber/smc9300_devif.py +0 -345
- egse/stages/huber/smc9300_protocol.py +0 -113
- egse/stages/huber/smc9300_sim.py +0 -547
- egse/stages/huber/smc9300_ui.py +0 -973
- egse/state.py +0 -173
- egse/statemachine.py +0 -274
- egse/storage/__init__.py +0 -1067
- egse/storage/persistence.py +0 -2295
- egse/storage/storage.yaml +0 -79
- egse/storage/storage_cs.py +0 -231
- egse/styles/dark.qss +0 -343
- egse/styles/default.qss +0 -48
- egse/synoptics/__init__.py +0 -417
- egse/synoptics/syn.yaml +0 -9
- egse/synoptics/syn_cs.py +0 -195
- egse/system.py +0 -1611
- egse/tcs/__init__.py +0 -14
- egse/tcs/tcs.py +0 -879
- egse/tcs/tcs.yaml +0 -14
- egse/tcs/tcs_cs.py +0 -202
- egse/tcs/tcs_devif.py +0 -292
- egse/tcs/tcs_protocol.py +0 -180
- egse/tcs/tcs_sim.py +0 -177
- egse/tcs/tcs_ui.py +0 -543
- egse/tdms.py +0 -171
- egse/tempcontrol/__init__.py +0 -23
- egse/tempcontrol/agilent/agilent34970.py +0 -109
- egse/tempcontrol/agilent/agilent34970.yaml +0 -44
- egse/tempcontrol/agilent/agilent34970_cs.py +0 -114
- egse/tempcontrol/agilent/agilent34970_devif.py +0 -182
- egse/tempcontrol/agilent/agilent34970_protocol.py +0 -96
- egse/tempcontrol/agilent/agilent34972.py +0 -111
- egse/tempcontrol/agilent/agilent34972.yaml +0 -44
- egse/tempcontrol/agilent/agilent34972_cs.py +0 -115
- egse/tempcontrol/agilent/agilent34972_devif.py +0 -189
- egse/tempcontrol/agilent/agilent34972_protocol.py +0 -98
- egse/tempcontrol/beaglebone/beaglebone.py +0 -341
- egse/tempcontrol/beaglebone/beaglebone.yaml +0 -110
- egse/tempcontrol/beaglebone/beaglebone_cs.py +0 -117
- egse/tempcontrol/beaglebone/beaglebone_protocol.py +0 -134
- egse/tempcontrol/beaglebone/beaglebone_ui.py +0 -674
- egse/tempcontrol/digalox/digalox.py +0 -115
- egse/tempcontrol/digalox/digalox.yaml +0 -36
- egse/tempcontrol/digalox/digalox_cs.py +0 -108
- egse/tempcontrol/digalox/digalox_protocol.py +0 -56
- egse/tempcontrol/keithley/__init__.py +0 -33
- egse/tempcontrol/keithley/daq6510.py +0 -662
- egse/tempcontrol/keithley/daq6510.yaml +0 -105
- egse/tempcontrol/keithley/daq6510_cs.py +0 -163
- egse/tempcontrol/keithley/daq6510_devif.py +0 -343
- egse/tempcontrol/keithley/daq6510_protocol.py +0 -79
- egse/tempcontrol/keithley/daq6510_sim.py +0 -186
- egse/tempcontrol/lakeshore/__init__.py +0 -33
- egse/tempcontrol/lakeshore/lsci.py +0 -361
- egse/tempcontrol/lakeshore/lsci.yaml +0 -162
- egse/tempcontrol/lakeshore/lsci_cs.py +0 -174
- egse/tempcontrol/lakeshore/lsci_devif.py +0 -292
- egse/tempcontrol/lakeshore/lsci_protocol.py +0 -76
- egse/tempcontrol/lakeshore/lsci_ui.py +0 -387
- egse/tempcontrol/ni/__init__.py +0 -0
- egse/tempcontrol/spid/spid.py +0 -109
- egse/tempcontrol/spid/spid.yaml +0 -81
- egse/tempcontrol/spid/spid_controller.py +0 -279
- egse/tempcontrol/spid/spid_cs.py +0 -136
- egse/tempcontrol/spid/spid_protocol.py +0 -107
- egse/tempcontrol/spid/spid_ui.py +0 -723
- egse/tempcontrol/srs/__init__.py +0 -22
- egse/tempcontrol/srs/ptc10.py +0 -867
- egse/tempcontrol/srs/ptc10.yaml +0 -227
- egse/tempcontrol/srs/ptc10_cs.py +0 -128
- egse/tempcontrol/srs/ptc10_devif.py +0 -116
- egse/tempcontrol/srs/ptc10_protocol.py +0 -39
- egse/tempcontrol/srs/ptc10_ui.py +0 -906
- egse/ups/apc/apc.py +0 -236
- egse/ups/apc/apc.yaml +0 -45
- egse/ups/apc/apc_cs.py +0 -101
- egse/ups/apc/apc_protocol.py +0 -125
- egse/user.yaml +0 -7
- egse/vacuum/beaglebone/beaglebone.py +0 -149
- egse/vacuum/beaglebone/beaglebone.yaml +0 -44
- egse/vacuum/beaglebone/beaglebone_cs.py +0 -108
- egse/vacuum/beaglebone/beaglebone_devif.py +0 -159
- egse/vacuum/beaglebone/beaglebone_protocol.py +0 -192
- egse/vacuum/beaglebone/beaglebone_ui.py +0 -638
- egse/vacuum/instrutech/igm402.py +0 -91
- egse/vacuum/instrutech/igm402.yaml +0 -90
- egse/vacuum/instrutech/igm402_controller.py +0 -124
- egse/vacuum/instrutech/igm402_cs.py +0 -108
- egse/vacuum/instrutech/igm402_interface.py +0 -49
- egse/vacuum/instrutech/igm402_simulator.py +0 -36
- egse/vacuum/keller/kellerBus.py +0 -256
- egse/vacuum/keller/leo3.py +0 -100
- egse/vacuum/keller/leo3.yaml +0 -38
- egse/vacuum/keller/leo3_controller.py +0 -81
- egse/vacuum/keller/leo3_cs.py +0 -101
- egse/vacuum/keller/leo3_interface.py +0 -33
- egse/vacuum/mks/evision.py +0 -86
- egse/vacuum/mks/evision.yaml +0 -75
- egse/vacuum/mks/evision_cs.py +0 -101
- egse/vacuum/mks/evision_devif.py +0 -313
- egse/vacuum/mks/evision_interface.py +0 -60
- egse/vacuum/mks/evision_simulator.py +0 -24
- egse/vacuum/mks/evision_ui.py +0 -701
- egse/vacuum/pfeiffer/acp40.py +0 -87
- egse/vacuum/pfeiffer/acp40.yaml +0 -60
- egse/vacuum/pfeiffer/acp40_controller.py +0 -117
- egse/vacuum/pfeiffer/acp40_cs.py +0 -109
- egse/vacuum/pfeiffer/acp40_interface.py +0 -40
- egse/vacuum/pfeiffer/acp40_simulator.py +0 -37
- egse/vacuum/pfeiffer/tc400.py +0 -87
- egse/vacuum/pfeiffer/tc400.yaml +0 -83
- egse/vacuum/pfeiffer/tc400_controller.py +0 -136
- egse/vacuum/pfeiffer/tc400_cs.py +0 -109
- egse/vacuum/pfeiffer/tc400_interface.py +0 -70
- egse/vacuum/pfeiffer/tc400_simulator.py +0 -35
- egse/vacuum/pfeiffer/tpg261.py +0 -80
- egse/vacuum/pfeiffer/tpg261.yaml +0 -66
- egse/vacuum/pfeiffer/tpg261_controller.py +0 -150
- egse/vacuum/pfeiffer/tpg261_cs.py +0 -109
- egse/vacuum/pfeiffer/tpg261_interface.py +0 -59
- egse/vacuum/pfeiffer/tpg261_simulator.py +0 -23
- egse/version.py +0 -174
- egse/visitedpositions.py +0 -398
- egse/windowing.py +0 -213
- egse/zmq/__init__.py +0 -28
- egse/zmq/spw.py +0 -160
- egse/zmq_ser.py +0 -41
- scripts/alerts/cold.yaml +0 -278
- scripts/alerts/example_alerts.yaml +0 -54
- scripts/alerts/transition.yaml +0 -14
- scripts/alerts/warm.yaml +0 -49
- scripts/analyse_n_fee_hk_data.py +0 -52
- scripts/check_hdf5_files.py +0 -192
- scripts/check_register_sync.py +0 -47
- scripts/check_tcs_calib_coef.py +0 -90
- scripts/correct_ccd_cold_temperature_cal.py +0 -157
- scripts/create_hdf5_report.py +0 -293
- scripts/csl_model.py +0 -420
- scripts/csl_restore_setup.py +0 -229
- scripts/export-grafana-dashboards.py +0 -49
- scripts/fdir/cs_recovery/fdir_cs_recovery.py +0 -54
- scripts/fdir/fdir_table.yaml +0 -70
- scripts/fdir/fdir_test_recovery.py +0 -10
- scripts/fdir/hw_recovery/fdir_agilent_hw_recovery.py +0 -73
- scripts/fdir/limit_recovery/fdir_agilent_limit.py +0 -61
- scripts/fdir/limit_recovery/fdir_bb_heater_limit.py +0 -59
- scripts/fdir/limit_recovery/fdir_ensemble_limit.py +0 -33
- scripts/fdir/limit_recovery/fdir_pressure_limit_recovery.py +0 -71
- scripts/fix_csv.py +0 -80
- scripts/ias/correct_ccd_temp_cal_elfique.py +0 -43
- scripts/ias/correct_ccd_temp_cal_floreffe.py +0 -43
- scripts/ias/correct_trp_swap_achel.py +0 -199
- scripts/inta/correct_ccd_temp_cal_duvel.py +0 -43
- scripts/inta/correct_ccd_temp_cal_gueuze.py +0 -43
- scripts/n_fee_supply_voltage_calculation.py +0 -92
- scripts/playground.py +0 -30
- scripts/print_hdf5_hk_data.py +0 -68
- scripts/print_register_map.py +0 -43
- scripts/remove_lines_between_matches.py +0 -188
- scripts/sron/commanding/control_heaters.py +0 -44
- scripts/sron/commanding/pumpdown.py +0 -46
- scripts/sron/commanding/set_pid_setpoint.py +0 -19
- scripts/sron/commanding/shutdown_bbb_heaters.py +0 -10
- scripts/sron/commanding/shutdown_pumps.py +0 -33
- scripts/sron/correct_mgse_coordinates_brigand_chimay.py +0 -272
- scripts/sron/correct_trp_swap_brigand.py +0 -204
- scripts/sron/gimbal_conversions.py +0 -75
- scripts/sron/tm_gen/tm_gen_agilent.py +0 -37
- scripts/sron/tm_gen/tm_gen_heaters.py +0 -4
- scripts/sron/tm_gen/tm_gen_spid.py +0 -13
- scripts/update_operational_cgse.py +0 -268
- 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
|