diff --git a/.gitlab-ci.yml b/.gitlab-ci.yml index c4d3d5ba6246b038763fc27a2199497d8682d109..398ce9c49319e1b96243f2b4d8cb0949eca08938 100644 --- a/.gitlab-ci.yml +++ b/.gitlab-ci.yml @@ -1,6 +1,7 @@ stages: - check - test + - automated_test .before_script: &before_script before_script: @@ -45,11 +46,30 @@ pytest: - export LANG=C # Hopefully detect anything relying on locale - python3 -m pip install ".[test]" - python3 -m pytest --color yes --verbose --cov=cal_tools --cov=xfel_calibrate -# Nope... https://docs.gitlab.com/12.10/ee/user/project/merge_requests/test_coverage_visualization.html#enabling-the-feature -# - coverage xml -# artifacts: -# reports: -# cobertura: coverage.xml + # Nope... https://docs.gitlab.com/12.10/ee/user/project/merge_requests/test_coverage_visualization.html#enabling-the-feature + # - coverage xml + # artifacts: + # reports: + # cobertura: coverage.xml + +automated_test: + variables: + OUTPUT: $CI_MERGE_REQUEST_SOURCE_BRANCH_NAME + REFERENCE: reference_folder + DETECTORS: all + CALIBRATION: all + stage: automated_test + only: [merge_requests] + when: manual + allow_failure: false + <<: *before_script + script: + - export LANG=C # Hopefully detect anything relying on locale + - python3 -m pip install ".[automated_test]" + - echo "Running automated test. This can take sometime to finish depending on the test data." + - echo "Given variables are REFERENCE=$REFERENCE, OUTPUT=$OUTPUT, DETECTOR=$DETECTORS, CALIBRATION=$CALIBRATION" + - python3 -m pytest ./tests/test_reference_runs --color yes --verbose --release-test --reference-folder /gpfs/exfel/data/scratch/xcaltst/test/$REFERENCE --out-folder /gpfs/exfel/data/scratch/xcaltst/test/$OUTPUT --detectors $DETECTORS --calibration $CALIBRATION + timeout: 24 hours cython-editable-install-test: stage: test @@ -57,4 +77,4 @@ cython-editable-install-test: <<: *before_script script: - python3 -m pip install -e ".[test]" - - python3 -m pytest --color yes --verbose ./tests/test_cythonalgs.py + - python3 -m pytest --color yes --verbose ./tests/test_cythonalgs.py diff --git a/setup.py b/setup.py index 9d2db821165c9533c7189eb7a13f7ddcfe5067af..bec7ab7ea42aabd602f3e98925d61df2f57e47da 100644 --- a/setup.py +++ b/setup.py @@ -161,6 +161,16 @@ setup( "testpath", "unittest-xml-reporting==3.0.2", ], + "automated_test": [ + "coverage", + "nbval", + "pytest-asyncio", + "pytest-cov", + "pytest-subprocess", + "pytest>=5.4.0", + "testpath", + "unittest-xml-reporting==3.0.2", + ], "dev": [ "nbqa[toolchain]", "pre-commit", diff --git a/tests/pytest.ini b/tests/pytest.ini new file mode 100644 index 0000000000000000000000000000000000000000..74e59734360638a1a5a894bb5ae9647dbd8e22a5 --- /dev/null +++ b/tests/pytest.ini @@ -0,0 +1,6 @@ +[pytest] +asyncio_mode = strict +log_cli = 1 +log_cli_level = INFO +log_cli_format = %(asctime)s [%(levelname)8s] %(message)s (%(filename)s:%(lineno)s) +log_cli_date_format=%Y-%m-%d %H:%M:%S \ No newline at end of file diff --git a/tests/test_reference_runs/__init__.py b/tests/test_reference_runs/__init__.py new file mode 100644 index 0000000000000000000000000000000000000000..e69de29bb2d1d6434b8b29ae775ad8c2e48c5391 diff --git a/tests/test_reference_runs/callab_tests.py b/tests/test_reference_runs/callab_tests.py new file mode 100644 index 0000000000000000000000000000000000000000..e24e48639fd52a2caeba9c09ef0686cc589f898a --- /dev/null +++ b/tests/test_reference_runs/callab_tests.py @@ -0,0 +1,644 @@ +automated_test_config = { + "SPB_DET_AGIPD1M-1-CORRECT-FIXED": { + "det_type": "AGIPD", + "cal_type": "CORRECT", + "config": { + "blc-stripes": True, + "out-folder": "{}/{}/{}", + # "/gpfs/exfel/exp/SPB/202131/p900215/raw" + "in-folder": "/gpfs/exfel/exp/CALLAB/202130/p900203/raw", + "run": "9014", # Original run: "95" + "karabo-id-control": "SPB_IRU_AGIPD1M1", + "karabo-id": "SPB_DET_AGIPD1M-1", + "slurm-mem": "750", + "sequences": "0", + "rel-gain": True, + "n-cores-files": 2, + "ctrl-source-template": "{}/MDL/FPGA_COMP", + }, + "reference-folder": "{}/{}/{}", + }, + "SPB_DET_AGIPD1M-1-DARK-FIXED": { + "det_type": "AGIPD", + "cal_type": "DARK", + "config": { + "out-folder": "{}/{}/{}", + # "/gpfs/exfel/exp/SPB/202131/p900215/raw" + "in-folder": "/gpfs/exfel/exp/CALLAB/202130/p900203/raw", + "run-high": "9011", # Original run: "91" + "run-med": "9012", # Original run: "92" + "run-low": "9013", # Original run "93" + "karabo-id-control": "SPB_IRU_AGIPD1M1", + "karabo-id": "SPB_DET_AGIPD1M-1", + "ctrl-source-template": "{}/MDL/FPGA_COMP", + }, + "reference-folder": "{}/{}/{}", + }, + "SPB_DET_AGIPD1M-1-CORRECT-ADAPTIVE": { + "det_type": "AGIPD", + "cal_type": "CORRECT", + "config": { + "blc-stripes": True, + "rel-gain": True, + "out-folder": "{}/{}/{}", + # "/gpfs/exfel/exp/SPB/202131/p900215/raw" + "in-folder": "/gpfs/exfel/exp/CALLAB/202130/p900203/raw", + "run": "9018", # Original run 262 + "karabo-id-control": "SPB_IRU_AGIPD1M1", + "karabo-id": "SPB_DET_AGIPD1M-1", + "slurm-mem": "750", + "sequences": "0", + "n-cores-files": 2, + "ctrl-source-template": "{}/MDL/FPGA_COMP", + }, + "reference-folder": "{}/{}/{}", + }, + "SPB_DET_AGIPD1M-1-DARK-ADAPTIVE": { + "det_type": "AGIPD", + "cal_type": "DARK", + "config": { + "out-folder": "{}/{}/{}", + # "/gpfs/exfel/exp/SPB/202131/p900215/raw" + "in-folder": "/gpfs/exfel/exp/CALLAB/202130/p900203/raw", + "run-high": "9015", # Original run: "216", + "run-med": "9016", # Original run: "217", + "run-low": "9017", # Original run: "218", + "karabo-id-control": "SPB_IRU_AGIPD1M1", + "karabo-id": "SPB_DET_AGIPD1M-1", + "ctrl-source-template": "{}/MDL/FPGA_COMP", + }, + "reference-folder": "{}/{}/{}", + }, + "MID_DET_AGIPD1M-1-CORRECT-ADAPTIVE": { + "det_type": "AGIPD", + "cal_type": "CORRECT", + "config": { + "blc-stripes": True, + "rel-gain": True, + "out-folder": "{}/{}/{}", + # "/gpfs/exfel/exp/MID/202121/p002929/raw" + "in-folder": "/gpfs/exfel/exp/CALLAB/202130/p900203/raw", + "run": "9022", # Original run: "21", + "karabo-id-control": "MID_EXP_AGIPD1M1", + "karabo-id": "MID_DET_AGIPD1M-1", + "slurm-mem": "750", + "sequences": "0,1", + "n-cores-files": 2, + "ctrl-source-template": "{}/MDL/FPGA_COMP", + }, + "reference-folder": "{}/{}/{}", + }, + "MID_DET_AGIPD1M-1-CORRECT-SELECT_CELLS_CM": { + "det_type": "AGIPD", + "cal_type": "CORRECT", + "config": { + "blc-stripes": True, + "rel-gain": True, + "out-folder": "{}/{}/{}", + # "/gpfs/exfel/exp/MID/202121/p002929/raw" + "in-folder": "/gpfs/exfel/exp/CALLAB/202130/p900203/raw", + "run": "9022", # Original run: "21", + "max-pulses": [0, 160, 2], + "common-mode": True, + "karabo-id-control": "MID_EXP_AGIPD1M1", + "karabo-id": "MID_DET_AGIPD1M-1", + "slurm-mem": "750", + "sequences": "0,1", + "ctrl-source-template": "{}/MDL/FPGA_COMP", + "n-cores-files": 2, + "sequences-per-node": 1, + }, + "reference-folder": "{}/{}/{}", + }, + "MID_DET_AGIPD1M-1-CORRECT-SELECT_CELLS_NOCM": { + "det_type": "AGIPD", + "cal_type": "CORRECT", + "config": { + "blc-stripes": True, + "rel-gain": True, + "out-folder": "{}/{}/{}", + # "/gpfs/exfel/exp/MID/202121/p002929/raw" + "in-folder": "/gpfs/exfel/exp/CALLAB/202130/p900203/raw", + "run": "9022", # Original run: "21", + "max-pulses": [0, 160, 2], + "no-common-mode": True, + "karabo-id-control": "MID_EXP_AGIPD1M1", + "karabo-id": "MID_DET_AGIPD1M-1", + "slurm-mem": "750", + "sequences": "0,1", + "n-cores-files": 2, + "ctrl-source-template": "{}/MDL/FPGA_COMP", + }, + "reference-folder": "{}/{}/{}", + }, + "MID_DET_AGIPD1M-1-DARK-ADAPTIVE": { + "det_type": "AGIPD", + "cal_type": "DARK", + "config": { + "out-folder": "{}/{}/{}", + # "/gpfs/exfel/exp/MID/202121/p002929/raw" + "in-folder": "/gpfs/exfel/exp/CALLAB/202130/p900203/raw", + "run-high": "9019", # Original run: "4" + "run-med": "9020", # Original run: "5" + "run-low": "9021", # Original run: "6" + "karabo-id-control": "MID_EXP_AGIPD1M1", + "karabo-id": "MID_DET_AGIPD1M-1", + "ctrl-source-template": "{}/MDL/FPGA_COMP", + }, + "reference-folder": "{}/{}/{}", + }, + "HED_DET_AGIPD500K2G-CORRECT-ADAPTIVE": { + "det_type": "AGIPD", + "cal_type": "CORRECT", + "config": { + "blc-stripes": True, + "rel-gain": True, + "out-folder": "{}/{}/{}", + # "/gpfs/exfel/exp/HED/202131/p900228/raw" + "in-folder": "/gpfs/exfel/exp/CALLAB/202130/p900203/raw", + "run": "9026", # Original run: "29", + "karabo-id-control": "HED_EXP_AGIPD500K2G", + "karabo-id": "HED_DET_AGIPD500K2G", + "slurm-mem": "750", + "n-cores-files": 2, + "sequences": "0,1", + "ctrl-source-template": "{}/MDL/FPGA_COMP", + }, + "reference-folder": "{}/{}/{}", + }, + "HED_DET_AGIPD500K2G-DARK-ADAPTIVE": { + "det_type": "AGIPD", + "cal_type": "DARK", + "config": { + "out-folder": "{}/{}/{}", + # "/gpfs/exfel/exp/HED/202131/p900228/raw" + "in-folder": "/gpfs/exfel/exp/CALLAB/202130/p900203/raw", + "run-high": "9023", # Original run: "25", + "run-med": "9024", # Original run: "26", + "run-low": "9025", # Original run: "27", + "karabo-id-control": "HED_EXP_AGIPD500K2G", + "karabo-id": "HED_DET_AGIPD500K2G", + "ctrl-source-template": "{}/MDL/FPGA_COMP", + }, + "reference-folder": "{}/{}/{}", + }, + "FXE_XAD_JF500K-DARK-SINGLE": { + "det_type": "JUNGFRAU", + "cal_type": "DARK", + "config": { + "out-folder": "{}/{}/{}", + # "/gpfs/exfel/exp/FXE/202101/p002478/raw/" + "in-folder": "/gpfs/exfel/exp/CALLAB/202130/p900203/raw", + "run-high": "9031", # Original run: "47" + "run-med": "9032", # Original run: "48" + "run-low": "9033", # Original run: "49" + "karabo-id-control": "", + "karabo-id": "FXE_XAD_JF500K", + "karabo-da": "JNGFR03", + }, + "reference-folder": "{}/{}/{}", + }, + "FXE_XAD_JF500K-CORRECT-SINGLE": { + "det_type": "JUNGFRAU", + "cal_type": "CORRECT", + "config": { + "out-folder": "{}/{}/{}", + # "/gpfs/exfel/exp/FXE/202101/p002478/raw/" + "in-folder": "/gpfs/exfel/exp/CALLAB/202130/p900203/raw", + "run": "9030", # Original from: 52 + "sequences": "0,50", + "karabo-id-control": "", + "karabo-id": "FXE_XAD_JF500K", + "karabo-da": "JNGFR03", + }, + "reference-folder": "{}/{}/{}", + }, + "FXE_XAD_JF1M-DARK-BURST": { + "det_type": "JUNGFRAU", + "cal_type": "DARK", + "config": { + "out-folder": "{}/{}/{}", + # "/gpfs/exfel/exp/FXE/202131/p900226/raw/" + "in-folder": "/gpfs/exfel/exp/CALLAB/202130/p900203/raw", + "run-high": "9035", # Original run: "108" + "run-med": "9036", # Original run: "109" + "run-low": "9037", # Original run: "110" + "karabo-id-control": "", + "karabo-id": "FXE_XAD_JF1M", + "karabo-da": ["JNGFR01", "JNGFR02"], + }, + "reference-folder": "{}/{}/{}", + }, + "FXE_XAD_JF1M-CORRECT-BURST": { + "det_type": "JUNGFRAU", + "cal_type": "CORRECT", + "config": { + "out-folder": "{}/{}/{}", + "in-folder": "/gpfs/exfel/exp/CALLAB/202130/p900203/raw", + # "/gpfs/exfel/exp/FXE/202131/p900226/raw" + "run": "9034", # Run cloned from run 106. + "sequences": "0,1", + "karabo-id-control": "", + "karabo-id": "FXE_XAD_JF1M", + "karabo-da": ["JNGFR01", "JNGFR02"], + }, + "reference-folder": "{}/{}/{}", + }, +# "FXE_XAD_JF1M-DARK-BURST_LONGRUN": { +# "det_type": "JUNGFRAU", +# "cal_type": "DARK", +# "config": { +# "out-folder": "{}/{}/{}", +# "in-folder": "/gpfs/exfel/exp/FXE/202321/p004576/raw", +# "run-high": "112", +# "run-med": "113", +# "run-low": "118", +# "karabo-id-control": "", +# "karabo-id": "FXE_XAD_JF1M", +# "karabo-da": ["JNGFR01", "JNGFR02"], +# }, +# "reference-folder": "{}/{}/{}", +# }, + "FXE_XAD_JF1M-CORRECT-SINGLE": { + "det_type": "JUNGFRAU", + "cal_type": "CORRECT", + "config": { + "out-folder": "{}/{}/{}", + # "/gpfs/exfel/exp/FXE/202101/p002478/raw/" + "in-folder": "/gpfs/exfel/exp/CALLAB/202130/p900203/raw", + "run": "9030", # Original from: 52 + "sequences": "1,30,49", + "karabo-id-control": "", + "karabo-id": "FXE_XAD_JF1M", + "karabo-da": ["JNGFR01", "JNGFR02"], + }, + "reference-folder": "{}/{}/{}", + }, + "FXE_XAD_JF1M-DARK-SINGLE": { + "det_type": "JUNGFRAU", + "cal_type": "DARK", + "config": { + "out-folder": "{}/{}/{}", + # "/gpfs/exfel/exp/FXE/202101/p002478/raw/" + "in-folder": "/gpfs/exfel/exp/CALLAB/202130/p900203/raw", + "run-high": "9031", # Original run: "47" + "run-med": "9032", # Original run: "48" + "run-low": "9033", # Original run: "49" + "karabo-id-control": "", + "karabo-id": "FXE_XAD_JF1M", + "karabo-da": ["JNGFR01", "JNGFR02"], + }, + "reference-folder": "{}/{}/{}", + }, + "SPB_IRDA_JF4M-CORRECT-SINGLE": { + "det_type": "JUNGFRAU", + "cal_type": "CORRECT", + "config": { + "out-folder": "{}/{}/{}", + "in-folder": "/gpfs/exfel/exp/CALLAB/202130/p900203/raw", + "run": "9007", + "karabo-id-control": "", + "karabo-id": "SPB_IRDA_JF4M", + "karabo-da": [ + "JNGFR01", + "JNGFR02", + "JNGFR03", + "JNGFR04", + "JNGFR05", + "JNGFR06", + "JNGFR07", + "JNGFR08", + ], + }, + "reference-folder": "{}/{}/{}", + }, + "SPB_IRDA_JF4M-DARK-ADAPTIVE": { + "det_type": "JUNGFRAU", + "cal_type": "DARK", + "config": { + "out-folder": "{}/{}/{}", + "in-folder": "/gpfs/exfel/exp/CALLAB/202130/p900203/raw", + "run-high": "9004", + "run-med": "9005", + "run-low": "9006", + "karabo-id-control": "", + "karabo-id": "SPB_IRDA_JF4M", + "karabo-da": [ + "JNGFR01", + "JNGFR02", + "JNGFR03", + "JNGFR04", + "JNGFR05", + "JNGFR06", + "JNGFR07", + "JNGFR08", + ], + }, + "reference-folder": "{}/{}/{}", + }, + # MISSING FROM REFERENCE RUNS. + # "SPB_IRDA_JF4M-DARK-FIXED": + # { + # "det_type": "JUNGFRAU", + # "cal_type": "DARK", + # "config": + # { + # "out-folder": "{}/{}/{}", + # "in-folder": "/gpfs/exfel/exp/SPB/202202/p003051/raw", + # "run-high": "166", + # "run-med": "168", + # "run-low": "169", + # "karabo-id-control": "", + # "karabo-id": "SPB_IRDA_JF4M", + # "karabo-da": [ + # "JNGFR01", "JNGFR02", "JNGFR03", "JNGFR04", + # "JNGFR05", "JNGFR06", "JNGFR07", "JNGFR08" + # ], + # }, + # "reference-folder": "{}/{}/{}", + # }, + "HED_IA1_JF500K1-DARK-ADAPTIVE": { + "det_type": "JUNGFRAU", + "cal_type": "DARK", + "config": { + "out-folder": "{}/{}/{}", + # "/gpfs/exfel/exp/HED/202102/p002656/raw" + "in-folder": "/gpfs/exfel/exp/CALLAB/202130/p900203/raw", + "run-high": "9039", # Original run: "219", + "run-med": "9040", # Original run: "220", + "run-low": "9041", # Original run: "221", + "karabo-id": "HED_IA1_JF500K1", + "karabo-da": "JNGFR01", + }, + "reference-folder": "{}/{}/{}", + }, + "HED_IA1_JF500K2-DARK-ADAPTIVE": { + "det_type": "JUNGFRAU", + "cal_type": "DARK", + "config": { + "out-folder": "{}/{}/{}", + # "/gpfs/exfel/exp/HED/202102/p002656/raw" + "in-folder": "/gpfs/exfel/exp/CALLAB/202130/p900203/raw", + "run-high": "9039", # Original run: "219", + "run-med": "9040", # Original run: "220", + "run-low": "9041", # Original run: "221", + "karabo-id": "HED_IA1_JF500K2", + "karabo-da": "JNGFR02", + }, + "reference-folder": "{}/{}/{}", + }, + "HED_IA1_JF500K3-DARK-ADAPTIVE": { + "det_type": "JUNGFRAU", + "cal_type": "DARK", + "config": { + "out-folder": "{}/{}/{}", + # "/gpfs/exfel/exp/HED/202102/p002656/raw" + "in-folder": "/gpfs/exfel/exp/CALLAB/202130/p900203/raw", + "run-high": "9039", # Original run: "219", + "run-med": "9040", # Original run: "220", + "run-low": "9041", # Original run: "221", + "karabo-id": "HED_IA1_JF500K3", + "karabo-da": "JNGFR03", + }, + "reference-folder": "{}/{}/{}", + }, +# "HED_IA1_JF500K1-DARK-FIXED": +# { +# "det_type": "JUNGFRAU", +# "cal_type": "DARK", +# "config": +# { +# "out-folder": "{}/{}/{}", +# "in-folder": "/gpfs/exfel/exp/HED/202131/p900227/raw", +# "run-high": "65", +# "run-med": "66", +# "run-low": "67", +# "karabo-id": "HED_IA1_JF500K1", +# "karabo-da": "JNGFR01", +# }, +# "reference-folder": "{}/{}/{}", +# }, + "HED_IA1_JF500K1-CORRECT-ADAPTIVE": { + "det_type": "JUNGFRAU", + "cal_type": "CORRECT", + "config": { + "out-folder": "{}/{}/{}", + # "/gpfs/exfel/exp/HED/202102/p002656/raw" + "in-folder": "/gpfs/exfel/exp/CALLAB/202130/p900203/raw", + "run": "9038", # Original run: "230" + "sequences": "0,2,4", + "karabo-id": "HED_IA1_JF500K1", + "karabo-da": "JNGFR01", + }, + "reference-folder": "{}/{}/{}", + }, + "HED_IA1_JF500K2-CORRECT-ADAPTIVE": { + "det_type": "JUNGFRAU", + "cal_type": "CORRECT", + "config": { + "out-folder": "{}/{}/{}", + # "/gpfs/exfel/exp/HED/202102/p002656/raw" + "in-folder": "/gpfs/exfel/exp/CALLAB/202130/p900203/raw", + "run": "9038", # Original run: "230" + "sequences": "0", + "karabo-id": "HED_IA1_JF500K2", + "karabo-da": "JNGFR02", + }, + "reference-folder": "{}/{}/{}", + }, + "HED_IA1_JF500K3-CORRECT-ADAPTIVE": { + "det_type": "JUNGFRAU", + "cal_type": "CORRECT", + "config": { + "out-folder": "{}/{}/{}", + # "/gpfs/exfel/exp/HED/202102/p002656/raw" + "in-folder": "/gpfs/exfel/exp/CALLAB/202130/p900203/raw", + "run": "9038", # Original run: "230" + "sequences": "1,3", + "karabo-id": "HED_IA1_JF500K3", + "karabo-da": "JNGFR03", + }, + "reference-folder": "{}/{}/{}", + }, + "HED_IA1_EPX100-1-DARK": { + "det_type": "EPIX100", + "cal_type": "DARK", + "config": { + "out-folder": "{}/{}/{}", + # "/gpfs/exfel/exp/HED/202102/p002656/raw" + "in-folder": "/gpfs/exfel/exp/CALLAB/202130/p900203/raw", + "run": "9047", # Original run: "219", + "karabo-id": "HED_IA1_EPX100-1", + "karabo-da": "EPIX01", + }, + "reference-folder": "{}/{}/{}", + }, + # "MID_EXP_EPIX-1-CORRECT": { + # "det_type": "EPIX100", + # "cal_type": "CORRECT", + # "config": { + # "out-folder": "{}/{}/{}", + # # Original proposal 2936 + # "in-folder": "/gpfs/exfel/exp/CALLAB/202130/p900203/raw", + # "run": "9042", # Original run: "160" + # "karabo-id": "MID_EXP_EPIX-1", + # "karabo-da": "EPIX01", + # "sequences": "1,2,3", + # "fix-temperature": 290, + # }, + # "reference-folder": "{}/{}/{}", + # }, + "HED_IA1_EPX100-1-CORRECT": { + "det_type": "EPIX100", + "cal_type": "CORRECT", + "config": { + "out-folder": "{}/{}/{}", + # "/gpfs/exfel/exp/HED/202102/p002656/raw" + "in-folder": "/gpfs/exfel/exp/CALLAB/202130/p900203/raw", + "run": "9038", # Original run: "230" + "karabo-id": "HED_IA1_EPX100-1", + "karabo-da": "EPIX01", + "sequences": "1,2,3", + "fix-temperature": 290, + }, + "reference-folder": "{}/{}/{}", + }, + "MID_EXP_EPIX-2-DARK": { + "det_type": "EPIX100", + "cal_type": "DARK", + "config": { + "out-folder": "{}/{}/{}", + "in-folder": "/gpfs/exfel/exp/CALLAB/202130/p900203/raw", + "run": "9045", # Original run: 224 proposal 2655 + "karabo-id": "MID_EXP_EPIX-2", + "karabo-da": "EPIX02", + }, + "reference-folder": "{}/{}/{}", + }, + "SQS_NQS_PNCCD1MP-CORRECT": { + "det_type": "PNCCD", + "cal_type": "CORRECT", + "config": { + "out-folder": "{}/{}/{}", + # "/gpfs/exfel/exp/SQS/202031/p900166/raw", + "in-folder": "/gpfs/exfel/exp/CALLAB/202130/p900203/raw", + "run": "9001", # Original run: "347", + "karabo-id": "SQS_NQS_PNCCD1MP", + "karabo-da": "PNCCD01", + }, + "reference-folder": "{}/{}/{}", + }, + "SQS_NQS_PNCCD1MP-DARK": { + "det_type": "PNCCD", + "cal_type": "DARK", + "config": { + "out-folder": "{}/{}/{}", + # "/gpfs/exfel/exp/SQS/202031/p900166/raw" + "in-folder": "/gpfs/exfel/exp/CALLAB/202130/p900203/raw", + "run": "9000", # Original run: "345", + "karabo-id": "SQS_NQS_PNCCD1MP", + "karabo-da": "PNCCD01", + }, + "reference-folder": "{}/{}/{}", + }, + "FXE_DET_LPD1M-1-DARK": { + "det_type": "LPD", + "cal_type": "DARK", + "config": { + "out-folder": "{}/{}/{}", + # "/gpfs/exfel/exp/FXE/202131/p900226/raw" + "in-folder": "/gpfs/exfel/exp/CALLAB/202130/p900203/raw", + "run-high": "9008", # Original run: "145", + "run-med": "9009", # Original run: "146", + "run-low": "9010", # Original run: "147", + "karabo-id": "FXE_DET_LPD1M-1", + }, + "reference-folder": "{}/{}/{}", + }, + "FXE_DET_LPD1M-1-CORRECT": { # THIS IS A DARK RUN. + "det_type": "LPD", + "cal_type": "CORRECT", + "config": { + "out-folder": "{}/{}/{}", + # "/gpfs/exfel/exp/FXE/202131/p900226/raw" + "in-folder": "/gpfs/exfel/exp/CALLAB/202130/p900203/raw", + "run": "9008", # Original run: "145", + "sequences": "0", + "karabo-id": "FXE_DET_LPD1M-1", + "slurm-mem": "750", + "num-workers": 4, + }, + "reference-folder": "{}/{}/{}", + }, + "SCS_DET_DSSC1M-1-DARK": { + "det_type": "DSSC", + "cal_type": "DARK", + "config": { + "out-folder": "{}/{}/{}", + # "/gpfs/exfel/exp/SCS/202122/p002937/raw" + "in-folder": "/gpfs/exfel/exp/CALLAB/202130/p900203/raw", + "run": "9028", # Original run: "1723", + "karabo-id": "SCS_DET_DSSC1M-1", + "slow-data-path": "SCS_CDIDET_DSSC/FPGA/PPT_Q", + "slow-data-aggregators": + - 1 + - 2 + - 3 + - 4 + }, + "reference-folder": "{}/{}/{}", + }, + "SCS_DET_DSSC1M-1-CORRECT": { + "det_type": "DSSC", + "cal_type": "CORRECT", + "config": { + "out-folder": "{}/{}/{}", + # "/gpfs/exfel/exp/SCS/202122/p002937/raw" + "in-folder": "/gpfs/exfel/exp/CALLAB/202130/p900203/raw", + "run": "9028", # Original run: "1723", + "karabo-id": "SCS_DET_DSSC1M-1", + "slow-data-path": "SCS_CDIDET_DSSC/FPGA/PPT_Q", + }, + "reference-folder": "{}/{}/{}", + }, + # "FXE_XAD_G2XES-DARK": { + # "det_type": "Gotthard2", + # "cal_type": "DARK", + # "config": { + # "out-folder": "{}/{}/{}", + # "in-folder": "/gpfs/exfel/exp/FXE/202231/p900298/raw", + # "karabo-da": "GH201", + # "run-high": "7", + # "run-med": "8", + # "run-low": "9", + # "karabo-id": "FXE_XAD_G2XES", + # }, + # "reference-folder": "{}/{}/{}", + # }, + # "FXE_XAD_G2XES-CORRECT": { + # "det_type": "Gotthard2", + # "cal_type": "CORRECT", + # "config": { + # "out-folder": "{}/{}/{}", + # "in-folder": "/gpfs/exfel/exp/FXE/202231/p900298/raw", + # "karabo-da": "GH201", + # "run": "7", + # "karabo-id": "FXE_XAD_G2XES", + # "no-offset-correction": True, + # }, + # "reference-folder": "{}/{}/{}", + # }, + # "SPB_50UM_GH2-CORRECT": { + # "det_type": "Gotthard2", + # "cal_type": "CORRECT", + # "config": { + # "out-folder": "{}/{}/{}", + # "in-folder": "/gpfs/exfel/exp/SPB/202321/p004577/raw", + # "karabo-da": "GH200", + # "run": "98", + # "sequences": "0,3,6", + # "karabo-id": "SPB_50UM_GH2", + # }, + # "reference-folder": "{}/{}/{}", + # }, +} diff --git a/tests/test_reference_runs/conftest.py b/tests/test_reference_runs/conftest.py new file mode 100644 index 0000000000000000000000000000000000000000..19e3d9cdc8d2449b2696b63dd0a83851627fc080 --- /dev/null +++ b/tests/test_reference_runs/conftest.py @@ -0,0 +1,111 @@ +import pytest + + +def pytest_addoption(parser): + + parser.addoption( + "--release-test", + action="store_true", + default=False, + help="Start release test for all supported calibration notebooks", + ) + # detectors_choices = [] + # for k in ['agipd', 'jungfrau', 'pnccd', 'epix100', 'all']: + # detectors_choices += [k, k.upper()] + parser.addoption( + "--detectors", + default=["all"], + nargs="+", + ) + + parser.addoption( + "--calibration", + type=str, + choices=["correct", "dark", "all"], + default="all", + ) + + parser.addoption( + "--no-numerical-validation", + action="store_true", + default=False, + help="Skips tests for numerical validation for produced h5files.", + ) + + parser.addoption( + "--validation-only", + action="store_true", + default=False, + help=("Skips running xfel-calibrate CLI and " + "apply validation test on numerical data only."), + ) + + parser.addoption( + "--find-difference", + action="store_true", + default=False, + help=( + "In case of non numerical validation of h5file. " + "Find the different attribute and fail on the first one."), + ) + + parser.addoption( + "--use-slurm", + action="store_true", + default=False, + help=("Run xfel-calibrate CLI without " + "--no-cluster-job option and use Slurm"), + ) + + parser.addoption( + "--picked-test", + type=str, + default=None, + ) + + parser.addoption( + "--out-folder", + type=str, + ) + + parser.addoption( + "--reference-folder", + type=str, + ) + + +@pytest.fixture +def release_test_config(request): + detectors = request.config.getoption("--detectors") + calibration = request.config.getoption("--calibration") + skip_numerical_validation = request.config.getoption( + "--no-numerical-validation") + validate_only = request.config.getoption( + "--validation-only") + find_difference = request.config.getoption( + "--find-difference") + use_slurm = request.config.getoption( + "--use-slurm") + picked_test = request.config.getoption("--picked-test") + reference_folder = request.config.getoption("--reference-folder") + out_folder = request.config.getoption("--out-folder") + return ( + detectors, calibration, picked_test, + skip_numerical_validation, validate_only, + find_difference, use_slurm, reference_folder, + out_folder, + ) + + +def pytest_configure(config): + config.addinivalue_line( + "markers", + "manual_run(): marks skips for tests that required to be run manually", + ) + + +def pytest_runtest_setup(item): + if list(item.iter_markers(name="manual_run")) and not item.config.getoption( + "--release-test" + ): + pytest.skip("Test initialized manually") diff --git a/tests/test_reference_runs/test_pre_deployment.py b/tests/test_reference_runs/test_pre_deployment.py new file mode 100644 index 0000000000000000000000000000000000000000..7a4f36e74191ba90380f76439b8a6893b70a1ff8 --- /dev/null +++ b/tests/test_reference_runs/test_pre_deployment.py @@ -0,0 +1,499 @@ +import hashlib +import io +import logging +import multiprocessing +import pathlib +import tempfile +import time +from contextlib import redirect_stdout +from datetime import datetime +from functools import partial +from subprocess import PIPE, run +from typing import Any, Dict, List, Tuple + +import h5py +import numpy as np +import pytest + +import xfel_calibrate.calibrate as calibrate + +from .callab_tests import automated_test_config + +LOGGER = logging.getLogger(__name__) + + +def file_md5( + tested_file: str, + block_size: int = 2 ** 20, +) -> bytes: + """Generating MD5 checksum for a file. + + Args: + tested_file: File to be tested. + block_size (_type_, optional): Block size for reading the file. + Defaults to 2**20. + """ + f = open(tested_file, "rb") + md5 = hashlib.md5() + while True: + data = f.read(block_size) + if not data: + break + md5.update(data) + f.close() + return md5.digest() + + +def collect_attrs(groups, datasets, objects, exclude_attrs, name, node): + """Collect h5 attrs in groups, datasets, and objects lists.""" + if node.name not in exclude_attrs: + if isinstance(node, h5py.Group): + groups.append(name) + elif isinstance(node, h5py.Dataset): + if node.dtype == 'object': + objects.append(name) + else: + datasets.append(name) + + +def compare_datasets( + file1, + file2, + datasets: list +): + """Compare the values of datasets in two h5 files.""" + h5_diff = [] + for d in datasets: + try: + if not np.allclose(file1[d][()], file2[d][()], equal_nan=True): + h5_diff.append(d) + except ValueError as e: + LOGGER.error(f"ValueError: {e}, {d}") + h5_diff.append(d) + except AttributeError as e: + LOGGER.error(f"AttributeError: {e}, {d}") + h5_diff.append(d) + return h5_diff + + +def compare_objects( + file1, + file2, + objects: list +): + """Compare the objects in two h5 files.""" + h5_diff = [] + for d in objects: + try: + if isinstance(file1[d][()], bytes): + if ( + file1[d][()].decode('utf-8') != file2[d][()].decode('utf-8') # noqa + ): + h5_diff.append(d) + elif ( + file1[d][()].dtype != file1[d][()].dtype and + not file1[d][()] != file2[d][()] + ): # pnccd files has only list of bytes + h5_diff.append(d) + except ValueError as e: + LOGGER.error(f"ValueError: {e}, {d}") + h5_diff.append(d) + except AttributeError as e: + LOGGER.error(f"AttributeError: {e}, {d}, " + f"{file1[d][()].decode('utf-8')}") + h5_diff.append(d) + return h5_diff + + +def find_differences( + test_file, + reference_file, + exclude_attrs, +): + """ + Find difference in groups, datasets, and objects between two h5files. + Args: + file1: first h5 file. + file2: second h5 file. + """ + + groups_f1 = [] + datasets_f1 = [] + objects_f1 = [] + + groups_f2 = [] + datasets_f2 = [] + objects_f2 = [] + + with h5py.File(test_file, 'r') as file1, h5py.File(reference_file, 'r') as file2: # noqa + + # Fill groups, datasets, and objects list + # to compare both h5files' attrs. + file1.visititems( + partial( + collect_attrs, + groups_f1, + datasets_f1, + objects_f1, + exclude_attrs, + )) + file2.visititems( + partial( + collect_attrs, + groups_f2, + datasets_f2, + objects_f2, + exclude_attrs, + )) + + start_time = time.perf_counter() + # Compare groups, datasets, and objects to have the same content. + assert set(groups_f1) == set(groups_f2), f"{test_file} and {reference_file} consists of different groups." # noqa + assert set(datasets_f1) == set(datasets_f2), f"{test_file} and {reference_file} consists of different datasets." # noqa + assert set(objects_f1) == set(objects_f2), f"{test_file} and {reference_file} consists of different datasets." # noqa + duration = time.perf_counter() - start_time + LOGGER.info("Elapsed time comparing groups, " + f"datasets, and objects: {duration} seconds") + LOGGER.info("Groups, datasets, and objects have the same content.") + + # Compare datasets and objects. + start_time = time.perf_counter() + h5_diff_datasets = compare_datasets(file1, file2, datasets_f1) + duration = time.perf_counter() - start_time + LOGGER.info(f"Elapsed time comparing datasets: {duration} seconds") + start_time = time.perf_counter() + h5_diff_objects = compare_objects(file1, file2, objects_f1) + LOGGER.info(f"Elapsed time comparing objects: {duration} seconds") + + assert not h5_diff_datasets, f"{[d for d in h5_diff_datasets]} datasets contain different values for {test_file} and {reference_file}" # noqa + LOGGER.info("Datasets are validated.") + assert not h5_diff_objects, f"{[d for d in h5_diff_objects]} objects contain different values for {test_file} and {reference_file}" # noqa + LOGGER.info("Objects are validated.") + + +def validate_files( + ref_folder: pathlib.PosixPath, + out_folder: pathlib.PosixPath, + exclude_attrs: list, + test_file: pathlib.PosixPath, +) -> Tuple[bool, pathlib.PosixPath]: + """Validate file similarities. Create temporary files to exclude + h5 attributes known to be different. e.g `report` for constants. + If both files are not identical, the function is able to loop over + both files and find and fail on the difference. + + Args: + ref_folder: The reference folder for validating the files + out_folder: The output folder for the test constant files. + test_file: The output file to be validated. + exclude_attrs: A list of datasets, groups to exclude + from validated files. + Returns: + result: validation result for metadata. + test_file: The validated file. + """ + import h5py + start_validating = time.perf_counter() + + def exclude_sources(source_file, dest, excluded_sources): + # Open the source file in read-only mode + with h5py.File(source_file, 'r') as source: + + # Recursively visit all objects in the source file + def visit_func(name, obj): + # Check if the object should be excluded + if name in excluded_sources: + return + + # Check if the object is a dataset + if isinstance(obj, h5py.Dataset): + # Create a new dataset in the destination + # file and copy the data + dest.create_dataset(name, data=obj[()]) + + # Visit all objects in the source file and + # copy them to the destination file + source.visititems(visit_func) + + with tempfile.NamedTemporaryFile( + dir=out_folder, + suffix=".tmp", + prefix="cal_", + delete=True, + ) as out_tf, tempfile.NamedTemporaryFile( + dir=out_folder, + suffix=".tmp", + prefix="cal_", + delete=True, + ) as ref_tf: + + # Create in-memory HDF5 files for validation + with h5py.File(out_tf.name, 'a') as hp1, h5py.File(ref_tf.name, 'a') as hp2: # noqa + + start_time = time.perf_counter() + # Copy h5 files for validation and exclude selected attrs. + exclude_sources(test_file, hp1, exclude_attrs) + + duration = time.perf_counter() - start_time + LOGGER.info(f"Elapsed time copying {test_file}: " + f"{duration} seconds") + + start_time = time.perf_counter() + exclude_sources(ref_folder / test_file.name, hp2, exclude_attrs) + + duration = time.perf_counter() - start_time + LOGGER.info(f"Elapsed time copying {ref_folder / test_file.name}: " + f"{duration} seconds") + + start_time = time.perf_counter() + result = file_md5(out_tf.name) == file_md5(ref_tf.name) + LOGGER.info(f"MD5 validation for {test_file}: {duration} seconds") + duration = time.perf_counter() - start_validating + return result, test_file + + +def parse_config( + cmd: List[str], + config: Dict[str, Any], + out_folder: str +) -> List[str]: + """Convert a dictionary to a list of arguments. + + Values that are not strings will be cast. + Lists will be converted to several strings following their `--key` + flag. + Booleans will be converted to a `--key` flag, where `key` is the + dictionary key. + """ + + for key, value in config.items(): + if ' ' in key or (isinstance(value, str) and ' ' in value): + raise ValueError('Spaces are not allowed', key, value) + + if isinstance(value, list): + cmd.append(f"--{key}") + cmd += [str(v) for v in value] + elif isinstance(value, bool): + if value: + cmd += ["--{}".format(key)] + else: + if value in ['""', "''"]: + value = "" + if key == "out-folder": + value = out_folder + cmd += ["--{}".format(key), str(value)] + return cmd + + +def validate_hdf5_files( + test_key: str, + out_folder: pathlib.Path, + reference_folder: pathlib.Path, + cal_type: str, + find_difference: bool +): + """Apply HDF5 data validation. + + Args: + test_key (str): The test name. + out_folder (pathlib.Path): The OUT folder for the tested data. + reference_folder (pathlib.Path): The Reference folder for + the reference data to validate against + cal_type (str): The type of calibration processing. + e.g. dark or correct. + find_difference (bool): A flag indicating a need to find the + difference between two files if tested data was + not identical to the reference data. + """ + # 3rd Check number of produced h5 files. + h5files = list(out_folder.glob("*.h5")) + expected_h5files = list(reference_folder.glob("*.h5")) + assert len(h5files) == len(expected_h5files), f"{test_key} failure, number of files are not as expected." # noqa + LOGGER.info(f"{test_key}'s calibration h5files numbers are as expected.") + + non_valid_files = [] + # Hard coded datasets to exclude from numerical validation. + # These datasets are know to be updated everytime. + if cal_type.lower() == "correct": + exclude_attrs = ["METADATA/creationDate", "METADATA/updateDate"] + else: + exclude_attrs = ["report"] + + # 4th check that test and reference h5files are identical. + _validate_files = partial( + validate_files, + reference_folder, + out_folder, + exclude_attrs, + ) + with multiprocessing.pool.ThreadPool(processes=8) as executor: + result = executor.map(_validate_files, h5files) + + # Collect non-valid files, if any, to display them in the error message. + for valid, file in result: + if not valid: + non_valid_files.append(file) + + if len(non_valid_files) > 0: + if find_difference: + LOGGER.error(f"Found non valid files: {non_valid_files}. " + f"Checking differences for {non_valid_files[0]}") + find_differences( + non_valid_files[0], + reference_folder / non_valid_files[0].name, + exclude_attrs + ) + LOGGER.info(f"No difference found for {non_valid_files[0]}") + else: + assert len(non_valid_files) == 0, f"{test_key} failure, while validating metadata for {non_valid_files}" # noqa + LOGGER.info(f"{test_key}'s calibration h5files" + " are validated successfully.") + + +def slurm_watcher( + test_key: str, + std_out: str +): + """ + Watch for submitted slurm jobs and wait for them to finish. + After they finish apply first test and check + if they were `COMPLETED`, successfully. + + Args: + test_key (str): Test name. + out_str (str): xfel-calibrate CLU std output. + """ + slurm_watcher = True + + LOGGER.info(f"{test_key} - xfel-calibrate std out: {std_out}") + + for r in std_out.split("\n"): + if "Submitted the following SLURM jobs:" in r: + _, jobids = r.split(":") + + # Adding a sleep for the slurm jobs initialization + time.sleep(len(jobids.split(","))) + jobids = jobids.strip() + while slurm_watcher: + cmd = ["sacct", "-j", jobids, "--format=state"] + + res = run(cmd, stdout=PIPE) + states = res.stdout.decode().split("\n")[2:-1] + + if not any(s.strip() in [ + "COMPLETING", + "RUNNING", + "CONFIGURING", + "PENDING", + ] for s in states): + slurm_watcher = False + else: + time.sleep(2) + + # 1st check that all jobs were COMPLETED without errors. + states = res.stdout.decode().split("\n")[2:-1] + assert all(s.strip() == "COMPLETED" for s in states), f"{test_key} failure, calibration jobs were not completed. {jobids}: {states}" # noqa + LOGGER.info(f"{test_key}'s jobs were COMPLETED") + time.sleep(1.0) + + +@pytest.mark.manual_run +@pytest.mark.parametrize( + "test_key, val_dict", + list(automated_test_config.items()), + ids=list(automated_test_config.keys()), +) +def test_xfel_calibrate( + test_key: str, val_dict: dict, + release_test_config: Tuple[bool, bool, bool, bool] +): + """ Test xfel calibrate detectors and calibrations written + in the given callab_test YAML file. + Args: + test_key : Key for the xfel-calibrate test. + val_dict: Dictionary of the configurations for the running test. + release_test_config: Tuple of booleans to pick or skip tests + based on the given boolean configs. + """ + + ( + detectors, calibration, picked_test, + skip_numerical_validation, only_validate, find_difference, + use_slurm, reference_dir_base, out_dir_base, + ) = release_test_config + + cal_type = val_dict["cal_type"] + det_type = val_dict["det_type"] + + if not picked_test: + # Skip non-selected detectors + if ( + detectors != ["all"] and + det_type.lower() not in [d.lower() for d in detectors] + ): + pytest.skip() + + # Skip non-selected calibration + if calibration != "all" and cal_type.lower() != calibration: + pytest.skip() + else: + if test_key != picked_test: + pytest.skip() + + cmd = ["xfel-calibrate", det_type, cal_type] + + cal_conf = val_dict["config"] + + out_folder = pathlib.Path(cal_conf["out-folder"].format( + out_dir_base, cal_conf["karabo-id"], test_key)) + reference_folder = pathlib.Path(val_dict["reference-folder"].format( + reference_dir_base, cal_conf["karabo-id"], test_key)) + + report_name = ( + out_folder / + f"{test_key}_{datetime.now().strftime('%y%m%d_%H%M%S')}") + + cal_conf["report-to"] = str(report_name) + + cmd = parse_config(cmd, cal_conf, out_folder) + + if only_validate: + validate_hdf5_files( + test_key, + out_folder, + reference_folder, + cal_type, + find_difference, + ) + return + + if not use_slurm: # e.g. for Gitlab CI. + cmd += ["--no-cluster-job"] + + cmd += ["--slurm-name", test_key, "--cal-db-interface", "tcp://max-exfl-cal001:8015#8045"] + f = io.StringIO() + LOGGER.info(f"Submitting CL: {cmd}") + with redirect_stdout(f): + errors = calibrate.run(cmd) + out_str = f.getvalue() + + if use_slurm: + slurm_watcher(test_key, out_str) + else: + # confirm that all jobs succeeded. + assert errors == 0 + + # 2nd check for report availability. + report_file = out_folder / f"{report_name}.pdf" + assert report_file.exists(), f"{test_key} failure, report doesn't exists." + LOGGER.info("Report found.") + + # Stop tests at this point, if desired. + if not skip_numerical_validation: + validate_hdf5_files( + test_key, + out_folder, + reference_folder, + cal_type, + find_difference, + )