Skip to content

Commit

Permalink
Merge pull request #467 from ISISComputingGroup/Ticket6556_move_emula…
Browse files Browse the repository at this point in the history
…tor_and_tests

Ticket 6556: Move emulator and tests
  • Loading branch information
aaron-long authored Jul 13, 2021
2 parents 1fa0841 + 2e83b58 commit 0db3020
Show file tree
Hide file tree
Showing 7 changed files with 56 additions and 157 deletions.
2 changes: 2 additions & 0 deletions Makefile
Original file line number Diff line number Diff line change
@@ -0,0 +1,2 @@
ioctests:
$(PYTHON3) run_tests.py
13 changes: 11 additions & 2 deletions README.md
Original file line number Diff line number Diff line change
Expand Up @@ -13,7 +13,7 @@ It recommended that you don't have the server side of IBEX running when testing

### Running all tests

To run all the tests in the test framework, use:
To run all the tests that are bundled in the test framework, use:

```
C:\Instrument\Apps\EPICS\config_env.bat
Expand All @@ -23,7 +23,9 @@ Then `cd` to `C:\Instrument\Apps\EPICS\support\IocTestFramework\master` and use:
python run_tests.py
```

There is a batch file which does this for you, called `run_all_tests.bat`
There is a batch file which does this for you, called `run_all_tests.bat`. Or, if you are already in an EPICS
terminal you can call `make ioctests` (note that if your MAKEFLAGS env variable contains `-Otarget` you will
not see the test output until all tests are complete.

### Running tests in modules

Expand Down Expand Up @@ -83,6 +85,13 @@ Sometimes you might want to run all the tests only in RECSIM or only in DEVSIM.

> `python run_tests.py -tm RECSIM`
### Run test and emulator from specific directory

For newer IOCs the emulator and tests live in the support folder of the IOC. To specify this to the test framework
you can use:

> `python run_tests.py --test_and_emulator C:\Instrument\Apps\EPICS\support\CCD100\master\system_tests`
## Troubleshooting

If all tests are failing then it is likely that the PV prefix is incorrect.
Expand Down
50 changes: 25 additions & 25 deletions run_tests.py
Original file line number Diff line number Diff line change
Expand Up @@ -8,6 +8,7 @@
import traceback
import unittest
from typing import List, Any
import importlib

import six
import xmlrunner
Expand All @@ -17,7 +18,8 @@
from run_utils import ModuleTests

from utils.device_launcher import device_launcher, device_collection_launcher
from utils.emulator_launcher import LewisLauncher, NullEmulatorLauncher, MultiLewisLauncher, Emulator, TestEmulatorData
from utils.emulator_launcher import LewisLauncher, NullEmulatorLauncher, MultiLewisLauncher, Emulator, TestEmulatorData, \
DEVICE_EMULATOR_PATH
from utils.ioc_launcher import IocLauncher, EPICS_TOP, IOCS_DIR
from utils.free_ports import get_free_ports
from utils.test_modes import TestModes
Expand Down Expand Up @@ -98,10 +100,11 @@ def make_device_launchers_from_module(test_module, mode):

if "emulator" in ioc and mode != TestModes.RECSIM:
emulator_launcher_class = ioc.get("emulator_launcher_class", LewisLauncher)
emulator_launcher = emulator_launcher_class(test_module.__name__, ioc["emulator"], var_dir,
emulator_launcher = emulator_launcher_class(test_module.__name__, ioc["emulator"], emulator_path, var_dir,
emmulator_port, ioc)
elif "emulator" in ioc:
emulator_launcher = NullEmulatorLauncher(test_module.__name__, ioc["emulator"], var_dir, None, ioc)
emulator_launcher = NullEmulatorLauncher(test_module.__name__, ioc["emulator"], emulator_path, var_dir,
None, ioc)
elif "emulators" in ioc and mode != TestModes.RECSIM:
emulator_launcher_class = ioc.get("emulators_launcher_class", MultiLewisLauncher)
test_emulator_data: List[TestEmulatorData] = ioc.get("emulators", [])
Expand Down Expand Up @@ -273,7 +276,7 @@ def run_tests(prefix, module_name, tests_to_run, device_launchers, failfast_swit
'EPICS_CA_ADDR_LIST': "127.255.255.255"
}

test_names = ["{}.{}".format(arguments.tests_path, test) for test in tests_to_run]
test_names = [f"tests.{test}" for test in tests_to_run]

runner = xmlrunner.XMLTestRunner(output='test-reports', stream=sys.stdout, failfast=failfast_switch)
test_suite = unittest.TestLoader().loadTestsFromNames(test_names)
Expand All @@ -294,13 +297,6 @@ def run_tests(prefix, module_name, tests_to_run, device_launchers, failfast_swit
print("IOC system tests should now be run under python 3. Aborting.")
sys.exit(-1)

pythondir = os.environ.get("PYTHON3DIR", None)

if pythondir is not None:
emulator_path = os.path.join(pythondir, "scripts")
else:
emulator_path = None

parser = argparse.ArgumentParser(
description='Test an IOC under emulation by running tests against it')
parser.add_argument('-l', '--list-devices',
Expand All @@ -309,10 +305,6 @@ def run_tests(prefix, module_name, tests_to_run, device_launchers, failfast_swit
help='Report devices that have not been tested.', action="store_true")
parser.add_argument('-pf', '--prefix', default=os.environ.get("MYPVPREFIX", None),
help='The instrument prefix; e.g. TE:NDW1373')
parser.add_argument('-e', '--emulator-path', default=emulator_path,
help="The path of the lewis.py file")
parser.add_argument('-py', '--python-path', default="C:\Instrument\Apps\Python3\python.exe",
help="The path of python.exe")
parser.add_argument('--var-dir', default=None,
help="Directory in which to create a log dir to write log file to and directory in which to "
"create tmp dir which contains environments variables for the IOC. Defaults to "
Expand All @@ -322,7 +314,7 @@ def run_tests(prefix, module_name, tests_to_run, device_launchers, failfast_swit
Module just runs the tests in a module.
Module.class runs the the test class in Module.
Module.class.method runs a specific test.""")
parser.add_argument('-tp', '--tests-path', default="tests",
parser.add_argument('-tp', '--tests-path', default=f"{os.path.dirname(os.path.realpath(__file__))}\\tests",
help="""Path to find the tests in, this must be a valid python module.
Default is in the tests folder of this repo""")
parser.add_argument('-f', '--failfast', action='store_true',
Expand All @@ -332,17 +324,29 @@ def run_tests(prefix, module_name, tests_to_run, device_launchers, failfast_swit
emulator/IOC or attach debugger for tests""")
parser.add_argument('-tm', '--tests-mode', default=None, choices=['DEVSIM', 'RECSIM'],
help="""Tests mode to run e.g. DEVSIM or RECSIM (default: both).""")
parser.add_argument('--test_and_emulator', default=None,
help="""Specify a folder that holds both the tests (in a folder called tests) and a lewis
emulator (in a folder called lewis_emulators).""")

arguments = parser.parse_args()

if arguments.test_and_emulator:
arguments.tests_path = os.path.join(arguments.test_and_emulator, "tests")
emulator_path = arguments.test_and_emulator
else:
emulator_path = DEVICE_EMULATOR_PATH

if os.path.dirname(arguments.tests_path):
full_path = os.path.abspath(arguments.tests_path)
if not os.path.isdir(full_path):
print("Test path {} not found".format(full_path))
init_file_path = os.path.join(full_path, "__init__.py")
if not os.path.isfile(init_file_path):
print(f"Test path {full_path} not found")
sys.exit(-1)
tests_module_path = os.path.dirname(full_path)
sys.path.insert(0, tests_module_path)
arguments.tests_path = os.path.basename(arguments.tests_path)
# Import the specified path as the tests module
spec = importlib.util.spec_from_file_location("tests", init_file_path)
module = importlib.util.module_from_spec(spec)
sys.modules[spec.name] = module
spec.loader.exec_module(module)

if arguments.list_devices:
print("Available tests:")
Expand All @@ -356,10 +360,6 @@ def run_tests(prefix, module_name, tests_to_run, device_launchers, failfast_swit
print("Cannot run without instrument prefix, you may need to run this using an EPICS terminal")
sys.exit(-1)

if arguments.emulator_path is None:
print("Cannot run without emulator path, you may need to run this using an EPICS terminal")
sys.exit(-1)

tests = arguments.tests if arguments.tests is not None else package_contents(arguments.tests_path)
failfast = arguments.failfast
report_coverage = arguments.report_coverage
Expand Down
9 changes: 4 additions & 5 deletions run_utils.py
Original file line number Diff line number Diff line change
Expand Up @@ -3,15 +3,14 @@
from contextlib import contextmanager


def package_contents(package_name):
def package_contents(package_path):
"""
Finds all the modules in a package.
Finds all the files in a package.
:param package_name: the name of the package
:param package_path: the name of the package
:return: a set containing all the module names
"""
pathname = importlib.util.find_spec(package_name).name
return set([os.path.splitext(module)[0] for module in os.listdir(pathname)
return set([os.path.splitext(module)[0] for module in os.listdir(package_path)
if module.endswith('.py') and not module.startswith("__init__")])


Expand Down
113 changes: 0 additions & 113 deletions tests/ccd100.py

This file was deleted.

3 changes: 1 addition & 2 deletions utils/channel_access.py
Original file line number Diff line number Diff line change
Expand Up @@ -242,9 +242,8 @@ def _wrapper(message):
if message is None:
message = "Expected function '{}' to evaluate to True when reading PV '{}'." \
.format(func.__name__, self.create_pv_with_prefix(pv))

err = self._wait_for_pv_lambda(partial(_wrapper, message), timeout)

if err is not None:
raise AssertionError(err)

Expand Down
23 changes: 13 additions & 10 deletions utils/emulator_launcher.py
Original file line number Diff line number Diff line change
Expand Up @@ -64,11 +64,12 @@ def remove_emulator(cls, name):
@six.add_metaclass(abc.ABCMeta)
class EmulatorLauncher(object):

def __init__(self, test_name, device, var_dir, port, options):
def __init__(self, test_name, device, emulator_path, var_dir, port, options):
"""
Args:
test_name: The name of the test we are creating a device emulator for
device: The name of the device to emulate
emulator_path: The path where the emulator can be found
var_dir: The directory in which to store logs
port: The TCP port to listen on for connections
options: Dictionary of any additional options required by specific launchers
Expand All @@ -79,6 +80,7 @@ def __init__(self, test_name, device, var_dir, port, options):
self._port = port
self._options = options
self._test_name = test_name
self._emulator_path = emulator_path

def __enter__(self):
self._open()
Expand Down Expand Up @@ -354,22 +356,23 @@ class LewisLauncher(EmulatorLauncher):

_DEFAULT_LEWIS_PATH = os.path.join(DEFAULT_PY_PATH, "scripts")

def __init__(self, test_name, device, var_dir, port, options):
def __init__(self, test_name, device, emulator_path, var_dir, port, options):
"""
Constructor that also launches Lewis.
Args:
test_name: name of test we are creating device emulator for
device: device to start
emulator_path: The path where the emulator can be found
var_dir: location of directory to write log file and macros directories
port: the port to use
"""
super(LewisLauncher, self).__init__(test_name, device, var_dir, port, options)
super(LewisLauncher, self).__init__(test_name, device, emulator_path, var_dir, port, options)

self._lewis_path = options.get("lewis_path", LewisLauncher._DEFAULT_LEWIS_PATH)
self._python_path = options.get("python_path", os.path.join(DEFAULT_PY_PATH, "python.exe"))
self._lewis_protocol = options.get("lewis_protocol", "stream")
self._lewis_additional_path = options.get("lewis_additional_path", DEVICE_EMULATOR_PATH)
self._lewis_additional_path = options.get("lewis_additional_path", emulator_path)
self._lewis_package = options.get("lewis_package", "lewis_emulators")
self._default_timeout = options.get("default_timeout", 5)
self._speed = options.get("speed", 100)
Expand Down Expand Up @@ -624,8 +627,8 @@ def backdoor_run_function_on_device(self, launcher_address, function_name, argum

class CommandLineEmulatorLauncher(EmulatorLauncher):

def __init__(self, test_name, device, var_dir, port, options):
super(CommandLineEmulatorLauncher, self).__init__(test_name, device, var_dir, port, options)
def __init__(self, test_name, device, emulator_path, var_dir, port, options):
super(CommandLineEmulatorLauncher, self).__init__(test_name, device, emulator_path, var_dir, port, options)
try:
self.command_line = options["emulator_command_line"]
except KeyError:
Expand Down Expand Up @@ -677,7 +680,7 @@ def backdoor_run_function_on_device(self, *args, **kwargs):

class BeckhoffEmulatorLauncher(CommandLineEmulatorLauncher):

def __init__(self, test_name, device, var_dir, port, options):
def __init__(self, test_name, device, emulator_path, var_dir, port, options):
try:
self.beckhoff_root = options["beckhoff_root"]
self.solution_path = options["solution_path"]
Expand All @@ -695,20 +698,20 @@ def __init__(self, test_name, device, var_dir, port, options):

options["emulator_command_line"] = self.startup_command
options["emulator_wait_to_finish"] = True
super(BeckhoffEmulatorLauncher, self).__init__(test_name, device, var_dir, port, options)
super(BeckhoffEmulatorLauncher, self).__init__(test_name, device, emulator_path, var_dir, port, options)
else:
raise IOError("Unable to find AutomationTools.exe. Hint: You must build the solution located at:"
" {} \n".format(automation_tools_dir))


class DAQMxEmulatorLauncher(CommandLineEmulatorLauncher):
def __init__(self, test_name, device, var_dir, port, options):
def __init__(self, test_name, device, emulator_path, var_dir, port, options):
labview_scripts_dir = os.path.join(DEVICE_EMULATOR_PATH, "other_emulators", "DAQmx")
self.start_command = os.path.join(labview_scripts_dir, "start_sim.bat")
self.stop_command = os.path.join(labview_scripts_dir, "stop_sim.bat")
options["emulator_command_line"] = self.start_command
options["emulator_wait_to_finish"] = True
super(DAQMxEmulatorLauncher, self).__init__(test_name, device, var_dir, port, options)
super(DAQMxEmulatorLauncher, self).__init__(test_name, device, emulator_path, var_dir, port, options)

def _close(self):
self.disconnect_device()
Expand Down

0 comments on commit 0db3020

Please sign in to comment.