cgse 2023.38.0__py3-none-any.whl → 2024.1.4__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-2024.1.4.dist-info/METADATA +38 -0
- cgse-2024.1.4.dist-info/RECORD +5 -0
- {cgse-2023.38.0.dist-info → cgse-2024.1.4.dist-info}/WHEEL +1 -2
- cgse-2023.38.0.dist-info/COPYING +0 -674
- cgse-2023.38.0.dist-info/COPYING.LESSER +0 -165
- cgse-2023.38.0.dist-info/METADATA +0 -144
- cgse-2023.38.0.dist-info/RECORD +0 -649
- cgse-2023.38.0.dist-info/entry_points.txt +0 -75
- cgse-2023.38.0.dist-info/top_level.txt +0 -2
- egse/__init__.py +0 -12
- egse/__main__.py +0 -32
- egse/aeu/aeu.py +0 -5235
- egse/aeu/aeu_awg.yaml +0 -265
- egse/aeu/aeu_crio.yaml +0 -273
- egse/aeu/aeu_cs.py +0 -626
- egse/aeu/aeu_devif.py +0 -321
- egse/aeu/aeu_main_ui.py +0 -912
- 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 -234
- egse/alert/alertman_ui.py +0 -603
- 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 -130
- egse/alert/gsm/beaglebone_protocol.py +0 -48
- 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 -129
- 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 -1015
- egse/confman/confman.yaml +0 -67
- egse/confman/confman_cs.py +0 -239
- egse/confman/confman_ui.py +0 -381
- egse/confman/setup_ui.py +0 -565
- egse/control.py +0 -442
- egse/coordinates/__init__.py +0 -531
- egse/coordinates/avoidance.py +0 -103
- egse/coordinates/cslmodel.py +0 -127
- egse/coordinates/laser_tracker_to_dict.py +0 -120
- egse/coordinates/point.py +0 -707
- egse/coordinates/pyplot.py +0 -195
- 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 -1247
- 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 -415
- egse/device.py +0 -269
- egse/dpu/__init__.py +0 -2681
- egse/dpu/ccd_ui.py +0 -508
- egse/dpu/dpu.py +0 -786
- egse/dpu/dpu.yaml +0 -153
- egse/dpu/dpu_cs.py +0 -272
- egse/dpu/dpu_ui.py +0 -668
- egse/dpu/fitsgen.py +0 -2077
- egse/dpu/fitsgen_test.py +0 -752
- 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/constants.py +0 -220
- egse/dsi/esl.py +0 -870
- egse/dsi/rmap.py +0 -1042
- egse/dsi/rmapci.py +0 -37
- egse/dsi/spw.py +0 -154
- egse/dsi/spw_state.py +0 -29
- egse/dummy.py +0 -258
- egse/dyndummy.py +0 -179
- egse/env.py +0 -278
- egse/exceptions.py +0 -88
- egse/fdir/__init__.py +0 -28
- egse/fdir/fdir_manager.py +0 -85
- egse/fdir/fdir_manager.yaml +0 -51
- egse/fdir/fdir_manager_controller.py +0 -228
- egse/fdir/fdir_manager_cs.py +0 -164
- egse/fdir/fdir_manager_interface.py +0 -25
- egse/fdir/fdir_remote.py +0 -73
- egse/fdir/fdir_remote.yaml +0 -37
- egse/fdir/fdir_remote_controller.py +0 -50
- egse/fdir/fdir_remote_cs.py +0 -97
- egse/fdir/fdir_remote_interface.py +0 -14
- egse/fdir/fdir_remote_popup.py +0 -31
- egse/fee/__init__.py +0 -114
- egse/fee/f_fee_register.yaml +0 -43
- egse/fee/fee.py +0 -631
- egse/fee/feesim.py +0 -750
- egse/fee/n_fee_hk.py +0 -761
- egse/fee/nfee.py +0 -187
- egse/filterwheel/__init__.py +0 -4
- egse/filterwheel/eksma/__init__.py +0 -24
- egse/filterwheel/eksma/fw8smc4.py +0 -661
- 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 -81
- egse/filterwheel/eksma/fw8smc4_ui.py +0 -940
- egse/filterwheel/eksma/fw8smc5.py +0 -111
- 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 -1068
- egse/filterwheel/eksma/testpythonfw.py +0 -215
- egse/fov/__init__.py +0 -65
- egse/fov/fov_hk.py +0 -712
- egse/fov/fov_ui.py +0 -861
- 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 -135
- 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 -1281
- egse/gui/formatter.py +0 -10
- egse/gui/led.py +0 -162
- egse/gui/limitswitch.py +0 -143
- egse/gui/mechanisms.py +0 -588
- egse/gui/states.py +0 -148
- egse/gui/stripchart.py +0 -729
- 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 -138
- 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 -196
- egse/hexapod/symetrie/puna_protocol.py +0 -131
- egse/hexapod/symetrie/puna_ui.py +0 -434
- 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 -415
- egse/hexapod/symetrie/zonda_protocol.py +0 -119
- egse/hexapod/symetrie/zonda_ui.py +0 -449
- egse/hk.py +0 -765
- 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 -69
- 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/macOS/ESL-RMAP_v34_86.dylib +0 -0
- egse/lib/macOS/EtherSpaceLink_v34_86.dylib +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 -73
- egse/logger/__init__.py +0 -243
- egse/logger/log_cs.py +0 -321
- egse/metrics.py +0 -98
- 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 -163
- 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 -811
- 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 -603
- egse/proxy.py +0 -522
- egse/randomwalk.py +0 -140
- egse/reg.py +0 -585
- egse/reload.py +0 -122
- egse/reprocess.py +0 -675
- egse/resource.py +0 -333
- egse/rst.py +0 -135
- egse/search.py +0 -182
- egse/serialdevice.py +0 -190
- egse/services.py +0 -212
- egse/services.yaml +0 -51
- egse/settings.py +0 -379
- egse/settings.yaml +0 -980
- egse/setup.py +0 -1180
- 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 -69
- 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 -1479
- egse/stages/__init__.py +0 -12
- egse/stages/aerotech/ensemble.py +0 -247
- 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 -193
- egse/stages/arun/smd3.py +0 -111
- egse/stages/arun/smd3.yaml +0 -68
- egse/stages/arun/smd3_controller.py +0 -472
- 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 -904
- 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 -111
- 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 -1004
- egse/storage/persistence.py +0 -2295
- egse/storage/storage.yaml +0 -72
- egse/storage/storage_cs.py +0 -214
- egse/styles/dark.qss +0 -343
- egse/styles/default.qss +0 -48
- egse/synoptics/__init__.py +0 -412
- egse/synoptics/syn.yaml +0 -9
- egse/synoptics/syn_cs.py +0 -195
- egse/system.py +0 -1408
- egse/tcs/__init__.py +0 -14
- egse/tcs/tcs.py +0 -874
- 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 -177
- 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 -116
- egse/tempcontrol/agilent/agilent34970_devif.py +0 -182
- egse/tempcontrol/agilent/agilent34970_protocol.py +0 -99
- egse/tempcontrol/agilent/agilent34972.py +0 -111
- egse/tempcontrol/agilent/agilent34972.yaml +0 -44
- egse/tempcontrol/agilent/agilent34972_cs.py +0 -117
- egse/tempcontrol/agilent/agilent34972_devif.py +0 -189
- egse/tempcontrol/agilent/agilent34972_protocol.py +0 -101
- egse/tempcontrol/beaglebone/beaglebone.py +0 -342
- egse/tempcontrol/beaglebone/beaglebone.yaml +0 -110
- egse/tempcontrol/beaglebone/beaglebone_cs.py +0 -117
- egse/tempcontrol/beaglebone/beaglebone_protocol.py +0 -135
- egse/tempcontrol/beaglebone/beaglebone_ui.py +0 -681
- egse/tempcontrol/digalox/digalox.py +0 -107
- egse/tempcontrol/digalox/digalox.yaml +0 -36
- egse/tempcontrol/digalox/digalox_cs.py +0 -112
- egse/tempcontrol/digalox/digalox_protocol.py +0 -55
- 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 -78
- 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 -73
- egse/tempcontrol/lakeshore/lsci_ui.py +0 -389
- 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 -727
- egse/tempcontrol/srs/__init__.py +0 -22
- egse/tempcontrol/srs/ptc10.py +0 -875
- egse/tempcontrol/srs/ptc10.yaml +0 -227
- egse/tempcontrol/srs/ptc10_cs.py +0 -128
- egse/tempcontrol/srs/ptc10_devif.py +0 -118
- egse/tempcontrol/srs/ptc10_protocol.py +0 -42
- 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 -164
- egse/vacuum/beaglebone/beaglebone_protocol.py +0 -193
- egse/vacuum/beaglebone/beaglebone_ui.py +0 -638
- egse/vacuum/instrutech/igm402.py +0 -92
- egse/vacuum/instrutech/igm402.yaml +0 -90
- egse/vacuum/instrutech/igm402_controller.py +0 -128
- 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 -102
- egse/vacuum/keller/leo3.yaml +0 -38
- egse/vacuum/keller/leo3_controller.py +0 -83
- 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 -316
- egse/vacuum/mks/evision_interface.py +0 -60
- egse/vacuum/mks/evision_simulator.py +0 -24
- egse/vacuum/mks/evision_ui.py +0 -704
- 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 -39
- egse/vacuum/pfeiffer/tc400.py +0 -113
- egse/vacuum/pfeiffer/tc400.yaml +0 -83
- egse/vacuum/pfeiffer/tc400_controller.py +0 -140
- egse/vacuum/pfeiffer/tc400_cs.py +0 -109
- egse/vacuum/pfeiffer/tc400_interface.py +0 -70
- egse/vacuum/pfeiffer/tc400_simulator.py +0 -24
- egse/vacuum/pfeiffer/tpg261.py +0 -81
- 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 -60
- egse/vacuum/pfeiffer/tpg261_simulator.py +0 -24
- 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 -44
- scripts/check_hdf5_files.py +0 -192
- scripts/check_register_sync.py +0 -47
- scripts/create_hdf5_report.py +0 -295
- scripts/csl_model.py +0 -436
- scripts/csl_restore_setup.py +0 -230
- scripts/export-grafana-dashboards.py +0 -50
- scripts/fdir/cs_recovery/fdir_cs_recovery.py +0 -59
- scripts/fdir/fdir_table.yaml +0 -70
- scripts/fdir/fdir_test_recovery.py +0 -11
- scripts/fdir/hw_recovery/fdir_agilent_hw_recovery.py +0 -73
- scripts/fdir/limit_recovery/fdir_agilent_limit.py +0 -64
- scripts/fdir/limit_recovery/fdir_bb_heater_limit.py +0 -61
- 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/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/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/tm_gen/tm_gen_agilent.py +0 -38
- 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/protocol.py
DELETED
|
@@ -1,603 +0,0 @@
|
|
|
1
|
-
"""
|
|
2
|
-
CommandProtocol is a base class for communicating commands with the hardware or
|
|
3
|
-
the control server. This class implements methods to send command messages and
|
|
4
|
-
receive responses.
|
|
5
|
-
|
|
6
|
-
The protocol also knows how to load the commands from the YAML file that contains
|
|
7
|
-
command definitions.
|
|
8
|
-
|
|
9
|
-
"""
|
|
10
|
-
import abc
|
|
11
|
-
import inspect
|
|
12
|
-
import logging
|
|
13
|
-
import pickle
|
|
14
|
-
|
|
15
|
-
from prometheus_client import Counter
|
|
16
|
-
from prometheus_client import Summary
|
|
17
|
-
|
|
18
|
-
from egse.command import Command
|
|
19
|
-
from egse.command import CommandExecution
|
|
20
|
-
from egse.control import ControlServer
|
|
21
|
-
from egse.control import Failure
|
|
22
|
-
from egse.device import DeviceConnectionObserver
|
|
23
|
-
from egse.system import format_datetime
|
|
24
|
-
from egse.zmq_ser import bind_address
|
|
25
|
-
|
|
26
|
-
logger = logging.getLogger(__name__)
|
|
27
|
-
|
|
28
|
-
COMMAND_REQUESTS = Counter("cs_command_requests_count", "Count the number of commands", ["target"])
|
|
29
|
-
EXECUTION_TIME = Summary("cs_command_execution_time_seconds", "Time spent executing a command")
|
|
30
|
-
|
|
31
|
-
|
|
32
|
-
def get_method(parent_obj, method_name: str):
|
|
33
|
-
"""
|
|
34
|
-
Returns a bound method from a given class instance.
|
|
35
|
-
|
|
36
|
-
Args:
|
|
37
|
-
parent_obj: the class instance that provides the method
|
|
38
|
-
method_name: name of the method that is requested
|
|
39
|
-
|
|
40
|
-
Returns:
|
|
41
|
-
the method [type: method].
|
|
42
|
-
|
|
43
|
-
.. note::
|
|
44
|
-
The method returned is an bound class instance method and therefore
|
|
45
|
-
this method *does not* expects as its first argument the class
|
|
46
|
-
instance, i.e. self, when you call this as a function.
|
|
47
|
-
|
|
48
|
-
"""
|
|
49
|
-
if method_name is None or method_name == "None":
|
|
50
|
-
return None
|
|
51
|
-
|
|
52
|
-
if hasattr(parent_obj, method_name):
|
|
53
|
-
method = getattr(parent_obj, method_name)
|
|
54
|
-
if inspect.ismethod(method) or hasattr(method, "__method_wrapper"):
|
|
55
|
-
return method
|
|
56
|
-
logger.warning(f"{method_name} is not a method, type={type(method)}")
|
|
57
|
-
else:
|
|
58
|
-
logger.warning(f"{parent_obj!r} has no method called {method_name}")
|
|
59
|
-
|
|
60
|
-
return None
|
|
61
|
-
|
|
62
|
-
|
|
63
|
-
def get_function(parent_class, method_name: str):
|
|
64
|
-
"""
|
|
65
|
-
Returns a function (unbound method) from a given class.
|
|
66
|
-
|
|
67
|
-
Args:
|
|
68
|
-
parent_class: the class that provides the method
|
|
69
|
-
method_name: name of the method that is requested
|
|
70
|
-
|
|
71
|
-
Returns:
|
|
72
|
-
the method [type: function].
|
|
73
|
-
|
|
74
|
-
.. note::
|
|
75
|
-
The function returned is an unbound class instance method and
|
|
76
|
-
therefore this function expects as its first argument the class
|
|
77
|
-
instance, i.e. self, when you call it as a function.
|
|
78
|
-
|
|
79
|
-
"""
|
|
80
|
-
if method_name is None or method_name == "None":
|
|
81
|
-
return None
|
|
82
|
-
|
|
83
|
-
if hasattr(parent_class, method_name):
|
|
84
|
-
func = getattr(parent_class, method_name)
|
|
85
|
-
if inspect.isfunction(func):
|
|
86
|
-
return func
|
|
87
|
-
logger.warning(f"{method_name} is not a function, type={type(func)}")
|
|
88
|
-
else:
|
|
89
|
-
logger.warning(
|
|
90
|
-
f"{parent_class.__module__}.{parent_class.__name__} has no method called {method_name}"
|
|
91
|
-
)
|
|
92
|
-
|
|
93
|
-
return None
|
|
94
|
-
|
|
95
|
-
|
|
96
|
-
class BaseCommandProtocol(DeviceConnectionObserver):
|
|
97
|
-
|
|
98
|
-
def __init__(self, control_server: ControlServer):
|
|
99
|
-
super().__init__()
|
|
100
|
-
self.__socket = None
|
|
101
|
-
self.__control_server = control_server
|
|
102
|
-
|
|
103
|
-
def bind(self, socket):
|
|
104
|
-
"""Bind to a socket to listen for commands."""
|
|
105
|
-
self.__socket = socket
|
|
106
|
-
self.__socket.bind(self.get_bind_address())
|
|
107
|
-
|
|
108
|
-
def get_bind_address(self):
|
|
109
|
-
"""
|
|
110
|
-
Returns a string with the bind address, the endpoint, for accepting connections
|
|
111
|
-
and bind a socket to.
|
|
112
|
-
|
|
113
|
-
This method should be implemented by the sub-class since it contains the protocol
|
|
114
|
-
and port number that are specific for the sub-class.
|
|
115
|
-
|
|
116
|
-
Returns:
|
|
117
|
-
a string with the protocol and port to bind a socket to.
|
|
118
|
-
"""
|
|
119
|
-
return bind_address(
|
|
120
|
-
self.__control_server.get_communication_protocol(),
|
|
121
|
-
self.__control_server.get_commanding_port(),
|
|
122
|
-
)
|
|
123
|
-
|
|
124
|
-
def is_alive(self) -> bool:
|
|
125
|
-
"""
|
|
126
|
-
This method can be overridden by a sub-class to check whether any Thread or sub-process
|
|
127
|
-
that was started is still alive.
|
|
128
|
-
"""
|
|
129
|
-
return True
|
|
130
|
-
|
|
131
|
-
def get_control_server(self):
|
|
132
|
-
return self.__control_server
|
|
133
|
-
|
|
134
|
-
def get_status(self):
|
|
135
|
-
"""
|
|
136
|
-
Returns a dictionary with status information for the control server, enhanced by the
|
|
137
|
-
sub-class with device specific status information.
|
|
138
|
-
|
|
139
|
-
This method should be implemented/overridden by the sub-class. The sub-class specific
|
|
140
|
-
method should update the dictionary returned by this super-class method with device
|
|
141
|
-
specific status values.
|
|
142
|
-
|
|
143
|
-
The dict returned by this method includes the following keywords:
|
|
144
|
-
|
|
145
|
-
* timestamp (str): a string representation of the current datetime
|
|
146
|
-
* PID (int): the Process ID for the control server
|
|
147
|
-
* Up (float): the uptime of the control server [s]
|
|
148
|
-
* UUID (uuid1): a UUID for the control server
|
|
149
|
-
* RSS (int): the 'Resident Set Size', this is the non-swapped physical memory a process
|
|
150
|
-
has used [byte]
|
|
151
|
-
* USS (int): the 'Unique Set Size', this is the memory which is unique to a process [byte]
|
|
152
|
-
* CPU User (float): time spent in user mode [s]
|
|
153
|
-
* CPU System (float): time spent in kernel mode [s]
|
|
154
|
-
* CPU% (float): the process CPU utilization as a percentage [%]
|
|
155
|
-
|
|
156
|
-
Check the documentation for `psutil.Process` for more in-depth information about the
|
|
157
|
-
dict keys.
|
|
158
|
-
|
|
159
|
-
Returns:
|
|
160
|
-
a dictionary with status information.
|
|
161
|
-
"""
|
|
162
|
-
status = {
|
|
163
|
-
"timestamp": format_datetime(),
|
|
164
|
-
"delay": self.__control_server.delay,
|
|
165
|
-
}
|
|
166
|
-
status.update(self.__control_server.get_process_status())
|
|
167
|
-
return status
|
|
168
|
-
|
|
169
|
-
def get_housekeeping(self) -> dict:
|
|
170
|
-
"""Returns a dictionary with housekeeping information about the device."""
|
|
171
|
-
raise NotImplementedError(
|
|
172
|
-
f"The get_housekeeping() method shall be implemented for {self.__class__.__name__}."
|
|
173
|
-
)
|
|
174
|
-
|
|
175
|
-
def get_device(self):
|
|
176
|
-
"""Returns the device object for the device that is controlled by this protocol."""
|
|
177
|
-
raise NotImplementedError
|
|
178
|
-
|
|
179
|
-
def send(self, data):
|
|
180
|
-
"""
|
|
181
|
-
Send a message to the ControlServer. The message shall be fully populated
|
|
182
|
-
and is only serialized before sending over the ZeroMQ socket.
|
|
183
|
-
|
|
184
|
-
FIXME: We need to add error handling here, e.g. what if the send() fails? Do we need
|
|
185
|
-
to implement retries as with Proxy?
|
|
186
|
-
"""
|
|
187
|
-
pickle_string = pickle.dumps(data)
|
|
188
|
-
self.__socket.send(pickle_string)
|
|
189
|
-
|
|
190
|
-
def receive(self):
|
|
191
|
-
"""
|
|
192
|
-
Receive a serialized message from the ControlServer. The message will not
|
|
193
|
-
be decoded/de-serialized, but is returned as it was sent. Decoding shall
|
|
194
|
-
be handled by the calling method.
|
|
195
|
-
"""
|
|
196
|
-
pickle_string = self.__socket.recv()
|
|
197
|
-
return pickle.loads(pickle_string)
|
|
198
|
-
|
|
199
|
-
# FIXME:
|
|
200
|
-
# We might want to reconsider how commands are send over the ZeroMQ sockets.
|
|
201
|
-
# it can be very useful to use multipart messages here with the type and
|
|
202
|
-
# origin etc. to ease the if..else.. constructs.
|
|
203
|
-
|
|
204
|
-
@EXECUTION_TIME.time()
|
|
205
|
-
def execute(self):
|
|
206
|
-
cs = self.get_control_server()
|
|
207
|
-
data = self.receive()
|
|
208
|
-
cmd = None
|
|
209
|
-
if isinstance(data, CommandExecution):
|
|
210
|
-
cmd = data.get_cmd()
|
|
211
|
-
cmd_name = cmd.get_name()
|
|
212
|
-
args = data.get_args()
|
|
213
|
-
kwargs = data.get_kwargs()
|
|
214
|
-
elif isinstance(data, dict):
|
|
215
|
-
cmd_name = data.get("cmd")
|
|
216
|
-
args = data.get("args")
|
|
217
|
-
kwargs = data.get("kwargs")
|
|
218
|
-
elif isinstance(data, str):
|
|
219
|
-
cmd_name = data
|
|
220
|
-
else:
|
|
221
|
-
cmd_name = None
|
|
222
|
-
|
|
223
|
-
logger.log(0, f"cmd_name = {cmd_name}")
|
|
224
|
-
|
|
225
|
-
# Server availability request - Ping-Pong
|
|
226
|
-
|
|
227
|
-
if cmd_name == "Ping":
|
|
228
|
-
COMMAND_REQUESTS.labels(target="ping").inc()
|
|
229
|
-
self.send("Pong")
|
|
230
|
-
elif cmd_name == "send_commands":
|
|
231
|
-
logger.warning("send_commands was commanded for a DynamicCommandProtocol!")
|
|
232
|
-
elif cmd_name == "get_service_port":
|
|
233
|
-
self.send(self.__control_server.get_service_port())
|
|
234
|
-
elif cmd_name == "get_monitoring_port":
|
|
235
|
-
self.send(self.__control_server.get_monitoring_port())
|
|
236
|
-
elif cmd_name == "get_commanding_port":
|
|
237
|
-
self.send(self.__control_server.get_commanding_port())
|
|
238
|
-
elif cmd_name == "get_ip_address":
|
|
239
|
-
self.send(self.__control_server.get_ip_address())
|
|
240
|
-
elif cmd:
|
|
241
|
-
COMMAND_REQUESTS.labels(target="device").inc()
|
|
242
|
-
cmd.server_call(self, *args, **kwargs)
|
|
243
|
-
else:
|
|
244
|
-
COMMAND_REQUESTS.labels(target="invalid").inc()
|
|
245
|
-
logger.warning(f"Invalid command received: {cmd_name}")
|
|
246
|
-
self.send(Failure(f"Invalid command: {cmd_name}"))
|
|
247
|
-
|
|
248
|
-
def quit(self):
|
|
249
|
-
"""
|
|
250
|
-
This method can be overridden by a sub-class to cleanup and stop threads that it
|
|
251
|
-
started.
|
|
252
|
-
"""
|
|
253
|
-
|
|
254
|
-
logger.info("quit() method called on Protocol base class.")
|
|
255
|
-
|
|
256
|
-
|
|
257
|
-
|
|
258
|
-
class DynamicCommandProtocol(BaseCommandProtocol, metaclass=abc.ABCMeta):
|
|
259
|
-
def __init__(self, control_server: ControlServer):
|
|
260
|
-
super().__init__(control_server)
|
|
261
|
-
|
|
262
|
-
def handle_device_method(self, cmd: Command, *args, **kwargs):
|
|
263
|
-
"""
|
|
264
|
-
Call the device method with the given arguments.
|
|
265
|
-
|
|
266
|
-
Args:
|
|
267
|
-
cmd: the devices command class that knows which device command shall be called
|
|
268
|
-
args: the arguments that will be passed on to the device command
|
|
269
|
-
kwargs: the keyword arguments that will be passed on to the device command
|
|
270
|
-
"""
|
|
271
|
-
# The lookup table contains object (bound) methods, so we do not have to
|
|
272
|
-
# provide the 'self' argument anymore.
|
|
273
|
-
|
|
274
|
-
method_name = cmd.get_device_method_name()
|
|
275
|
-
method = get_method(self.get_device(), method_name)
|
|
276
|
-
|
|
277
|
-
# We treat the get_response function special as it needs to send the ``cmd`` string
|
|
278
|
-
# to the device we need to pass the processed cmd string into the method.
|
|
279
|
-
|
|
280
|
-
try:
|
|
281
|
-
if method_name == "get_response":
|
|
282
|
-
device_cmd_string = cmd.get_cmd_string(*args, *kwargs)
|
|
283
|
-
logger.log(5, f"Executing method {method.__name__}({device_cmd_string})")
|
|
284
|
-
response = method(device_cmd_string)
|
|
285
|
-
else:
|
|
286
|
-
logger.log(5, f"Executing method {method.__name__}({args}, {kwargs})")
|
|
287
|
-
response = method(*args, **kwargs)
|
|
288
|
-
except Exception as exc:
|
|
289
|
-
logger.exception(f"Executing {method_name} failed.")
|
|
290
|
-
# Pass the exception on to the client as a Failure message
|
|
291
|
-
response = Failure(f"Executing {method_name} failed: ", exc)
|
|
292
|
-
|
|
293
|
-
# Enable the following message only when debugging, because this log message can become
|
|
294
|
-
# very long for data storage commands.
|
|
295
|
-
# logger.debug(f"handle_device_method: {device_name}({args}, {kwargs}) -> {response!s}")
|
|
296
|
-
|
|
297
|
-
self.send(response)
|
|
298
|
-
|
|
299
|
-
|
|
300
|
-
# TODO (rik): The CommandProtocol shall also inherit from the BaseCommandProtocol
|
|
301
|
-
|
|
302
|
-
class CommandProtocol(DeviceConnectionObserver, metaclass=abc.ABCMeta):
|
|
303
|
-
"""
|
|
304
|
-
This class is the glue between the control servers and the hardware
|
|
305
|
-
controllers on one side, and between the control server and the connected
|
|
306
|
-
proxy classes on the other side.
|
|
307
|
-
|
|
308
|
-
The connection with the hardware controllers is when the ``execute()`` method
|
|
309
|
-
calls the ``server_call()`` method of the command class.
|
|
310
|
-
|
|
311
|
-
The connection with the proxy classes is when the ``client_call()`` method is added to the
|
|
312
|
-
interface of the Proxy subclass (by the ``_add_commands()`` method).
|
|
313
|
-
|
|
314
|
-
FIXME: Protocol is not used at the client side, i.e. the Proxy class.
|
|
315
|
-
"""
|
|
316
|
-
|
|
317
|
-
def __init__(self):
|
|
318
|
-
super().__init__()
|
|
319
|
-
self.__socket = None
|
|
320
|
-
self._commands = {} # variable is used by sub classes
|
|
321
|
-
self.control_server: ControlServer | None = None # variable set by the sub-class
|
|
322
|
-
self._method_lookup = {} # lookup table for device methods
|
|
323
|
-
|
|
324
|
-
def bind(self, socket):
|
|
325
|
-
"""Bind to a socket to listen for commands."""
|
|
326
|
-
self.__socket = socket
|
|
327
|
-
|
|
328
|
-
bind_address = self.get_bind_address()
|
|
329
|
-
logger.info(f"Binding to {bind_address}")
|
|
330
|
-
|
|
331
|
-
self.__socket.bind(bind_address)
|
|
332
|
-
|
|
333
|
-
# FIXME:
|
|
334
|
-
# We might want to reconsider how commands are send over the ZeroMQ sockets.
|
|
335
|
-
# it can be very useful to use multipart messages here with the type and
|
|
336
|
-
# origin etc. to ease the if..else.. constructs.
|
|
337
|
-
|
|
338
|
-
@EXECUTION_TIME.time()
|
|
339
|
-
def execute(self):
|
|
340
|
-
data = self.receive()
|
|
341
|
-
cmd = None
|
|
342
|
-
if isinstance(data, CommandExecution):
|
|
343
|
-
cmd = data.get_cmd()
|
|
344
|
-
cmd_name = cmd.get_name()
|
|
345
|
-
args = data.get_args()
|
|
346
|
-
kwargs = data.get_kwargs()
|
|
347
|
-
elif isinstance(data, dict):
|
|
348
|
-
cmd_name = data.get("cmd")
|
|
349
|
-
args = data.get("args")
|
|
350
|
-
kwargs = data.get("kwargs")
|
|
351
|
-
elif isinstance(data, str):
|
|
352
|
-
cmd_name = data
|
|
353
|
-
else:
|
|
354
|
-
cmd_name = None
|
|
355
|
-
|
|
356
|
-
logger.log(0, f"cmd_name = {cmd_name}")
|
|
357
|
-
|
|
358
|
-
# Server availability request - Ping-Pong
|
|
359
|
-
|
|
360
|
-
if cmd_name == "Ping":
|
|
361
|
-
COMMAND_REQUESTS.labels(target="ping").inc()
|
|
362
|
-
self.send("Pong")
|
|
363
|
-
elif cmd_name == "send_commands":
|
|
364
|
-
self.send_commands()
|
|
365
|
-
elif cmd_name == "get_service_port":
|
|
366
|
-
self.send(self.control_server.get_service_port())
|
|
367
|
-
elif cmd_name == "get_monitoring_port":
|
|
368
|
-
self.send(self.control_server.get_monitoring_port())
|
|
369
|
-
elif cmd_name == "get_commanding_port":
|
|
370
|
-
self.send(self.control_server.get_commanding_port())
|
|
371
|
-
elif cmd_name == "get_ip_address":
|
|
372
|
-
self.send(self.control_server.get_ip_address())
|
|
373
|
-
elif cmd_name == "get_storage_mnemonic":
|
|
374
|
-
self.send(self.control_server.get_storage_mnemonic())
|
|
375
|
-
elif cmd:
|
|
376
|
-
COMMAND_REQUESTS.labels(target="device").inc()
|
|
377
|
-
cmd.server_call(self, *args, **kwargs)
|
|
378
|
-
else:
|
|
379
|
-
COMMAND_REQUESTS.labels(target="invalid").inc()
|
|
380
|
-
logger.warning(f"Invalid command received: {cmd_name}")
|
|
381
|
-
self.send(Failure(f"Invalid command: {cmd_name}"))
|
|
382
|
-
|
|
383
|
-
def quit(self):
|
|
384
|
-
"""
|
|
385
|
-
This method can be overridden by a sub-class to cleanup and stop threads that it
|
|
386
|
-
started.
|
|
387
|
-
"""
|
|
388
|
-
|
|
389
|
-
logger.info("quit() method called on Protocol base class.")
|
|
390
|
-
|
|
391
|
-
def is_alive(self) -> bool:
|
|
392
|
-
"""
|
|
393
|
-
This method can be overridden by a sub-class to check whether any Thread or sub-process
|
|
394
|
-
that was started is still alive.
|
|
395
|
-
"""
|
|
396
|
-
return True
|
|
397
|
-
|
|
398
|
-
@abc.abstractmethod
|
|
399
|
-
def get_bind_address(self):
|
|
400
|
-
"""
|
|
401
|
-
Returns a string with the bind address, the endpoint, for accepting connections
|
|
402
|
-
and bind a socket to.
|
|
403
|
-
|
|
404
|
-
This method should be implemented by the sub-class since it contains the protocol
|
|
405
|
-
and port number that are specific for the sub-class.
|
|
406
|
-
|
|
407
|
-
Returns:
|
|
408
|
-
a string with the protocol and port to bind a socket to.
|
|
409
|
-
"""
|
|
410
|
-
pass
|
|
411
|
-
|
|
412
|
-
@abc.abstractmethod
|
|
413
|
-
def get_status(self):
|
|
414
|
-
"""
|
|
415
|
-
Returns a dictionary with status information for the control server, enhanced by the
|
|
416
|
-
sub-class with device specific status information.
|
|
417
|
-
|
|
418
|
-
This method should be implemented/overridden by the sub-class. The sub-class specific
|
|
419
|
-
method should update the dictionary returned by this super-class method with device
|
|
420
|
-
specific status values.
|
|
421
|
-
|
|
422
|
-
The dict returned by this method includes the following keywords:
|
|
423
|
-
|
|
424
|
-
* timestamp (str): a string representation of the current datetime
|
|
425
|
-
* PID (int): the Process ID for the control server
|
|
426
|
-
* Up (float): the uptime of the control server [s]
|
|
427
|
-
* UUID (uuid1): a UUID for the control server
|
|
428
|
-
* RSS (int): the 'Resident Set Size', this is the non-swapped physical memory a process
|
|
429
|
-
has used [byte]
|
|
430
|
-
* USS (int): the 'Unique Set Size', this is the memory which is unique to a process [byte]
|
|
431
|
-
* CPU User (float): time spent in user mode [s]
|
|
432
|
-
* CPU System (float): time spent in kernel mode [s]
|
|
433
|
-
* CPU% (float): the process CPU utilization as a percentage [%]
|
|
434
|
-
|
|
435
|
-
Check the documentation for `psutil.Process` for more in-depth information about the
|
|
436
|
-
dict keys.
|
|
437
|
-
|
|
438
|
-
Returns:
|
|
439
|
-
a dictionary with status information.
|
|
440
|
-
"""
|
|
441
|
-
status = {
|
|
442
|
-
"timestamp": format_datetime(),
|
|
443
|
-
"delay": self.control_server.delay,
|
|
444
|
-
}
|
|
445
|
-
status.update(self.control_server.get_process_status())
|
|
446
|
-
return status
|
|
447
|
-
|
|
448
|
-
def get_housekeeping(self) -> dict:
|
|
449
|
-
"""Returns a dictionary with housekeeping information about the device."""
|
|
450
|
-
raise NotImplementedError(
|
|
451
|
-
f"The get_housekeeping() method shall be implemented for {self.__class__.__name__}."
|
|
452
|
-
)
|
|
453
|
-
|
|
454
|
-
def send(self, data):
|
|
455
|
-
"""
|
|
456
|
-
Send a message to the ControlServer. The message shall be fully populated
|
|
457
|
-
and is only serialized before sending over the ZeroMQ socket.
|
|
458
|
-
|
|
459
|
-
FIXME: We need to add error handling here, e.g. what if the send() fails? Do we need
|
|
460
|
-
to implement retries as with Proxy?
|
|
461
|
-
"""
|
|
462
|
-
pickle_string = pickle.dumps(data)
|
|
463
|
-
self.__socket.send(pickle_string)
|
|
464
|
-
|
|
465
|
-
def receive(self):
|
|
466
|
-
"""
|
|
467
|
-
Receive a serialized message from the ControlServer. The message will not
|
|
468
|
-
be decoded/de-serialized, but is returned as it was sent. Decoding shall
|
|
469
|
-
be handled by the calling method.
|
|
470
|
-
"""
|
|
471
|
-
pickle_string = self.__socket.recv()
|
|
472
|
-
data = pickle.loads(pickle_string)
|
|
473
|
-
return data
|
|
474
|
-
|
|
475
|
-
def send_commands(self):
|
|
476
|
-
"""
|
|
477
|
-
Send the command definitions that were loaded for the specific device.
|
|
478
|
-
"""
|
|
479
|
-
self.send(self._commands)
|
|
480
|
-
|
|
481
|
-
def load_commands(self, command_settings, command_class, device_class):
|
|
482
|
-
"""
|
|
483
|
-
Loads the command definitions from the given ``command_settings`` and builds an internal
|
|
484
|
-
dictionary containing the command names as keys and the corresponding ``Command`` class
|
|
485
|
-
objects as values.
|
|
486
|
-
|
|
487
|
-
The ``command_settings`` is usually loaded from a YAML configuration file containing the
|
|
488
|
-
command definitions for the device.
|
|
489
|
-
|
|
490
|
-
Args:
|
|
491
|
-
command_settings: a dictionary containing the command definitions for this device
|
|
492
|
-
command_class: the type of command to create, a subclass of Command
|
|
493
|
-
device_class: the type of the base device class from which the methods are loaded
|
|
494
|
-
"""
|
|
495
|
-
for name in command_settings:
|
|
496
|
-
command_settings_name = command_settings[name]
|
|
497
|
-
if "cmd" in command_settings_name:
|
|
498
|
-
cmd = command_settings_name["cmd"]
|
|
499
|
-
else:
|
|
500
|
-
cmd = ""
|
|
501
|
-
|
|
502
|
-
if "description" in command_settings_name:
|
|
503
|
-
description = command_settings_name["description"]
|
|
504
|
-
else:
|
|
505
|
-
description = None
|
|
506
|
-
|
|
507
|
-
# The response field is the name of a function from the CommandProtocol class or a
|
|
508
|
-
# sub-class. This function shall send a response back to the client (Proxy). That's
|
|
509
|
-
# why this field is called response.
|
|
510
|
-
# By convention we like that this method name would start with `handle_` so the we
|
|
511
|
-
# can make a distinction between response commands and normal methods in Protocol.
|
|
512
|
-
# Remember that response methods should send a reply back to the client (which will
|
|
513
|
-
# be waiting for it..).
|
|
514
|
-
# If no response field is given, then the `handle_device_method` will be called.
|
|
515
|
-
|
|
516
|
-
if "response" in command_settings_name:
|
|
517
|
-
response_method = get_function(self.__class__, command_settings_name["response"])
|
|
518
|
-
else:
|
|
519
|
-
response_method = get_function(self.__class__, "handle_device_method")
|
|
520
|
-
|
|
521
|
-
# The device_method field is used in the `handle_device_method` to call the method on
|
|
522
|
-
# the device class. That is the class that implements the DeviceInterface and is
|
|
523
|
-
# usually called a Controller or a Simulator.
|
|
524
|
-
#
|
|
525
|
-
# If no device_name field is given, the name from the command_settings is used.
|
|
526
|
-
|
|
527
|
-
if "device_method" in command_settings_name:
|
|
528
|
-
device_method_name = command_settings_name["device_method"]
|
|
529
|
-
else:
|
|
530
|
-
device_method_name = name
|
|
531
|
-
|
|
532
|
-
# check if the device_method exists in the device base class
|
|
533
|
-
|
|
534
|
-
if device_method_name == "None":
|
|
535
|
-
device_method = None
|
|
536
|
-
else:
|
|
537
|
-
device_method = get_function(device_class, device_method_name)
|
|
538
|
-
|
|
539
|
-
logger.log(
|
|
540
|
-
0,
|
|
541
|
-
f"Creating {command_class.__module__}.{command_class.__name__}(name='{name}', "
|
|
542
|
-
f"cmd='{cmd}', "
|
|
543
|
-
f"response={response_method}, device_method={device_method})",
|
|
544
|
-
)
|
|
545
|
-
logger.debug(f"Creating {command_class.__name__} command with {name=}, {cmd=}, {device_method=}")
|
|
546
|
-
|
|
547
|
-
self._commands[name] = command_class(
|
|
548
|
-
name=name,
|
|
549
|
-
cmd=cmd,
|
|
550
|
-
response=response_method,
|
|
551
|
-
description=description,
|
|
552
|
-
device_method=device_method,
|
|
553
|
-
)
|
|
554
|
-
|
|
555
|
-
def build_device_method_lookup_table(self, device_obj):
|
|
556
|
-
"""
|
|
557
|
-
Fill the lookup table with device command methods that are bound to the device object.
|
|
558
|
-
|
|
559
|
-
Args:
|
|
560
|
-
device_obj: instance of a device command class
|
|
561
|
-
"""
|
|
562
|
-
for cmd in self._commands.values():
|
|
563
|
-
method_name = cmd.get_device_method_name()
|
|
564
|
-
method = get_method(device_obj, method_name)
|
|
565
|
-
if method is not None:
|
|
566
|
-
self._method_lookup[method_name] = method
|
|
567
|
-
|
|
568
|
-
def handle_device_method(self, cmd: Command, *args, **kwargs):
|
|
569
|
-
"""
|
|
570
|
-
Call the device method with the given arguments.
|
|
571
|
-
|
|
572
|
-
Args:
|
|
573
|
-
cmd: the devices command class that knows which device command shall be called
|
|
574
|
-
args: the arguments that will be passed on to the device command
|
|
575
|
-
kwargs: the keyword arguments that will be passed on to the device command
|
|
576
|
-
"""
|
|
577
|
-
# The lookup table contains object (bound) methods, so we do not have to
|
|
578
|
-
# provide the 'self' argument anymore.
|
|
579
|
-
|
|
580
|
-
device_name = cmd.get_device_method_name()
|
|
581
|
-
method = self._method_lookup[device_name]
|
|
582
|
-
|
|
583
|
-
# We treat the get_response function special as it needs to send the ``cmd`` string
|
|
584
|
-
# to the device we need to pass the processed cmd string into the method.
|
|
585
|
-
|
|
586
|
-
try:
|
|
587
|
-
if device_name == "get_response":
|
|
588
|
-
device_cmd_string = cmd.get_cmd_string(*args, *kwargs)
|
|
589
|
-
logger.log(5, f"Executing method {method.__name__}({device_cmd_string})")
|
|
590
|
-
response = method(device_cmd_string)
|
|
591
|
-
else:
|
|
592
|
-
logger.log(5, f"Executing method {method.__name__}({args}, {kwargs})")
|
|
593
|
-
response = method(*args, **kwargs)
|
|
594
|
-
except Exception as exc:
|
|
595
|
-
logger.exception(f"Executing {device_name} failed.")
|
|
596
|
-
# Pass the exception on to the client as a Failure message
|
|
597
|
-
response = Failure(f"Executing {device_name} failed: ", exc)
|
|
598
|
-
|
|
599
|
-
# Enable the following message only when debugging, because this log message can become
|
|
600
|
-
# very long for data storage commands.
|
|
601
|
-
# logger.debug(f"handle_device_method: {device_name}({args}, {kwargs}) -> {response!s}")
|
|
602
|
-
|
|
603
|
-
self.send(response)
|