Automated Testing
Testing Framework
Python RIFT includes a very extensive framework for fully automated testing.
There are three types of tests:
-
Unit tests, which test the functionality of an individual Python module.
-
System tests, which test the RIFT protocol behavior for a group of RIFT nodes in a topology.
-
Interoperability tests, which test the interoperability between Python RIFT and other RIFT implementations (currently only Juniper RIFT).
Furthermore, Python RIFT uses pylint to check the Python code for issues.
All tests can be run manually; details are provided below.
Travis Continuous Integration (CI) is used to also run pylint, all unit tests, and all system tests automatically on each github commit.
During all of the tests, the –cov option in pytest is used to measure code coverage. The code coverage results are collected and graphically reported in codecov.
Pre-Commit Check
It is strongly recommended to run tools/pre-commit-checks before every commit to make sure that pylint, the unit tests, and the system tests will pass. Look for the All good; you can commit.
message at the end.
(env) $ tools/pre-commit-checks -------------------------------------------------------------------- Your code has been rated at 10.00/10 (previous run: 10.00/10, +0.00) -------------------------------------------------------------------- Your code has been rated at 10.00/10 (previous run: 10.00/10, +0.00) ======================================================= test session starts ======================================================== platform darwin -- Python 3.5.1, pytest-3.6.4, py-1.5.4, pluggy-0.7.1 rootdir: /Users/brunorijsman/rift-python, inifile: plugins: cov-2.5.1 collected 13 items tests/test_fsm.py ... [ 23%] tests/test_sys_2n_l0_l1.py . [ 30%] tests/test_sys_2n_l0_l2.py . [ 38%] tests/test_sys_2n_l1_l3.py . [ 46%] tests/test_sys_2n_un_l1.py . [ 53%] tests/test_sys_3n_l0_l1_l2.py . [ 61%] tests/test_sys_cli_commands.py . [ 69%] tests/test_table.py .... [100%] ---------- coverage: platform darwin, python 3.5.1-final-0 ----------- Name Stmts Miss Cover ---------------------------------------------------- rift/__main__.py 56 10 82% rift/cli_listen_handler.py 28 19 32% [...] tests/test_sys_cli_commands.py 96 0 100% tests/test_table.py 32 0 100% ---------------------------------------------------- TOTAL 2609 354 86% ==================================================== 13 passed in 99.27 seconds ==================================================== All good; you can commit.
This currently takes about 4 minutes to complete, but that time will grow as the number of system tests increases.
Note: when running the test suite on macOS, make sure that IPv6 link-local addresses are enabled on interface en0. To enable these, go to system preferences, select network, select Wifi advanced, select TCP/IP, and set configure IPv6 to link-local only or automatically.
Pylint
Python RIFT uses pylint to check all production code and all testing code for issues.
If you followed the installation instructions, you will have already installed pylint. If not, use pip to install it:
$ pip install pylint
To lint your code manually, use the following command. In this example we check the rift directory (where I have purposely introduce a warning). You also need to check the tests directory and the tools directory.
(env) $ pylint rift ************* Module rift.config rift/config.py:265:34: C0303: Trailing whitespace (trailing-whitespace) ------------------------------------------------------------------- Your code has been rated at 9.99/10 (previous run: 10.00/10, -0.01)
A perfect score (which is what you needs) looks like this:
(env) $ pylint rift -------------------------------------------------------------------- Your code has been rated at 10.00/10 (previous run: 10.00/10, +0.00)
Many Python code editors (I use Visual Studio Code) have a plugin to automatically detect and report pylint warning as you enter the code. I strongly recommend using such a plugin.
Python RIFT is very pedantic about not allowing any pylint issues. Pylint is run on every github commit, and even a single warning will cause a build failure to be declared.
The pylintrc file in the home directory disables a small number of pylint warnings, and documents why they are disabled. Furthermore, a few Python files contain # pylint: disable=...
comments to disable specific warnings, but this should be used very sparingly.
Unit Tests
The unit tests use pytest with code-coverage extensions to test an individual Python module.
If you followed the installation instructions, you will have already installed pytest-cov. If not, use pip to install it:
$ pip install pytest-cov
The unit tests are stored in the tests directory, and start with the test_ prefix (but not the test_sys_ prefix, those are system tests).
Use the following command to run an individual unit test:
(env) $ pytest tests/test_fsm.py =============================================================================================== test session starts ================================================================================================ platform linux -- Python 3.10.4, pytest-7.1.2, pluggy-0.13.1 rootdir: /home/ubuntu/rift-python plugins: cov-2.10.1, profiling-1.7.0 collected 6 items tests/test_fsm.py ...... [100%] ================================================================================================ 6 passed in 0.05s =================================================================================================
Use the following command to run all unit tests as well as all system tests (it is currently not possibly to run just the unit tests):
(env) $ pytest tests ======================================================= test session starts ======================================================== platform darwin -- Python 3.5.1, pytest-3.6.4, py-1.5.4, pluggy-0.7.1 rootdir: /Users/brunorijsman/rift-python, inifile: plugins: cov-2.5.1 collected 13 items tests/test_fsm.py ... [ 23%] tests/test_sys_2n_l0_l1.py . [ 30%] tests/test_sys_2n_l0_l2.py . [ 38%] tests/test_sys_2n_l1_l3.py . [ 46%] tests/test_sys_2n_un_l1.py . [ 53%] tests/test_sys_3n_l0_l1_l2.py . [ 61%] tests/test_sys_cli_commands.py . [ 69%] tests/test_table.py .... [100%] ==================================================== 13 passed in 98.54 seconds ====================================================
Use the following command to run a single test (test_rib in this example), to measure the code coverage using during the test, and to report the code coverage results in graphical manner using the web browser.
tools/cleanup && pytest -vvv -s tests/test_rib.py --cov --cov-report=html && open htmlcov/index.html
Note: The “open” command is used on Apple Mac computers to use an .hmtl file using the default web browser.
Once the report is generated, click on the module under test (rib.py in this example) to see which lines are covered and which not.
Note: Codecov (which is part of Continuous Integration process triggered by github commits) produces even nicer-looking graphical reports.
Diagnosing Unit Test Failures
If a unit test fails, you will get output that looks similar to this (in this example, I modified the test case to make if fail on purpose):
(env) $ pytest tests/test_fsm.py ======================================================= test session starts ======================================================== platform darwin -- Python 3.5.1, pytest-3.6.4, py-1.5.4, pluggy-0.7.1 rootdir: /Users/brunorijsman/rift-python, inifile: plugins: cov-2.5.1 collected 3 items tests/test_fsm.py ..F [100%] ============================================================= FAILURES ============================================================= __________________________________________________________ test_fsm_basic __________________________________________________________ dog = <tests.test_fsm.dog..Dog object at 0x1055dd0f0> def test_fsm_basic(dog): dog.fsm_instance.start() # Check initial state assert dog.fsm_instance.state == dog.State.SITTING # The state entry action sit for initial state sitting should have been executed assert dog.sits == 1 assert dog.total_actions == 1 dog.reset_action_counters() # Since there are no events queued, nothing should happen when we process queued events fsm.Fsm.process_queued_events() > assert dog.fsm_instance.state == dog.State.BARKING E AssertionError: assert <State.SITTING: 1> == <State.BARKING: 2> E + where <State.SITTING: 1> = <fsm.Fsm object at 0x1055dd128>.state E + where <fsm.Fsm object at 0x1055dd128> = <tests.test_fsm.dog. .Dog object at 0x1055dd0f0>.fsm_instance E + and <State.BARKING: 2> = <enum 'State'>.BARKING E + where <enum 'State'> = <tests.test_fsm.dog. .Dog object at 0x1055dd0f0>.State tests/test_fsm.py:128: AssertionError ================================================ 1 failed, 2 passed in 0.13 seconds ================================================ </pre> The line `tests/test_fsm.py:128: AssertionError` reports that there was an assertion failure (i.e. an unexpected outcome of the test case) on line 128 of file test_fsm.py. The line `assert dog.fsm_instance.state == dog.State.BARKING` is the line of code in the test that failed. The line `AssertionError: assert <State.SITTING: 1> == <State.BARKING: 2>` reports what the expected value and what the actual value was. Some Python code editors (including Visual Studio Code, which I use) allow you to run test cases in the editor with a debugger so you can set breakpoints and inspect variables. This hugely helps in writing test cases and debugging test failures.`` ## System Tests The system tests create a topology of multiple RIFT nodes, and use [Python Expect](https://pexpect.readthedocs.io/en/stable/) in combination with pytest to verify whether all nodes in the topology behave as expected using the following mechanisms: * Use *show* commands in the Command Line Interface (CLI) session to observe the state of the system (nodes, interfaces, etc.) after initial convergence and after some event (e.g. a link failure) occurred. The *show* commands in RIFT-Python provide extremely detailed output specifically for the purpose of enabling automated testing. * Use *set* commands in the Command Line Interface (CLI) session to perform actions such as simulating a uni-directional or bi-directional link failure. * Analyze the logs to determine whether various events (e.g. finite state machine transitions) occurred when they were expected to occur. The debug-level RIFT-Python logging is very detailed and very structured specifically for the purpose of enabling automated testing. The [tests directory in RIFT-Python](https://github.com/brunorijsman/rift-python/tree/master/tests) contains two helper modules (rift\_expect\_session.py and log\_expect\_session) to ease the task of writing automated tests. The unit tests are stored in the tests directory, and start with the test\_sys\_ prefix. The naming convention for the system test files indicates the number of nodes and the level of the nodes. For example `2n_un_l1` means a topology with 2 nodes, where one node has level undefined and the other node has level 1. Use the following command to run an individual system test: (env) $ pytest tests/test_sys_2n_l0_l1.py ======================================================= test session starts ======================================================== platform darwin -- Python 3.5.1, pytest-3.6.4, py-1.5.4, pluggy-0.7.1 rootdir: /Users/brunorijsman/rift-python, inifile: plugins: cov-2.5.1 collected 1 item tests/test_sys_2n_l0_l1.py . [100%] ==================================================== 1 passed in 18.62 seconds =====================================================Use the following command to run all system tests as well as all unit tests (it is currently not possibly to run just the system tests):(env) $ pytest tests ======================================================= test session starts ======================================================== platform darwin -- Python 3.5.1, pytest-3.6.4, py-1.5.4, pluggy-0.7.1 rootdir: /Users/brunorijsman/rift-python, inifile: plugins: cov-2.5.1 collected 13 items tests/test_fsm.py ... [ 23%] tests/test_sys_2n_l0_l1.py . [ 30%] tests/test_sys_2n_l0_l2.py . [ 38%] tests/test_sys_2n_l1_l3.py . [ 46%] tests/test_sys_2n_un_l1.py . [ 53%] tests/test_sys_3n_l0_l1_l2.py . [ 61%] tests/test_sys_cli_commands.py . [ 69%] tests/test_table.py .... [100%] ==================================================== 13 passed in 98.54 seconds ====================================================## Diagnosing System Test Failures (Unexpected Show Command Output) If a system test fails, you might get output that looks similar to this (in this example, I modified the test case to make if fail on purpose):(env) $ pytest tests/test_sys_2n_l0_l1.py ======================================================= test session starts ======================================================== platform darwin -- Python 3.5.1, pytest-3.6.4, py-1.5.4, pluggy-0.7.1 rootdir: /Users/brunorijsman/rift-python, inifile: plugins: cov-2.5.1 collected 1 item tests/test_sys_2n_l0_l1.py F [100%] ============================================================= FAILURES ============================================================= __________________________________________________________ test_2n_l0_l1 ___________________________________________________________ def test_2n_l0_l1(): passive_nodes = os.getenv("RIFT_PASSIVE_NODES", "").split(",") # Bring topology up res = RiftExpectSession("2n_l0_l1") les = LogExpectSession("rift.log") # Check that adjacency reaches 3-way, check offers, check levels if "node1" not in passive_nodes: > check_rift_node1_intf_up(res) tests/test_sys_2n_l0_l1.py:175: _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ tests/test_sys_2n_l0_l1.py:43: in check_rift_node1_intf_up interface="if1") tests/rift_expect_session.py:91: in check_adjacency_1way self.table_expect("| {} | | .* | ONE_WAY |".format(interface)) tests/rift_expect_session.py:77: in table_expect return self.expect(pattern, timeout) _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ self = <rift_expect_session.RiftExpectSession object at 0x1056e9cf8>, pattern = '[|] if1 +[|] +[|] .* +[|] ONE_WAY +[|]' timeout = 1.0 def expect(self, pattern, timeout=expect_timeout): msg = "\n\n*** Expect: {}\n\n".format(pattern) self._logfile.write(msg.encode()) try: self._expect_session.expect(pattern, timeout) except pexpect.TIMEOUT: # Report the failure outside of this block, otherwise pytest reports a huge callstack failed = True else: failed = False if failed: self.log_expect_failure() > pytest.fail('Timeout expecting "{} (see rift_expect.log for details)"'.format(pattern)) E Failed: Timeout expecting "[|] if1 +[|] +[|] .* +[|] ONE_WAY +[|] (see rift_expect.log for details)" tests/rift_expect_session.py:70: Failed ==================================================== 1 failed in 11.33 seconds ===================================================== (env) $In the above example, the line `see rift_expect.log for details` indicates that there was an unexpected output from a CLI show command. If you open the file `rift_expect.log` and scroll to the bottom, you might see something similar to this:[... many lines skipped ...] *** Expect: [|] if1 +[|] +[|] .* +[|] ONE_WAY +[|] show interfaces +-----------+-----------+-----------+-----------+-------------------+-------+ | Interface | Neighbor | Neighbor | Neighbor | Time in | Flaps | | Name | Name | System ID | State | State | | +-----------+-----------+-----------+-----------+-------------------+-------+ | if1 | node2-if1 | 2 | THREE_WAY | 0d 00h:00m:01.52s | 0 | +-----------+-----------+-----------+-----------+-------------------+-------+ node1> *** Did not find expected pattern File "/Users/brunorijsman/rift-python/tests/test_sys_2n_l0_l1.py", line 175, in test_2n_l0_l1 check_rift_node1_intf_up(res) File "/Users/brunorijsman/rift-python/tests/test_sys_2n_l0_l1.py", line 43, in check_rift_node1_intf_up interface="if1") File "tests/rift_expect_session.py", line 91, in check_adjacency_1way self.table_expect("| {} | | .* | ONE_WAY |".format(interface)) File "tests/rift_expect_session.py", line 77, in table_expect return self.expect(pattern, timeout) File "tests/rift_expect_session.py", line 69, in expect self.log_expect_failure() File "tests/rift_expect_session.py", line 53, in log_expect_failure for line in traceback.format_stack():This line shows a regular expression with the expected output:*** Expect: [|] if1 +[|] +[|] .* +[|] ONE_WAY +[|]This line shows the actual output. Here we see that we actually got `THREE_WAY` instead of the expected `ONE_WAY`:show interfaces +-----------+-----------+-----------+-----------+-------------------+-------+ | Interface | Neighbor | Neighbor | Neighbor | Time in | Flaps | | Name | Name | System ID | State | State | | +-----------+-----------+-----------+-----------+-------------------+-------+ | if1 | node2-if1 | 2 | THREE_WAY | 0d 00h:00m:01.52s | 0 | +-----------+-----------+-----------+-----------+-------------------+-------+The following lines show the call stack where the test failure occurred:File "/Users/brunorijsman/rift-python/tests/test_sys_2n_l0_l1.py", line 175, in test_2n_l0_l1 check_rift_node1_intf_up(res) File "/Users/brunorijsman/rift-python/tests/test_sys_2n_l0_l1.py", line 43, in check_rift_node1_intf_up interface="if1") File "tests/rift_expect_session.py", line 91, in check_adjacency_1way self.table_expect("| {} | | .* | ONE_WAY |".format(interface)) File "tests/rift_expect_session.py", line 77, in table_expect return self.expect(pattern, timeout) File "tests/rift_expect_session.py", line 69, in expect self.log_expect_failure() File "tests/rift_expect_session.py", line 53, in log_expect_failure for line in traceback.format_stack():Once again, just as in the unit test section, it is extremely helpful to use a Python Editor that allows you to use an interactive debugger with breakpoints and variable inspection to debug system test failures. ## Diagnosing System Test Failures (Unexpected FSM Transition) Alternatively, if a system test fails, you might also get output that looks similar to this (once again, I modified the test case to make if fail on purpose):(env) $ pytest tests/test_sys_2n_l0_l1.py ======================================================= test session starts ======================================================== platform darwin -- Python 3.5.1, pytest-3.6.4, py-1.5.4, pluggy-0.7.1 rootdir: /Users/brunorijsman/rift-python, inifile: plugins: cov-2.5.1 collected 1 item tests/test_sys_2n_l0_l1.py F [100%] ============================================================= FAILURES ============================================================= __________________________________________________________ test_2n_l0_l1 ___________________________________________________________ def test_2n_l0_l1(): passive_nodes = os.getenv("RIFT_PASSIVE_NODES", "").split(",") # Bring topology up les = LogExpectSession("rift.log") res = RiftExpectSession("2n_l0_l1") # Check that adjacency reaches 3-way, check offers, check levels if "node1" not in passive_nodes: check_rift_node1_intf_up(res) check_log_node1_intf_up(les) if "node2" not in passive_nodes: check_rift_node2_intf_up(res) > check_log_node2_intf_up(les) tests/test_sys_2n_l0_l1.py:179: _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ tests/test_sys_2n_l0_l1.py:160: in check_log_node2_intf_up les.check_lie_fsm_1way_unacc_hdr("node1", "if1") #!!! 3way tests/log_expect_session.py:230: in check_lie_fsm_1way_unacc_hdr to_state="ONE_WAY") tests/log_expect_session.py:132: in fsm_find self.expect_failure(msg) _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ self = <log_expect_session.LogExpectSession object at 0x1056cfdd8>, msg = 'Did not find FSM transition for target-id node1-if1' def expect_failure(self, msg): self._results_file.write(msg) # Generate a call stack in rift_expect.log for easier debugging # But pytest call stacks are very deep, so only show the "interesting" lines for line in traceback.format_stack(): if "tests/" in line: self._results_file.write(line.strip()) self._results_file.write("\n") > assert False, msg + " (see log_expect.log for details)" E AssertionError: Did not find FSM transition for target-id node1-if1 (see log_expect.log for details) tests/log_expect_session.py:39: AssertionError ==================================================== 1 failed in 11.31 seconds ===================================================== (env) $In the above example, the line `see log_expect.log for details` indicates that there was an unexpected line in the log (usually an unexpected FSM transition). If you open the file `log_expect.log ` and scroll to the bottom, you might see something similar to this:[... many lines skipped ...] Finding FSM transition: target-id = node1-if1 from-state = ONE_WAY event = UNACCEPTABLE_HEADER to-state = ONE_WAY Observed FSM transition: log-line-nr = 22 sequence-nr = 3 from-state = ONE_WAY event = TIMER_TICK actions-and-pushed-events = SEND_LIE to-state = None implicit = False [... many lines skipped ...] Observed FSM transition: log-line-nr = 216 sequence-nr = 87 from-state = THREE_WAY event = LIE_RECEIVED actions-and-pushed-events = process_lie to-state = None implicit = False Did not find FSM transition for node1-if1 File "/Users/brunorijsman/rift-python/tests/test_sys_2n_l0_l1.py", line 179, in test_2n_l0_l1 check_log_node2_intf_up(les) File "/Users/brunorijsman/rift-python/tests/test_sys_2n_l0_l1.py", line 160, in check_log_node2_intf_up les.check_lie_fsm_1way_unacc_hdr("node1", "if1") #!!! 3way File "tests/log_expect_session.py", line 230, in check_lie_fsm_1way_unacc_hdr to_state="ONE_WAY") File "tests/log_expect_session.py", line 132, in fsm_find self.expect_failure(msg) File "tests/log_expect_session.py", line 35, in expect_failure for line in traceback.format_stack():The following lines reports the FSM transition that the test case was looking for (here the word "finding" means that the FSM transition is expected to happen at any point in the future, and the word "expecting" means that the FSM transition is expected to be the very next transition for that FSM instance in the log):Finding FSM transition: target-id = node1-if1 from-state = ONE_WAY event = UNACCEPTABLE_HEADER to-state = ONE_WAYThe following lines report the FSM transitions that were actually observed. Note that this only reports transitions for the same FSM instance as the FSM instance of the expected transition. For example, if we are looking for an expected FSM transition on one particular interfaces, this will only report observed FSM transitions for that same interface and not FSM transitions for nodes or other interfaces.Observed FSM transition: log-line-nr = 22 sequence-nr = 3 from-state = ONE_WAY event = TIMER_TICK actions-and-pushed-events = SEND_LIE to-state = None implicit = FalseThe following line reports that the expected FSM transition was not observed:Did not find FSM transition for node1-if1And finally, the following lines report the call stack for the test case failure:File "/Users/brunorijsman/rift-python/tests/test_sys_2n_l0_l1.py", line 179, in test_2n_l0_l1 check_log_node2_intf_up(les) File "/Users/brunorijsman/rift-python/tests/test_sys_2n_l0_l1.py", line 160, in check_log_node2_intf_up les.check_lie_fsm_1way_unacc_hdr("node1", "if1") #!!! 3way File "tests/log_expect_session.py", line 230, in check_lie_fsm_1way_unacc_hdr to_state="ONE_WAY") File "tests/log_expect_session.py", line 132, in fsm_find self.expect_failure(msg) File "tests/log_expect_session.py", line 35, in expect_failure for line in traceback.format_stack():## Diagnosing Interoperability Test Failures Running an interop test suite, creates a directory whose name is `interop-results` followed by a timestamp, for example `interop-results-2018-08-16-13:26:24.651390`. This directory contains one subdirectory for each interop test case in the suite, for example:(env) $ ls -1 interop-results-2018-08-16-13\:26\:36.688356/ 2n_l0_l1-node1 2n_l0_l1-node2 2n_l0_l2-node1 2n_l0_l2-node2 2n_l1_l3-node1 2n_l1_l3-node2 2n_un_l1-node1 2n_un_l1-node2 3n_l0_l1_l2-node1 3n_l0_l1_l2-node1-node2 3n_l0_l1_l2-node2 3n_l0_l1_l2-node2-node3 3n_l0_l1_l2-node3Each of these subdirectories contains all the files necessary to debug why a particular test case failed:(env) $ ls -1 interop-results-2018-08-16-13\:26\:36.688356/2n_l0_l1-node1 juniper-2n_l0_l1.yaml juniper-rift.log pytest.log rift_expect.log## Interoperability Tests The interoperability tests build upon the system tests. The interoperability tests run the exact same topologies and test cases as in the system tests, except that one or more RIFT-Python nodes are replaced by RIFT-Juniper nodes. To run the interoperability tests, first make sure that the RIFT-Python executable (rift-environ) is in your search path (PATH). Then run the `test/interop.py' script:(env) $ tests/interop.py 2n_l0_l1-node1... Pass 2n_l0_l1-node2... Pass 2n_l0_l2-node1... Pass 2n_l0_l2-node2... Pass 2n_l1_l3-node1... Pass 2n_l1_l3-node2... Pass 3n_l0_l1_l2-node1... Pass 3n_l0_l1_l2-node2... Pass 3n_l0_l1_l2-node3... Pass 3n_l0_l1_l2-node1-node2... Pass 3n_l0_l1_l2-node2-node3... Pass 3n_l0_l1_l2-node1-node3... Pass 2n_un_l1-node1... Pass 2n_un_l1-node2... Pass 2n_un_l2-node1... Pass 2n_un_l2-node2... Pass 2n_un_l0-node1... Pass 2n_un_l0-node2... Pass (env) $This currently takes about 6 minutes to complete (but it will take longer when I introduce more test cases). Each line of output reports whether or not a particular interop test case passed or failed. The naming convention for interop test cases is the topology plus a list of nodes that are running as RIFT-Juniper nodes. For example, test case `2n_l0_l1-node2` is topology `2n_l0_l1` with node2 running as a RIFT-Juniper node. As another example, test case `3n_l0_l1_l2-node1-node2` is topology `3n_l0_l1_l2` where node1 and node2 are running as RIFT-Juniper nodes. At the beginning of the tile `tests/interop.py` you can see a list of test cases that are executed during the interop testing:TEST_CASES = [("test_sys_2n_l0_l1.py", "2n_l0_l1.yaml", ["node1"]), ("test_sys_2n_l0_l1.py", "2n_l0_l1.yaml", ["node2"]), ("test_sys_2n_l0_l2.py", "2n_l0_l2.yaml", ["node1"]), ("test_sys_2n_l0_l2.py", "2n_l0_l2.yaml", ["node2"]), ("test_sys_2n_l1_l3.py", "2n_l1_l3.yaml", ["node1"]), ("test_sys_2n_l1_l3.py", "2n_l1_l3.yaml", ["node2"]), ("test_sys_3n_l0_l1_l2.py", "3n_l0_l1_l2.yaml", ["node1"]), ("test_sys_3n_l0_l1_l2.py", "3n_l0_l1_l2.yaml", ["node2"]), ("test_sys_3n_l0_l1_l2.py", "3n_l0_l1_l2.yaml", ["node3"]), ("test_sys_3n_l0_l1_l2.py", "3n_l0_l1_l2.yaml", ["node1", "node2"]), ("test_sys_3n_l0_l1_l2.py", "3n_l0_l1_l2.yaml", ["node2", "node3"]), ("test_sys_3n_l0_l1_l2.py", "3n_l0_l1_l2.yaml", ["node1", "node3"]), ("test_sys_2n_un_l1.py", "2n_un_l1.yaml", ["node1"]), ("test_sys_2n_un_l1.py", "2n_un_l1.yaml", ["node2"])]Each line contains a 3-tuple (topology, sys\_test, juniper\_nodes) that describes a single integration test cases: * topology is the name of the topology file. * sys\_test is the name of the system test script. * juniper\_nodes is a list of node names that run RIFT-Juniper instead of RIFT-Python ## Code Coverage To measure how much code-under-test is actually covered by a test script, use the following steps to measure code coverage. First run the `cleanup` script. This removes all temporary files, including the code coverage data from previous runs.(env) $ tools/cleanup (env) $If you run multiple test scripts without doing a `cleanup` in between, then the code coverage report will cover the cumulative coverage across the multiple tests. Then, use `pytest` with a specific set of options including the `--cov` option to run one or more test cases:(env) $ pytest -vvv -s tests/test_table.py --cov --cov-report=html ======================================================= test session starts ======================================================== platform darwin -- Python 3.7.1, pytest-3.6.4, py-1.5.4, pluggy-0.7.1 -- /Users/brunorijsman/rift-python/env/bin/python3.7 cachedir: .pytest_cache rootdir: /Users/brunorijsman/rift-python, inifile: plugins: profiling-1.3.0, cov-2.5.1 collected 4 items tests/test_table.py::test_simple_table PASSED tests/test_table.py::test_multi_line_cells PASSED tests/test_table.py::test_format_extend PASSED tests/test_table.py::test_no_separators PASSED ---------- coverage: platform darwin, python 3.7.1-final-0 ----------- Coverage HTML written to dir htmlcov ===================================================== 4 passed in 0.13 seconds ===================================================== (env) $Finally, open the generated HTML files to view the coverage results. The following example assumes you are running on macOS and opens the default web browser to view the results.(env) $ open htmlcov/index.html (env) $The browser will display a directory containing all files whose coverage was measured, similar to this one: ![RIFT-Python Coverage Report Example: Directory](http://bit.ly/rift-python-coverage-report-example-directory-png) If you click on one of the files, a detailed report is displayed showing which lines are (green) or are not (red) covered. Black lines mean that the line does not contain executable code. In this example the file `table.py` contains the module-under-test: ![RIFT-Python Coverage Report Example: One File](http://bit.ly/rift-python-coverage-report-example-one-file-png) You can combine all steps into a single command line command for a very quick turn-around development cycle when you are developing your test script to cover as much of the code-under-test as possible:(env) $ tools/cleanup && pytest -vvv -s tests/test_table.py --cov --cov-report=html && open htmlcov/indexNote: the [Continuous Integration](/rift-python/doc/continuous-integration.html) process uses [codecov](https://codecov.io/gh/brunorijsman/rift-python) which produces some nicer code coverage reports than the ones described above. ## Code Profiling Use the following steps to profile the RIFT-Python code, i.e. to measure how much time RIFT-Python spends in each function. Start a test topology (in this example topology `4n_diamond_parallel`) as follows:$ cd rift $ python -m cProfile -o output __main__.py -i ../topology/4n_diamond_parallel.yamlThe `-m cProfile` option enables profiling. The `-o output` option causes the raw profile data to be written to the file `output`. Note that we start RIFT-Python a bit different than usual. Instead of using `python rift` to start the RIFT module, we go into the `rift` directory and start `python __main__.py`. This is needed because cProfile has some issues starting a module. Once the topology-under-test is running, you can execute whatever scenario you want to profile. In this case, we just let the topology converge for a few seconds, and then stop the RIFT engine:leaf> stop (env) $At this point, the file `output` contains the raw profile data. We can view a textual summary of the profile data by running the following Python script:import pstats from pstats import SortKey p = pstats.Stats('output') p.sort_stats(SortKey.TIME).print_stats(10)This example reports the top 10 functions, sorted by total time spend in the function itself:Wed Dec 19 14:44:20 2018 output 1515197 function calls (1414768 primitive calls) in 6.856 seconds Ordered by: internal time List reduced from 2672 to 10 due to restriction <10> ncalls tottime percall cumtime percall filename:lineno(function) 57 3.114 0.055 3.114 0.055 {built-in method select.select} 2 2.011 1.005 2.011 1.005 {built-in method time.sleep} 256 0.121 0.000 0.121 0.000 {method 'read' of '_io.FileIO' objects} 39855/366 0.082 0.000 0.193 0.001 /Users/brunorijsman/rift-python/env/lib/python3.7/copy.py:132(deepcopy) 256 0.079 0.000 0.200 0.001:914(get_data) 28 0.074 0.003 0.074 0.003 {built-in method _imp.create_dynamic} 256 0.045 0.000 0.045 0.000 {built-in method marshal.loads} 141087 0.034 0.000 0.046 0.000 {built-in method builtins.isinstance} 32430 0.034 0.000 0.060 0.000 /Users/brunorijsman/rift-python/env/lib/python3.7/site-packages/thrift/transport/TTransport.py:56(readAll) 1323 0.033 0.000 0.033 0.000 {built-in method posix.stat} <pstats.Stats object at 0x10d3549b0> </pre> To produce a graphical representation of the profile (which is much easier to understand and much more comprehensive than the above text output) use the following steps. First install the `gprof2dot` Python module: (env) $ pip install gprof2dot Collecting gprof2dot Installing collected packages: gprof2dot Successfully installed gprof2dot-2017.9.19 (env) $Then, use the following shell command to first convert the raw profile data into a "dot" graph, then convert the dot graph into a .png graphics file, and finally open the .png graphics file in a web browser (the `open` command assumes you are running on macOS):(env) $ python -m gprof2dot -f pstats output | dot -Tpng -o output.png && open output.pngThis produces a diagram similar to the following one: ![RIFT-Python Profile Report Example](http://bit.ly/example-rift-python-code-profile) ## Log Visualization Tool Once you start testing non-trivial topologies, it becomes extremely difficult to read the log files and to understand what is really happening. The [Log Visualization Tool](/rift-python/doc/log-visualization.html) converts a log file into a graphical ladder diagram, which is _much_ easier to understand. ## Temporary Files The automated tests generate many temporary files: * The `rift.log` file store the RIFT-Python logs * The `rift.log.html` file store the RIFT-Python log visualization * The `rift_expect.log` file stores information to help debug system test failures (unexpected CLI output) * The `log_expect.log` file stores information to help debug system test failures (unexpected FSM transition in log) * The `interop-results-*` directories contain information to help debug interop test failures * The `.coverage.*` files store code coverage data The script `tools/cleanup` cleans up all these temporary files. There is also a `.gitignore` file to ignore these files for git commits. ## Continuous Integration I use [Travis Continuous Integration (CI)](https://travis-ci.com/brunorijsman/rift-python) For every commit, Travis CI automatically runs pylint, the full unit test suite, and the full system test suite. The interoperability test suite is not automatically run - it must be run manually. I use [codecov](https://codecov.io/gh/brunorijsman/rift-python) for visualizing the code coverage results. See [Continuous Integration](/rift-python/doc/continuous-integration.html) for more details.