cgse 2024.7.0__py3-none-any.whl → 2025.0.2__py3-none-any.whl
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- README.md +27 -0
- bump.py +85 -0
- cgse-2025.0.2.dist-info/METADATA +38 -0
- cgse-2025.0.2.dist-info/RECORD +5 -0
- {cgse-2024.7.0.dist-info → cgse-2025.0.2.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/proxy.py
DELETED
|
@@ -1,531 +0,0 @@
|
|
|
1
|
-
"""
|
|
2
|
-
The Proxy module provides the base class for the Proxy objects for each device
|
|
3
|
-
controller.
|
|
4
|
-
|
|
5
|
-
The module also provides the connection state interface and classes for
|
|
6
|
-
maintaining the state of the Proxy connection to the control server.
|
|
7
|
-
"""
|
|
8
|
-
import logging
|
|
9
|
-
import pickle
|
|
10
|
-
import types
|
|
11
|
-
from types import MethodType
|
|
12
|
-
|
|
13
|
-
import zmq
|
|
14
|
-
|
|
15
|
-
from egse.control import Failure
|
|
16
|
-
from egse.decorators import dynamic_interface
|
|
17
|
-
from egse.mixin import DynamicClientCommandMixin
|
|
18
|
-
from egse.system import AttributeDict
|
|
19
|
-
from egse.zmq_ser import split_address
|
|
20
|
-
|
|
21
|
-
|
|
22
|
-
def set_docstring(func, cmd):
|
|
23
|
-
"""Decorator to set the docstring of the command on the dynamic method / function."""
|
|
24
|
-
|
|
25
|
-
def wrap_func(*args, **kwargs):
|
|
26
|
-
return func(*args, **kwargs)
|
|
27
|
-
|
|
28
|
-
wrap_func.__doc__ = cmd.__doc__
|
|
29
|
-
wrap_func.__name__ = f"{cmd.get_name()}"
|
|
30
|
-
return wrap_func
|
|
31
|
-
|
|
32
|
-
|
|
33
|
-
REQUEST_TIMEOUT = 6250
|
|
34
|
-
REQUEST_RETRIES = 0
|
|
35
|
-
|
|
36
|
-
|
|
37
|
-
class ControlServerConnectionInterface:
|
|
38
|
-
"""This interface defines the connection commands for control servers.
|
|
39
|
-
|
|
40
|
-
This interface shall be implemented by the Proxy class and guarantees that connection commands
|
|
41
|
-
do not interfere with the commands defined in the `DeviceConnectionInterface` (which will be
|
|
42
|
-
loaded from the control server).
|
|
43
|
-
"""
|
|
44
|
-
|
|
45
|
-
def __enter__(self):
|
|
46
|
-
self.connect_cs()
|
|
47
|
-
return self
|
|
48
|
-
|
|
49
|
-
def __exit__(self, exc_type, exc_val, exc_tb):
|
|
50
|
-
self.disconnect_cs()
|
|
51
|
-
|
|
52
|
-
@dynamic_interface
|
|
53
|
-
def connect_cs(self):
|
|
54
|
-
"""Connect to the control server.
|
|
55
|
-
|
|
56
|
-
Raises:
|
|
57
|
-
ConnectionError: when the connection can not be established.
|
|
58
|
-
"""
|
|
59
|
-
raise NotImplementedError
|
|
60
|
-
|
|
61
|
-
@dynamic_interface
|
|
62
|
-
def reconnect_cs(self):
|
|
63
|
-
"""Reconnect the control server after it has been disconnected.
|
|
64
|
-
|
|
65
|
-
Raises:
|
|
66
|
-
ConnectionError: when the connection can not be established.
|
|
67
|
-
"""
|
|
68
|
-
raise NotImplementedError
|
|
69
|
-
|
|
70
|
-
@dynamic_interface
|
|
71
|
-
def disconnect_cs(self):
|
|
72
|
-
"""Disconnect from the control server.
|
|
73
|
-
|
|
74
|
-
Raises:
|
|
75
|
-
ConnectionError: when the connection can not be closed.
|
|
76
|
-
"""
|
|
77
|
-
raise NotImplementedError
|
|
78
|
-
|
|
79
|
-
@dynamic_interface
|
|
80
|
-
def reset_cs_connection(self):
|
|
81
|
-
"""Resets the connection to the control server."""
|
|
82
|
-
raise NotImplementedError
|
|
83
|
-
|
|
84
|
-
@dynamic_interface
|
|
85
|
-
def is_cs_connected(self) -> bool:
|
|
86
|
-
"""Check if the control server is connected.
|
|
87
|
-
|
|
88
|
-
Returns:
|
|
89
|
-
True if the device is connected and responds to a command, False otherwise.
|
|
90
|
-
"""
|
|
91
|
-
raise NotImplementedError
|
|
92
|
-
|
|
93
|
-
|
|
94
|
-
class BaseProxy(ControlServerConnectionInterface):
|
|
95
|
-
def __init__(self, endpoint, timeout: int = REQUEST_TIMEOUT):
|
|
96
|
-
"""
|
|
97
|
-
The `timeout` argument specifies the number of milliseconds to wait for a reply from the
|
|
98
|
-
control server.
|
|
99
|
-
"""
|
|
100
|
-
|
|
101
|
-
self._logger = logging.getLogger(self.__class__.__name__)
|
|
102
|
-
|
|
103
|
-
self._ctx = zmq.Context.instance()
|
|
104
|
-
self._poller = zmq.Poller()
|
|
105
|
-
self._socket = None
|
|
106
|
-
self._endpoint = endpoint
|
|
107
|
-
self._timeout = timeout
|
|
108
|
-
|
|
109
|
-
self.connect_cs()
|
|
110
|
-
|
|
111
|
-
def __enter__(self):
|
|
112
|
-
if not self.ping():
|
|
113
|
-
raise ConnectionError(f"Proxy is not connected to endpoint ({self._endpoint}) when entering the context.")
|
|
114
|
-
|
|
115
|
-
return self
|
|
116
|
-
|
|
117
|
-
def __exit__(self, exc_type, exc_val, exc_tb):
|
|
118
|
-
if not self._socket.closed:
|
|
119
|
-
self.disconnect_cs()
|
|
120
|
-
|
|
121
|
-
def connect_cs(self):
|
|
122
|
-
self._logger.log(0, f"Trying to connect {self.__class__.__name__} to {self._endpoint}")
|
|
123
|
-
|
|
124
|
-
self._socket = self._ctx.socket(zmq.REQ)
|
|
125
|
-
self._socket.connect(self._endpoint)
|
|
126
|
-
self._poller.register(self._socket, zmq.POLLIN)
|
|
127
|
-
|
|
128
|
-
def disconnect_cs(self):
|
|
129
|
-
self._socket.setsockopt(zmq.LINGER, 0)
|
|
130
|
-
self._socket.close()
|
|
131
|
-
self._poller.unregister(self._socket)
|
|
132
|
-
|
|
133
|
-
def reconnect_cs(self):
|
|
134
|
-
self._logger.log(20, f"Trying to reconnect {self.__class__.__name__} to {self._endpoint}")
|
|
135
|
-
|
|
136
|
-
if not self._socket.closed:
|
|
137
|
-
self._socket.close(linger=0)
|
|
138
|
-
|
|
139
|
-
self._socket = self._ctx.socket(zmq.REQ)
|
|
140
|
-
self._socket.connect(self._endpoint)
|
|
141
|
-
self._poller.register(self._socket, zmq.POLLIN)
|
|
142
|
-
|
|
143
|
-
def reset_cs_connection(self):
|
|
144
|
-
self._logger.log(
|
|
145
|
-
10, f"Trying to reset the connection from {self.__class__.__name__} to {self._endpoint}"
|
|
146
|
-
)
|
|
147
|
-
|
|
148
|
-
self.disconnect_cs()
|
|
149
|
-
self.connect_cs()
|
|
150
|
-
|
|
151
|
-
def is_cs_connected(self) -> bool:
|
|
152
|
-
return self.ping()
|
|
153
|
-
|
|
154
|
-
def send(self, data, retries: int = REQUEST_RETRIES, timeout: int = None):
|
|
155
|
-
"""
|
|
156
|
-
Sends a command to the control server and waits for a response.
|
|
157
|
-
|
|
158
|
-
When not connected to the control server or when a timeout occurs, the
|
|
159
|
-
``send()`` command retries a number of times to send the command.
|
|
160
|
-
|
|
161
|
-
The number of retries is hardcoded and currently set to '2', the request
|
|
162
|
-
timeout is set to 2.5 seconds.
|
|
163
|
-
|
|
164
|
-
The command data will be pickled before sending. Make sure the ``data``
|
|
165
|
-
argument can be dumped by pickle.
|
|
166
|
-
|
|
167
|
-
Args:
|
|
168
|
-
data (str): the command that is sent to the control server, usually a
|
|
169
|
-
string, but that is not enforced.
|
|
170
|
-
timeout (int): the time to wait for a reply [in milliseconds]
|
|
171
|
-
retries (int): the number of time we should retry to send the message
|
|
172
|
-
|
|
173
|
-
Returns:
|
|
174
|
-
response: the response from the control server or ``None`` when there was
|
|
175
|
-
a problem or a timeout.
|
|
176
|
-
"""
|
|
177
|
-
timeout = timeout or self._timeout
|
|
178
|
-
|
|
179
|
-
pickle_string = pickle.dumps(data)
|
|
180
|
-
|
|
181
|
-
retries_left = retries
|
|
182
|
-
|
|
183
|
-
# When we enter this method, we assume the Proxy has been connected. It
|
|
184
|
-
# might be the server is not responding, but that is handled by the
|
|
185
|
-
# algorithm below where we have a number of retries to receive the response
|
|
186
|
-
# of the sent command. Remember that we are using ZeroMQ where the connect
|
|
187
|
-
# method returns gracefully even when no server is available.
|
|
188
|
-
|
|
189
|
-
if self._socket.closed:
|
|
190
|
-
self.reconnect_cs()
|
|
191
|
-
|
|
192
|
-
self._logger.log(0, f"Sending '{data}'")
|
|
193
|
-
self._socket.send(pickle_string)
|
|
194
|
-
|
|
195
|
-
while True:
|
|
196
|
-
socks = dict(self._poller.poll(timeout))
|
|
197
|
-
|
|
198
|
-
if self._socket in socks and socks[self._socket] == zmq.POLLIN:
|
|
199
|
-
pickle_string = self._socket.recv()
|
|
200
|
-
if not pickle_string:
|
|
201
|
-
break
|
|
202
|
-
response = pickle.loads(pickle_string)
|
|
203
|
-
self._logger.log(0, f"Receiving response: {response}")
|
|
204
|
-
return response
|
|
205
|
-
else:
|
|
206
|
-
# timeout - server unavailable
|
|
207
|
-
|
|
208
|
-
# We should disconnect here because socket is possibly confused.
|
|
209
|
-
# Close the socket and remove from the poller.
|
|
210
|
-
|
|
211
|
-
self.disconnect_cs()
|
|
212
|
-
|
|
213
|
-
if retries_left == 0:
|
|
214
|
-
self._logger.critical("Control Server seems to be off-line, abandoning")
|
|
215
|
-
return None
|
|
216
|
-
retries_left -= 1
|
|
217
|
-
|
|
218
|
-
self._logger.log(logging.CRITICAL, f"Reconnecting {self.__class__.__name__}, {retries_left=}")
|
|
219
|
-
|
|
220
|
-
self.reconnect_cs()
|
|
221
|
-
|
|
222
|
-
# Now try to send the request again
|
|
223
|
-
|
|
224
|
-
self._socket.send(pickle_string)
|
|
225
|
-
|
|
226
|
-
def ping(self):
|
|
227
|
-
return_code = self.send("Ping", retries=0, timeout=1000)
|
|
228
|
-
self._logger.log(0, f"Check if control server is available: Ping - {return_code}")
|
|
229
|
-
return return_code == "Pong"
|
|
230
|
-
|
|
231
|
-
def get_endpoint(self):
|
|
232
|
-
""" Returns the endpoint."""
|
|
233
|
-
return self._endpoint
|
|
234
|
-
|
|
235
|
-
def get_monitoring_port(self) -> int:
|
|
236
|
-
""" Returns the monitoring port. """
|
|
237
|
-
return self.send("get_monitoring_port")
|
|
238
|
-
|
|
239
|
-
def get_commanding_port(self) -> int:
|
|
240
|
-
""" Returns the commanding port."""
|
|
241
|
-
return self.send("get_commanding_port")
|
|
242
|
-
|
|
243
|
-
def get_service_port(self) -> int:
|
|
244
|
-
""" Returns the service port. """
|
|
245
|
-
return self.send("get_service_port")
|
|
246
|
-
|
|
247
|
-
def get_ip_address(self) -> int:
|
|
248
|
-
""" Returns the hostname of the control server."""
|
|
249
|
-
return self.send("get_ip_address")
|
|
250
|
-
|
|
251
|
-
def get_service_proxy(self):
|
|
252
|
-
"""Return a ServiceProxy for the control server of this proxy object."""
|
|
253
|
-
from egse.services import ServiceProxy # prevent circular import problem
|
|
254
|
-
|
|
255
|
-
transport, address, _ = split_address(self._endpoint)
|
|
256
|
-
|
|
257
|
-
port = self.send("get_service_port")
|
|
258
|
-
|
|
259
|
-
return ServiceProxy(
|
|
260
|
-
AttributeDict({"PROTOCOL": transport, "HOSTNAME": address, "SERVICE_PORT": port})
|
|
261
|
-
)
|
|
262
|
-
|
|
263
|
-
|
|
264
|
-
class DynamicProxy(BaseProxy, DynamicClientCommandMixin):
|
|
265
|
-
def __init__(self, *args, **kwargs):
|
|
266
|
-
super().__init__(*args, **kwargs)
|
|
267
|
-
|
|
268
|
-
|
|
269
|
-
# TODO (rik): remove all methods from Proxy that are also define in the BaseProxy
|
|
270
|
-
|
|
271
|
-
class Proxy(BaseProxy, ControlServerConnectionInterface):
|
|
272
|
-
"""
|
|
273
|
-
A Proxy object will forward CommandExecutions to the connected control server
|
|
274
|
-
and wait for a response. When the Proxy can not connect to its control server
|
|
275
|
-
during initialization, a ConnectionError will be raised.
|
|
276
|
-
"""
|
|
277
|
-
|
|
278
|
-
def __init__(self, endpoint, timeout: int = REQUEST_TIMEOUT):
|
|
279
|
-
"""
|
|
280
|
-
During initialization, the Proxy will connect to the control server and send a
|
|
281
|
-
handshaking `Ping` command. When that succeeds the Proxy will request and load the
|
|
282
|
-
available commands from the control server. When the connection with the control server
|
|
283
|
-
fails, no commands are loaded and the Proxy is left in a 'disconnected' state. The caller
|
|
284
|
-
can fix the problem with the control server and call `connect_cs()`, followed by a call to
|
|
285
|
-
`load_commands()`.
|
|
286
|
-
|
|
287
|
-
The `timeout` argument specifies the number of milliseconds
|
|
288
|
-
"""
|
|
289
|
-
|
|
290
|
-
super().__init__(endpoint, timeout)
|
|
291
|
-
|
|
292
|
-
self._commands = {}
|
|
293
|
-
|
|
294
|
-
if self.ping():
|
|
295
|
-
self.load_commands()
|
|
296
|
-
else:
|
|
297
|
-
self._logger.warning(
|
|
298
|
-
f"{self.__class__.__name__} could not connect to its control server at {endpoint}. "
|
|
299
|
-
f"No commands have been loaded."
|
|
300
|
-
)
|
|
301
|
-
|
|
302
|
-
def __enter__(self):
|
|
303
|
-
if not self.ping():
|
|
304
|
-
raise ConnectionError("Proxy is not connected when entering the context.")
|
|
305
|
-
|
|
306
|
-
# The following check is here because a CS might have come alive between the __init__
|
|
307
|
-
# and __enter__ calls, and while the ping() will reconnect, the Proxy will have no
|
|
308
|
-
# commands loaded.
|
|
309
|
-
|
|
310
|
-
if not self.has_commands():
|
|
311
|
-
self.load_commands()
|
|
312
|
-
|
|
313
|
-
return self
|
|
314
|
-
|
|
315
|
-
def __exit__(self, exc_type, exc_val, exc_tb):
|
|
316
|
-
if not self._socket.closed:
|
|
317
|
-
self.disconnect_cs()
|
|
318
|
-
|
|
319
|
-
def connect_cs(self):
|
|
320
|
-
self._logger.log(0, f"Trying to connect {self.__class__.__name__} to {self._endpoint}")
|
|
321
|
-
|
|
322
|
-
self._socket = self._ctx.socket(zmq.REQ)
|
|
323
|
-
self._socket.connect(self._endpoint)
|
|
324
|
-
self._poller.register(self._socket, zmq.POLLIN)
|
|
325
|
-
|
|
326
|
-
def disconnect_cs(self):
|
|
327
|
-
self._socket.setsockopt(zmq.LINGER, 0)
|
|
328
|
-
self._socket.close()
|
|
329
|
-
self._poller.unregister(self._socket)
|
|
330
|
-
|
|
331
|
-
def reconnect_cs(self):
|
|
332
|
-
self._logger.log(20, f"Trying to reconnect {self.__class__.__name__} to {self._endpoint}")
|
|
333
|
-
|
|
334
|
-
if not self._socket.closed:
|
|
335
|
-
self._socket.close(linger=0)
|
|
336
|
-
|
|
337
|
-
self._socket = self._ctx.socket(zmq.REQ)
|
|
338
|
-
self._socket.connect(self._endpoint)
|
|
339
|
-
self._poller.register(self._socket, zmq.POLLIN)
|
|
340
|
-
|
|
341
|
-
def reset_cs_connection(self):
|
|
342
|
-
self._logger.log(
|
|
343
|
-
10, f"Trying to reset the connection from {self.__class__.__name__} to {self._endpoint}"
|
|
344
|
-
)
|
|
345
|
-
|
|
346
|
-
self.disconnect_cs()
|
|
347
|
-
self.connect_cs()
|
|
348
|
-
|
|
349
|
-
def is_cs_connected(self) -> bool:
|
|
350
|
-
return self.ping()
|
|
351
|
-
|
|
352
|
-
def send(self, data, retries: int = REQUEST_RETRIES, timeout: int = None):
|
|
353
|
-
"""
|
|
354
|
-
Sends a command to the control server and waits for a response.
|
|
355
|
-
|
|
356
|
-
When not connected to the control server or when a timeout occurs, the
|
|
357
|
-
``send()`` command retries a number of times to send the command.
|
|
358
|
-
|
|
359
|
-
The number of retries is hardcoded and currently set to '2', the request
|
|
360
|
-
timeout is set to 2.5 seconds.
|
|
361
|
-
|
|
362
|
-
The command data will be pickled before sending. Make sure the ``data``
|
|
363
|
-
argument can be dumped by pickle.
|
|
364
|
-
|
|
365
|
-
Args:
|
|
366
|
-
data (str): the command that is sent to the control server, usually a
|
|
367
|
-
string, but that is not enforced.
|
|
368
|
-
timeout (int): the time to wait for a reply [in milliseconds]
|
|
369
|
-
retries (int): the number of time we should retry to send the message
|
|
370
|
-
|
|
371
|
-
Returns:
|
|
372
|
-
response: the response from the control server or ``None`` when there was
|
|
373
|
-
a problem or a timeout.
|
|
374
|
-
"""
|
|
375
|
-
timeout = timeout or self._timeout
|
|
376
|
-
|
|
377
|
-
pickle_string = pickle.dumps(data)
|
|
378
|
-
|
|
379
|
-
retries_left = retries
|
|
380
|
-
|
|
381
|
-
# When we enter this method, we assume the Proxy has been connected. It
|
|
382
|
-
# might be the server is not responding, but that is handled by the
|
|
383
|
-
# algorithm below where we have a number of retries to receive the response
|
|
384
|
-
# of the sent command. Remember that we are using ZeroMQ where the connect
|
|
385
|
-
# method returns gracefully even when no server is available.
|
|
386
|
-
|
|
387
|
-
if self._socket.closed:
|
|
388
|
-
self.reconnect_cs()
|
|
389
|
-
|
|
390
|
-
self._logger.log(0, f"Sending '{data}'")
|
|
391
|
-
self._socket.send(pickle_string)
|
|
392
|
-
|
|
393
|
-
while True:
|
|
394
|
-
socks = dict(self._poller.poll(timeout))
|
|
395
|
-
|
|
396
|
-
if self._socket in socks and socks[self._socket] == zmq.POLLIN:
|
|
397
|
-
pickle_string = self._socket.recv()
|
|
398
|
-
if not pickle_string:
|
|
399
|
-
break
|
|
400
|
-
response = pickle.loads(pickle_string)
|
|
401
|
-
self._logger.log(0, f"Receiving response: {response}")
|
|
402
|
-
return response
|
|
403
|
-
else:
|
|
404
|
-
# timeout - server unavailable
|
|
405
|
-
|
|
406
|
-
# We should disconnect here because socket is possibly confused.
|
|
407
|
-
# Close the socket and remove from the poller.
|
|
408
|
-
|
|
409
|
-
self.disconnect_cs()
|
|
410
|
-
|
|
411
|
-
if retries_left == 0:
|
|
412
|
-
self._logger.critical(f"Control Server seems to be off-line, abandoning ({data})")
|
|
413
|
-
from egse.control import Failure
|
|
414
|
-
return Failure(f"Control Server seems to be off-line, abandoning ({data})")
|
|
415
|
-
retries_left -= 1
|
|
416
|
-
|
|
417
|
-
self._logger.log(logging.CRITICAL, f"Reconnecting {self.__class__.__name__}, {retries_left=}")
|
|
418
|
-
|
|
419
|
-
self.reconnect_cs()
|
|
420
|
-
|
|
421
|
-
# Now try to send the request again
|
|
422
|
-
|
|
423
|
-
self._socket.send(pickle_string)
|
|
424
|
-
|
|
425
|
-
def _request_commands(self):
|
|
426
|
-
response = self.send("send_commands")
|
|
427
|
-
if isinstance(response, Failure):
|
|
428
|
-
raise response
|
|
429
|
-
self._commands = response
|
|
430
|
-
|
|
431
|
-
def _add_commands(self):
|
|
432
|
-
for key in self._commands:
|
|
433
|
-
if hasattr(self, key):
|
|
434
|
-
attribute = getattr(self, key)
|
|
435
|
-
if isinstance(attribute, types.MethodType) and not hasattr(
|
|
436
|
-
attribute, "__dynamic_interface"
|
|
437
|
-
):
|
|
438
|
-
self._logger.warning(
|
|
439
|
-
f"{self.__class__.__name__} already has an attribute '{key}', "
|
|
440
|
-
f"not overwriting."
|
|
441
|
-
)
|
|
442
|
-
continue
|
|
443
|
-
command = self._commands[key]
|
|
444
|
-
new_method = MethodType(command.client_call, self)
|
|
445
|
-
new_method = set_docstring(new_method, command)
|
|
446
|
-
setattr(self, key, new_method)
|
|
447
|
-
|
|
448
|
-
def get_service_proxy(self):
|
|
449
|
-
"""Return a ServiceProxy for the control server of this proxy object."""
|
|
450
|
-
from egse.services import ServiceProxy # prevent circular import problem
|
|
451
|
-
|
|
452
|
-
transport, address, _ = split_address(self._endpoint)
|
|
453
|
-
|
|
454
|
-
port = self.send("get_service_port")
|
|
455
|
-
|
|
456
|
-
return ServiceProxy(
|
|
457
|
-
AttributeDict({"PROTOCOL": transport, "HOSTNAME": address, "SERVICE_PORT": port})
|
|
458
|
-
)
|
|
459
|
-
|
|
460
|
-
def load_commands(self):
|
|
461
|
-
"""
|
|
462
|
-
Requests all available commands from the control server and adds them to
|
|
463
|
-
the Proxy public interface, i.e. each command will become a method for
|
|
464
|
-
this Proxy.
|
|
465
|
-
|
|
466
|
-
A warning will be issued when a command will overwrite an existing method
|
|
467
|
-
of the Proxy class. The original method will not be overwritten and the
|
|
468
|
-
behavior of the Proxy command will not be what is expected.
|
|
469
|
-
"""
|
|
470
|
-
# bind the client_call method from each Command to this Proxy object
|
|
471
|
-
if self.is_cs_connected():
|
|
472
|
-
try:
|
|
473
|
-
self._request_commands()
|
|
474
|
-
except Failure as exc:
|
|
475
|
-
self._logger.warning(f"Failed to request commands for {self.__class__.__name__}: {exc}")
|
|
476
|
-
return False
|
|
477
|
-
else:
|
|
478
|
-
self._add_commands()
|
|
479
|
-
return True
|
|
480
|
-
else:
|
|
481
|
-
self._logger.warning(f"{self.__class__.__name__} is not connected, try to reconnect.")
|
|
482
|
-
return False
|
|
483
|
-
|
|
484
|
-
def get_commands(self):
|
|
485
|
-
"""
|
|
486
|
-
Returns a list of command names that can be send to the device or the
|
|
487
|
-
control server.
|
|
488
|
-
|
|
489
|
-
The commands are defined in the YAML settings file of the device.
|
|
490
|
-
Special commands are available for the ServiceProxy which configure and
|
|
491
|
-
control the control servers.
|
|
492
|
-
"""
|
|
493
|
-
return list(self._commands.keys())
|
|
494
|
-
|
|
495
|
-
def has_commands(self):
|
|
496
|
-
"""Return `True` if commands have been loaded."""
|
|
497
|
-
return bool(self._commands)
|
|
498
|
-
|
|
499
|
-
def ping(self):
|
|
500
|
-
return_code = self.send("Ping", retries=0, timeout=1000)
|
|
501
|
-
self._logger.log(0, f"Check if control server is available: Ping - {return_code}")
|
|
502
|
-
return return_code == "Pong"
|
|
503
|
-
|
|
504
|
-
def get_endpoint(self):
|
|
505
|
-
""" Returns the endpoint.
|
|
506
|
-
|
|
507
|
-
Returns:
|
|
508
|
-
- Endpoint.
|
|
509
|
-
"""
|
|
510
|
-
|
|
511
|
-
return self._endpoint
|
|
512
|
-
|
|
513
|
-
def get_monitoring_port(self) -> int:
|
|
514
|
-
""" Returns the monitoring port. """
|
|
515
|
-
|
|
516
|
-
return self.send("get_monitoring_port")
|
|
517
|
-
|
|
518
|
-
def get_commanding_port(self) -> int:
|
|
519
|
-
""" Returns the commanding port."""
|
|
520
|
-
|
|
521
|
-
return self.send("get_commanding_port")
|
|
522
|
-
|
|
523
|
-
def get_service_port(self) -> int:
|
|
524
|
-
""" Returns the service port. """
|
|
525
|
-
|
|
526
|
-
return self.send("get_service_port")
|
|
527
|
-
|
|
528
|
-
def get_ip_address(self) -> int:
|
|
529
|
-
""" Returns the hostname of the control server."""
|
|
530
|
-
|
|
531
|
-
return self.send("get_ip_address")
|
egse/randomwalk.py
DELETED
|
@@ -1,140 +0,0 @@
|
|
|
1
|
-
"""
|
|
2
|
-
This module implements a random walk that can be used in simulators for devices like temperature sensors or power meter.
|
|
3
|
-
"""
|
|
4
|
-
from __future__ import annotations
|
|
5
|
-
|
|
6
|
-
import random
|
|
7
|
-
import sys
|
|
8
|
-
from typing import Optional
|
|
9
|
-
|
|
10
|
-
|
|
11
|
-
# Re-implement based on:
|
|
12
|
-
#
|
|
13
|
-
# 1. https://isquared.digital/blog/2020-04-12-random-walk/
|
|
14
|
-
# 2. https://isquared.digital/blog/2020-04-16-brownian-motion/
|
|
15
|
-
# 3. https://isquared.digital/blog/2020-05-01-drifted-brownian-motion/
|
|
16
|
-
|
|
17
|
-
class RandomWalk:
|
|
18
|
-
"""
|
|
19
|
-
This class implements a random walk.
|
|
20
|
-
|
|
21
|
-
The RandomWalk is an iterator which can be used in a for loop:
|
|
22
|
-
|
|
23
|
-
```Python
|
|
24
|
-
import matplotlib.pyplot as plt
|
|
25
|
-
values = [value for value in RandomWalk(start=15.0, boundary=(-100, 25), scale=0.01, count=10000)]
|
|
26
|
-
plt.plot(values)
|
|
27
|
-
plt.show()
|
|
28
|
-
```
|
|
29
|
-
|
|
30
|
-
The code above will create a plot like below, where 10000 values are plot in a random walk between -100 and 25,
|
|
31
|
-
starting at 15:
|
|
32
|
-
|
|
33
|
-

|
|
34
|
-
|
|
35
|
-
When you need a random walk that will return values infinitely, set count = 0.
|
|
36
|
-
|
|
37
|
-
By default, the returned values will represent a random walk between -1.0 and 1.0 in a step of 1.0.
|
|
38
|
-
"""
|
|
39
|
-
def __init__(self, start: float = 0.0, boundary: tuple = (-1.0, 1.0), threshold: tuple | float = 0.5,
|
|
40
|
-
scale: float = 1.0, count: int = 1000, seed: Optional[int] = None):
|
|
41
|
-
"""
|
|
42
|
-
|
|
43
|
-
Args:
|
|
44
|
-
start: the start value for the random walk [default = 0.0]
|
|
45
|
-
boundary: the lower and upper boundaries [default = (-1.0, 1.0)]
|
|
46
|
-
threshold: the probability to move up or down [default = 0.5]
|
|
47
|
-
scale: scale the step size [default = 1.0]
|
|
48
|
-
count: the number of iterations, when <= 0 count will be basically infinite [default = 1000]
|
|
49
|
-
seed: seed for the initialization of the random generator [default = None]
|
|
50
|
-
"""
|
|
51
|
-
self._threshold_up = threshold[1] if isinstance(threshold, tuple) else threshold
|
|
52
|
-
self._threshold_down = threshold[0] if isinstance(threshold, tuple) else threshold
|
|
53
|
-
self._boundary = boundary
|
|
54
|
-
self._min_value, self._max_value = boundary
|
|
55
|
-
self._count = count if count > 0 else sys.maxsize
|
|
56
|
-
self._start = start
|
|
57
|
-
self._last = start
|
|
58
|
-
self._scale = scale
|
|
59
|
-
self._seed = seed
|
|
60
|
-
self._random = random.Random()
|
|
61
|
-
|
|
62
|
-
if seed is not None:
|
|
63
|
-
self._random.seed(seed)
|
|
64
|
-
|
|
65
|
-
def __iter__(self):
|
|
66
|
-
return self
|
|
67
|
-
|
|
68
|
-
def __next__(self):
|
|
69
|
-
if self._count > 0:
|
|
70
|
-
probability = self._random.random()
|
|
71
|
-
if probability > self._threshold_up:
|
|
72
|
-
value = self._last + self._scale
|
|
73
|
-
value = min(value, self._max_value)
|
|
74
|
-
elif probability < self._threshold_down:
|
|
75
|
-
value = self._last - self._scale
|
|
76
|
-
value = max(value, self._min_value)
|
|
77
|
-
else:
|
|
78
|
-
value = self._last
|
|
79
|
-
self._last = value
|
|
80
|
-
self._count -= 1
|
|
81
|
-
return value
|
|
82
|
-
raise StopIteration
|
|
83
|
-
|
|
84
|
-
|
|
85
|
-
if __name__ == "__main__":
|
|
86
|
-
|
|
87
|
-
# https://towardsdatascience.com/generating-synthetic-time-series-data-with-random-walks-8701bb9a56a8
|
|
88
|
-
|
|
89
|
-
import matplotlib.pyplot as plt
|
|
90
|
-
|
|
91
|
-
# Random Walk, each time a different seed by default
|
|
92
|
-
|
|
93
|
-
y1 = RandomWalk(start=50.0, boundary=(0, 100), threshold=(0.25, 0.70), scale=1, count=1000)
|
|
94
|
-
# y2 = RandomWalk(start=50.0, boundary=(0, 100), threshold=0.49, scale=0.05, count=100000, seed=0)
|
|
95
|
-
# y3 = RandomWalk(start=50.0, boundary=(0, 100), threshold=0.48, scale=0.01, count=100000, seed=0)
|
|
96
|
-
# plt.plot(list(zip(y1, y2, y3)))
|
|
97
|
-
plt.plot(list(y1))
|
|
98
|
-
plt.show()
|
|
99
|
-
|
|
100
|
-
y1 = RandomWalk(start=50.0, boundary=(0, 100), threshold=0.5, scale=0.2, count=100000, seed=0)
|
|
101
|
-
y2 = RandomWalk(start=50.0, boundary=(0, 100), threshold=0.51, scale=0.2, count=100000, seed=0)
|
|
102
|
-
y3 = RandomWalk(start=50.0, boundary=(0, 100), threshold=0.52, scale=0.2, count=100000, seed=0)
|
|
103
|
-
plt.plot(list(zip(y1, y2, y3)))
|
|
104
|
-
plt.show()
|
|
105
|
-
|
|
106
|
-
y1 = RandomWalk(start=75.0, boundary=(0, 100), threshold=0.5, scale=0.2, count=100000, seed=0)
|
|
107
|
-
y2 = RandomWalk(start=50.0, boundary=(0, 100), threshold=0.5, scale=0.2, count=100000, seed=0)
|
|
108
|
-
y3 = RandomWalk(start=25.0, boundary=(0, 100), threshold=0.5, scale=0.2, count=100000, seed=0)
|
|
109
|
-
plt.plot(list(zip(y1, y2, y3)))
|
|
110
|
-
plt.show()
|
|
111
|
-
|
|
112
|
-
# Three random walks, two of them are fixed and reproducible
|
|
113
|
-
|
|
114
|
-
rng1 = RandomWalk(start=15.0, boundary=(-20, 25), scale=0.8, count=10000, seed=1)
|
|
115
|
-
rng2 = RandomWalk(start=15.0, boundary=(-20, 25), scale=0.5, count=10000, seed=1)
|
|
116
|
-
rng3 = RandomWalk(start=15.0, boundary=(-20, 25), scale=0.3, count=10000, seed=1)
|
|
117
|
-
|
|
118
|
-
values = list(zip(rng1, rng2, rng3))
|
|
119
|
-
plt.plot(values)
|
|
120
|
-
plt.ylim(-20, 25)
|
|
121
|
-
plt.show()
|
|
122
|
-
|
|
123
|
-
# Example 1:
|
|
124
|
-
|
|
125
|
-
rw = RandomWalk(start=25.0, boundary=(-100, 100), scale=0.2, count=0, seed=42)
|
|
126
|
-
values = [next(rw) for _ in range(1000000)]
|
|
127
|
-
plt.plot(values)
|
|
128
|
-
plt.show()
|
|
129
|
-
|
|
130
|
-
# Example 2: should give the same output as example 1
|
|
131
|
-
|
|
132
|
-
plt.plot(
|
|
133
|
-
list(RandomWalk(start=25.0, boundary=(-100, 100), scale=0.2, count=1000000, seed=42,))
|
|
134
|
-
)
|
|
135
|
-
plt.show()
|
|
136
|
-
|
|
137
|
-
# Example 3:
|
|
138
|
-
|
|
139
|
-
for i, value in enumerate(RandomWalk(start=25.0, boundary=(-100, 100), threshold=0.5, scale=2, count=20)):
|
|
140
|
-
print(f"{i}, {value}")
|