diff --git a/docs/src/Testing.rst b/docs/src/Testing.rst index 4371ee956..ad67326d8 100644 --- a/docs/src/Testing.rst +++ b/docs/src/Testing.rst @@ -101,7 +101,7 @@ To run only tests requiring virtual detectors use the following command: #in build python -m pytest -m detectorintegration ../python/tests/ -There is a helper test fixture in ``slsDetectorSoftware/python/tests/conftest.py`` called ``test_with_simulators`` that sets up virtual detectors and yields the test for all detectors. The set up is done for every test automatically. +There is a helper test fixture in ``slsDetectorSoftware/python/tests/conftest.py`` called ``session_simulator`` that sets up virtual detectors and yields the test for all detectors. The set up is done for every test automatically. Note that the fixture persist over the entire session e.g. the fixture is setup one detector at a time and runs all tests using this fixture before cleaning up and moving on to the next detector. It saves time if the setup and cleanup is expensive. Example usage: @@ -110,10 +110,13 @@ Example usage: import pytest @pytest.mark.detectorintegration - def test_example_with_simulator(test_with_simulators): + def test_example_with_simulator(session_simulator): # your test code here -If you want to run the test only for a specific test use the parametrized test fixture: +.. Note:: + As the detector is set up only once makes sure to not change the state of the detector in a way that affects other tests. If you want to change the state of the detector make sure to reset it at the end of your test. + +If you want to run the test only for a specific detector use the parametrized test fixture: Example usage: @@ -122,45 +125,10 @@ Example usage: import pytest @pytest.mark.detectorintegration - @pytest.mark.parametrize("setup_parameters", [([""], )], indirect=True) - def test_example_with_specific_simulators(test_with_simulators, setup_parameters): + @pytest.mark.parametrize("session_simulator", [("", , ), ("", , )], indirect=True) + def test_example_with_specific_simulators(session_simulator): # your test code here +.. Note:: + The parametrized test fixture is setup per file and not for the entire session. -There is another helper test fixture in ``slsDetectorSoftware/python/tests/conftest.py`` called ``session_simulator`` that sets up virtual detectors and yields the test for all detectors. The difference with the previous fixture ``test_with_simulators`` is that this fixture will set up one detector at a time and run all the tests using this fixture before cleaning up and moving on to the next detector. It saves time if the setup and cleanup is expensive. - -Example usage: - -.. code-block:: python - - import pytest - - @pytest.mark.detectorintegration - def test_define_reg(session_simulator, request): - """ Test setting define_reg for ctb and xilinx_ctb.""" - det_type, num_interfaces, num_mods, d = session_simulator - assert d is not None - - from slsdet import RegisterAddress - - if det_type in ['ctb', 'xilinx_ctb']: - # your test code here - -For more specific parameters, you can parametrize the fixture like below: - -.. code-block:: python - - import pytest - - @pytest.mark.detectorintegration - @pytest.mark.parametrize( - "session_simulator", - [ - ("ctb", 1, 1), - ("xilinx_ctb", 1, 1), - ], - indirect=True, - ) - def test_define_reg(session_simulator): - det_type, num_interfaces, num_mods, d = session_simulator - # your test code here diff --git a/python/tests/conftest.py b/python/tests/conftest.py index 3b1d65f69..e39c6cf20 100644 --- a/python/tests/conftest.py +++ b/python/tests/conftest.py @@ -58,39 +58,56 @@ DEFAULT_SIMULATOR_CONFIGS = [ SIMULATOR_IDS = [f"{det_type}_{num_interface}if_{num_mod}mod" for det_type, num_interface, num_mod in DEFAULT_SIMULATOR_CONFIGS] +''' +for more specific parameters +@pytest.mark.detectorintegration +@pytest.mark.parametrize( + "session_simulator", + [ + ("ctb", 1, 1), + ("xilinx_ctb", 1, 1), + ], + indirect=True, +) +def test_define_reg(session_simulator): + det_type, num_interfaces, num_mods, d = session_simulator +''' @pytest.fixture(scope="session") def session_simulator(request): """ Fixture to start the detector server once and clean up at the end. Expects request.param = (det_type, num_interfaces, num_mods) """ - det_type, num_interfaces, num_mods = request.param - fp = sys.stdout + try: + det_type, num_interfaces, num_mods = request.param + fp = sys.stdout - # set up: once per server - Log(LogLevel.INFOBLUE, - f'---- {det_type} | interfaces={num_interfaces} | modules={num_mods} ----', fp) + # set up: once per server + Log(LogLevel.INFOBLUE, + f'---- {det_type} | interfaces={num_interfaces} | modules={num_mods} ----', fp) - cleanup(fp) - startDetectorVirtualServer(det_type, num_mods, fp, True) - startReceiver(num_mods, fp, True) + cleanup(fp) + startDetectorVirtualServer(det_type, num_mods, fp, True, True) + startReceiver(num_mods, fp, True) - Log(LogLevel.INFOBLUE, f'Waiting for server to start up and connect', fp) - d = loadConfig( - name=det_type, - log_file_fp=fp, - num_mods=num_mods, - num_frames=1, - num_interfaces=num_interfaces, - ) + Log(LogLevel.INFOBLUE, f'Waiting for server to start up and connect', fp) + d = loadConfig( + name=det_type, + log_file_fp=fp, + num_mods=num_mods, + num_frames=1, + num_interfaces=num_interfaces, + ) - loadBasicSettings(name=det_type, d=d, fp=fp) + loadBasicSettings(name=det_type, d=d, fp=fp) - yield det_type, num_interfaces, num_mods, d - - cleanup(fp) + yield det_type, num_interfaces, num_mods, d + cleanup(fp) + except Exception as e: + Log(LogLevel.ERROR, f'Tests Failed.', fp) + cleanup(fp) def pytest_generate_tests(metafunc): if "session_simulator" not in metafunc.fixturenames: @@ -110,62 +127,4 @@ def pytest_generate_tests(metafunc): indirect=True ) -''' -for more specific parameters -@pytest.mark.detectorintegration -@pytest.mark.parametrize( - "session_simulator", - [ - ("ctb", 1, 1), - ("xilinx_ctb", 1, 1), - ], - indirect=True, -) -def test_define_reg(session_simulator): - det_type, num_interfaces, num_mods, d = session_simulator -''' - -#helper fixture for servers -@pytest.fixture(scope='module') -def setup_parameters(request): # only setup once per module if same parameters used for the scopes - try: - servers, nmods = request.param # comes from @pytest.mark.parametrize(..., indirect=True) - return servers, nmods - except AttributeError: - # fallback default if the test did not parametrize - return (['eiger', 'jungfrau', 'mythen3', 'gotthard2', 'ctb', 'moench', 'xilinx_ctb'], 2) - -@pytest.fixture(scope='module') -def test_with_simulators(setup_parameters): - """ Fixture to automatically setup virtual detector servers for testing. """ - - fp = sys.stdout - - servers, nmods = setup_parameters - print("servers:", servers) - print("nmods:", nmods) - try: - for server in servers: - for ninterfaces in range(1,2): - if ninterfaces == 2 and server != 'jungfrau' and server != 'moench': - continue - - msg = f'Starting Python API Tests for {server}' - - if server == 'jungfrau' or server == 'moench': - msg += f' with {ninterfaces} interfaces' - - Log(LogLevel.INFOBLUE, msg, fp) - cleanup(fp) - startDetectorVirtualServer(server, nmods, fp) - startReceiver(nmods, fp) - d = loadConfig(name=server, log_file_fp=fp, num_mods=nmods, num_frames=1, num_interfaces=ninterfaces) - #loadBasicSettings(name=server, d=d, fp=fp) - yield # run test - cleanup(fp) # teardown - except Exception as e: - traceback.print_exc(file=fp) - Log(LogLevel.ERROR, f'Tests Failed.', fp) - cleanup(fp) - diff --git a/python/tests/test_ROI.py b/python/tests/test_ROI.py new file mode 100644 index 000000000..f6117f90d --- /dev/null +++ b/python/tests/test_ROI.py @@ -0,0 +1,59 @@ +import pytest +import sys + +from conftest import session_simulator + +from slsdet import Detector + +from slsdet._slsdet import slsDetectorDefs + +detectorType = slsDetectorDefs.detectorType + +@pytest.mark.detectorintegration +def test_rx_ROI(session_simulator): + """ Test rx_ROI property of Detector class. """ + det_type, num_interfaces, num_mods, d = session_simulator + assert d is not None + + if d.type == detectorType.CHIPTESTBOARD or d.type == detectorType.XILINX_CHIPTESTBOARD: + pytest.skip("Skipping ROI test for ctb/xilinx_ctb detector types.") + + if(d.type == detectorType.MYTHEN3 or d.type == detectorType.GOTTHARD2): + d.rx_roi = (0, 10) + roi = d.rx_roi + assert roi == [(0, 10, -1, -1)] + + #d.rx_roi = [[5,15, 0, 1]] # not allowed for mythen3 + + d.rx_roi = [0,10, -1, -1] + + assert d.rx_roi == [(0,10,-1,-1)] + d.rx_clearroi() + else: + + d.rx_roi = (0, 10, 10, 20) + roi = d.rx_roi + assert roi == [(0, 10, 10, 20)] + + d.rx_roi = [5,15,15,25] + + assert d.rx_roi == [(5,15,15,25)] + + if d.nmod > 1 and (d.type != detectorType.JUNGFRAU) or (d.numinterfaces == 2 and d.type != detectorType.EIGER): + d.rx_roi = [[0,10,0,20], [5,20,410,420]] + + roi = d.rx_roi + assert roi == [(0,10,0,20), (5,20,410,420)] #in same file for jungfrau + + d.rx_clearroi() + roi = d.rx_roi + assert roi == [(-1,-1,-1,-1)] + + + + + + + + + diff --git a/python/tests/test_det_api.py b/python/tests/test_det_api.py index 7bf942e22..5c8cef0bc 100644 --- a/python/tests/test_det_api.py +++ b/python/tests/test_det_api.py @@ -12,6 +12,9 @@ from utils_for_test import ( ) from slsdet import Detector +from slsdet._slsdet import slsDetectorDefs + +detectorType = slsDetectorDefs.detectorType @pytest.mark.detectorintegration @@ -903,3 +906,17 @@ def test_dac(session_simulator, request): Log(LogLevel.INFOGREEN, f"✅ {request.node.name} passed") +@pytest.mark.detectorintegration +@pytest.mark.parametrize("session_simulator",[("moench", 1, 2)],indirect=True) +def test_type(session_simulator): + + d = Detector() + assert d.type == detectorType.MOENCH + + +@pytest.mark.detectorintegration +@pytest.mark.parametrize("session_simulator",[("moench", 1, 2), ("jungfrau", 1, 2)],indirect=True) +def test_numinterfaces(session_simulator): + + d = Detector() + assert d.numinterfaces == 1 \ No newline at end of file diff --git a/python/tests/test_free.py b/python/tests/test_free.py index 4f5393ca9..91e959ef4 100644 --- a/python/tests/test_free.py +++ b/python/tests/test_free.py @@ -17,46 +17,18 @@ sys.path.append(str(scripts_dir)) from slsdet import Detector, Ctb, freeSharedMemory + from utils_for_test import ( Log, LogLevel, - cleanup, - startDetectorVirtualServer, - connectToVirtualServers, SERVER_START_PORTNO ) -''' -scope = module =>Once per test file/module -to share expensive setup like startDetectorVirtualServer -''' -@pytest.fixture(scope="module") -def det_config(): - return { - "name": "ctb", - "num_mods": 1 - } - -@pytest.fixture(scope="module", autouse=True) -def setup_simulator(det_config): - """Fixture to start the detector server once and clean up at the end.""" - fp = sys.stdout - - cleanup(fp) - startDetectorVirtualServer(det_config["name"], det_config["num_mods"], fp) - - Log(LogLevel.INFOBLUE, f'Waiting for server to start up and connect') - connectToVirtualServers(det_config["name"], det_config["num_mods"]) - Log(LogLevel.INFOBLUE, f'Freeing shm before tests') - freeSharedMemory() - - yield # tests run here - - cleanup(fp) - +from conftest import session_simulator @pytest.mark.detectorintegration -def test_exptime_after_free_should_raise(setup_simulator): +@pytest.mark.parametrize("session_simulator",[("ctb", 1, 1)],indirect=True) +def test_exptime_after_free_should_raise(session_simulator): Log(LogLevel.INFOBLUE, f'\nRunning test_exptime_after_free_should_raise') @@ -78,7 +50,8 @@ def free_and_create_shm(): k.hostname = f"localhost:{SERVER_START_PORTNO}" # free and recreate shm, maps to local shm struct @pytest.mark.detectorintegration -def test_exptime_after_not_passing_var_should_raise(setup_simulator): +@pytest.mark.parametrize("session_simulator",[("ctb", 1, 1)],indirect=True) +def test_exptime_after_not_passing_var_should_raise(session_simulator): Log(LogLevel.INFOBLUE, f'\nRunning test_exptime_after_not_passing_var_should_raise') @@ -102,7 +75,8 @@ def free_and_create_shm_passing_ctb_var(k): k.hostname = f"localhost:{SERVER_START_PORTNO}" # free and recreate shm, maps to local shm struct @pytest.mark.detectorintegration -def test_exptime_after_passing_ctb_var_should_raise(setup_simulator): +@pytest.mark.parametrize("session_simulator",[("ctb", 1, 1)],indirect=True) +def test_exptime_after_passing_ctb_var_should_raise(session_simulator): Log(LogLevel.INFOBLUE, f'\nRunning test_exptime_after_passing_ctb_var_should_raise') d = Ctb() # creates multi shm (assuming no shm exists) @@ -125,7 +99,8 @@ def free_and_create_shm_returning_ctb(): return k @pytest.mark.detectorintegration -def test_exptime_after_returning_ctb_should_raise(setup_simulator): +@pytest.mark.parametrize("session_simulator",[("ctb", 1, 1)],indirect=True) +def test_exptime_after_returning_ctb_should_raise(session_simulator): Log(LogLevel.INFOBLUE, f'\nRunning test_exptime_after_returning_ctb_should_raise') d = Ctb() # creates multi shm (assuming no shm exists) @@ -148,7 +123,8 @@ def test_exptime_after_returning_ctb_should_raise(setup_simulator): assert str(exc_info.value) == "Shared memory is invalid or freed. Close resources before access." @pytest.mark.detectorintegration -def test_hostname_twice_acess_old_should_raise(setup_simulator): +@pytest.mark.parametrize("session_simulator",[("ctb", 1, 1)],indirect=True) +def test_hostname_twice_acess_old_should_raise(session_simulator): Log(LogLevel.INFOBLUE, f'\nRunning test_hostname_twice_acess_old_should_raise') d = Ctb() # creates multi shm (assuming no shm exists) diff --git a/python/tests/test_pythonAPI.py b/python/tests/test_pythonAPI.py deleted file mode 100644 index 9bcc4214b..000000000 --- a/python/tests/test_pythonAPI.py +++ /dev/null @@ -1,51 +0,0 @@ -import pytest -import sys - -from conftest import test_with_simulators - -from slsdet import Detector - -from utils_for_test import ( - Log, - LogLevel, -) - -@pytest.mark.detectorintegration -@pytest.mark.parametrize("setup_parameters", [(["moench"], 2)], indirect=True) -def test_rx_ROI_moench(test_with_simulators, setup_parameters): - """ Test setting and getting rx_ROI property of Detector class for moench. """ - - d = Detector() - d.rx_roi = (0, 10, 10, 20) - roi = d.rx_roi - assert roi == [(0, 10, 10, 20)] - - d.rx_roi = [5,15,15,25] - - assert d.rx_roi == [(5,15,15,25)] - - d.rx_roi = [[0,10,0,20], [5,20,410,420]] - - roi = d.rx_roi - assert roi == [(0,10,0,20), (5,20,410,420)] - - d.rx_clearroi() - roi = d.rx_roi - assert roi == [(-1,-1,-1,-1)] - -@pytest.mark.detectorintegration -@pytest.mark.parametrize("setup_parameters", [(["mythen3"], 1)], indirect=True) -def test_rx_ROI_mythen(test_with_simulators, setup_parameters): - """ Test setting and getting rx_ROI property of Detector class for mythen. """ - - d = Detector() - d.rx_roi = (0, 10) - roi = d.rx_roi - assert roi == [(0, 10, -1, -1)] - - #d.rx_roi = [[5,15, 0, 1]] # not allowed for mythen3 - - d.rx_roi = [0,10, -1, -1] - - assert d.rx_roi == [(0,10,-1,-1)] - diff --git a/tests/scripts/utils_for_test.py b/tests/scripts/utils_for_test.py index ae8930860..14c90e196 100644 --- a/tests/scripts/utils_for_test.py +++ b/tests/scripts/utils_for_test.py @@ -257,7 +257,7 @@ def connectToVirtualServers(name, num_mods, ctb_object=False): counts_sec = 5 while (counts_sec != 0): try: - d.virtual = [num_mods, SERVER_START_PORTNO] + d.virtual = [num_mods, SERVER_START_PORTNO] # sets the hostnames break except Exception as e: # stop server still not up, wait a bit longer @@ -288,13 +288,16 @@ def startReceiver(num_mods, fp, no_log_file = False, quiet_mode=False): def loadConfig(name, rx_hostname = 'localhost', settingsdir = None, log_file_fp = None, num_mods = 1, num_frames = 1, num_interfaces = 1): Log(LogLevel.INFO, 'Loading config', log_file_fp, True) try: - d = connectToVirtualServers(name, num_mods) + if name == 'ctb' or name == 'xilinx_ctb': + d = connectToVirtualServers(name, num_mods, ctb_object=True) + else: + d = connectToVirtualServers(name, num_mods) if name == 'jungfrau' or name == 'moench': d.numinterfaces = num_interfaces d.udp_dstport = DEFAULT_UDP_DST_PORTNO - if name == 'eiger' or num_interfaces == 2: + if d.numinterfaces == 2: d.udp_dstport2 = DEFAULT_UDP_DST_PORTNO + 1 d.rx_hostname = rx_hostname