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,
+        )