blob: fc6f550184f40b4e0ad492ff5eca9d1d016b9c35 [file] [log] [blame]
Damjan Marionf56b77a2016-10-03 19:44:57 +02001#!/usr/bin/env python
Damjan Marionf56b77a2016-10-03 19:44:57 +02002
Klement Sekeraacb9b8e2017-02-14 02:55:31 +01003from __future__ import print_function
4import gc
5import sys
6import os
7import select
Damjan Marionf56b77a2016-10-03 19:44:57 +02008import unittest
Klement Sekeraf62ae122016-10-11 11:47:09 +02009import tempfile
Klement Sekera277b89c2016-10-28 13:20:27 +020010import time
Klement Sekeraf62ae122016-10-11 11:47:09 +020011import resource
Klement Sekera3658adc2017-06-07 08:19:47 +020012import faulthandler
Klement Sekera6a6f4f72017-11-09 09:16:39 +010013import random
Klement Sekerae4504c62016-12-08 10:16:41 +010014from collections import deque
Klement Sekeraacb9b8e2017-02-14 02:55:31 +010015from threading import Thread, Event
Klement Sekera909a6a12017-08-08 04:33:53 +020016from inspect import getdoc, isclass
Klement Sekerab91017a2017-02-09 06:04:36 +010017from traceback import format_exception
Klement Sekeraacb9b8e2017-02-14 02:55:31 +010018from logging import FileHandler, DEBUG, Formatter
19from scapy.packet import Raw
Klement Sekera277b89c2016-10-28 13:20:27 +020020from hook import StepHook, PollHook
Klement Sekeraf62ae122016-10-11 11:47:09 +020021from vpp_pg_interface import VppPGInterface
Klement Sekeradab231a2016-12-21 08:50:14 +010022from vpp_sub_interface import VppSubInterface
Matej Klotton0178d522016-11-04 11:11:44 +010023from vpp_lo_interface import VppLoInterface
Klement Sekeraf62ae122016-10-11 11:47:09 +020024from vpp_papi_provider import VppPapiProvider
Klement Sekera277b89c2016-10-28 13:20:27 +020025from log import *
Klement Sekera10db26f2017-01-11 08:16:53 +010026from vpp_object import VppObjectRegistry
Klement Sekera75e7d132017-09-20 08:26:30 +020027from vpp_punt_socket import vpp_uds_socket_name
Klement Sekeraacb9b8e2017-02-14 02:55:31 +010028if os.name == 'posix' and sys.version_info[0] < 3:
29 # using subprocess32 is recommended by python official documentation
30 # @ https://docs.python.org/2/library/subprocess.html
31 import subprocess32 as subprocess
32else:
33 import subprocess
Klement Sekeraf62ae122016-10-11 11:47:09 +020034
35"""
36 Test framework module.
37
38 The module provides a set of tools for constructing and running tests and
39 representing the results.
40"""
41
Klement Sekeraf62ae122016-10-11 11:47:09 +020042
Damjan Marionf56b77a2016-10-03 19:44:57 +020043class _PacketInfo(object):
Klement Sekeraf62ae122016-10-11 11:47:09 +020044 """Private class to create packet info object.
45
46 Help process information about the next packet.
47 Set variables to default values.
Klement Sekeraf62ae122016-10-11 11:47:09 +020048 """
Matej Klotton86d87c42016-11-11 11:38:55 +010049 #: Store the index of the packet.
Damjan Marionf56b77a2016-10-03 19:44:57 +020050 index = -1
Matej Klotton86d87c42016-11-11 11:38:55 +010051 #: Store the index of the source packet generator interface of the packet.
Damjan Marionf56b77a2016-10-03 19:44:57 +020052 src = -1
Matej Klotton86d87c42016-11-11 11:38:55 +010053 #: Store the index of the destination packet generator interface
54 #: of the packet.
Damjan Marionf56b77a2016-10-03 19:44:57 +020055 dst = -1
Pavel Kotucek59dda062017-03-02 15:22:47 +010056 #: Store expected ip version
57 ip = -1
58 #: Store expected upper protocol
59 proto = -1
Matej Klotton86d87c42016-11-11 11:38:55 +010060 #: Store the copy of the former packet.
Damjan Marionf56b77a2016-10-03 19:44:57 +020061 data = None
Damjan Marionf56b77a2016-10-03 19:44:57 +020062
Matej Klotton16a14cd2016-12-07 15:09:13 +010063 def __eq__(self, other):
64 index = self.index == other.index
65 src = self.src == other.src
66 dst = self.dst == other.dst
67 data = self.data == other.data
68 return index and src and dst and data
69
Klement Sekeraf62ae122016-10-11 11:47:09 +020070
Klement Sekeraacb9b8e2017-02-14 02:55:31 +010071def pump_output(testclass):
72 """ pump output from vpp stdout/stderr to proper queues """
Klement Sekera6a6f4f72017-11-09 09:16:39 +010073 stdout_fragment = ""
74 stderr_fragment = ""
Klement Sekeraacb9b8e2017-02-14 02:55:31 +010075 while not testclass.pump_thread_stop_flag.wait(0):
76 readable = select.select([testclass.vpp.stdout.fileno(),
77 testclass.vpp.stderr.fileno(),
78 testclass.pump_thread_wakeup_pipe[0]],
79 [], [])[0]
80 if testclass.vpp.stdout.fileno() in readable:
Klement Sekera6a6f4f72017-11-09 09:16:39 +010081 read = os.read(testclass.vpp.stdout.fileno(), 102400)
82 if len(read) > 0:
83 split = read.splitlines(True)
84 if len(stdout_fragment) > 0:
85 split[0] = "%s%s" % (stdout_fragment, split[0])
86 if len(split) > 0 and split[-1].endswith("\n"):
87 limit = None
88 else:
89 limit = -1
90 stdout_fragment = split[-1]
91 testclass.vpp_stdout_deque.extend(split[:limit])
92 if not testclass.cache_vpp_output:
93 for line in split[:limit]:
94 testclass.logger.debug(
95 "VPP STDOUT: %s" % line.rstrip("\n"))
Klement Sekeraacb9b8e2017-02-14 02:55:31 +010096 if testclass.vpp.stderr.fileno() in readable:
Klement Sekera6a6f4f72017-11-09 09:16:39 +010097 read = os.read(testclass.vpp.stderr.fileno(), 102400)
98 if len(read) > 0:
99 split = read.splitlines(True)
100 if len(stderr_fragment) > 0:
101 split[0] = "%s%s" % (stderr_fragment, split[0])
102 if len(split) > 0 and split[-1].endswith("\n"):
103 limit = None
104 else:
105 limit = -1
106 stderr_fragment = split[-1]
107 testclass.vpp_stderr_deque.extend(split[:limit])
108 if not testclass.cache_vpp_output:
109 for line in split[:limit]:
110 testclass.logger.debug(
111 "VPP STDERR: %s" % line.rstrip("\n"))
Klement Sekeraacb9b8e2017-02-14 02:55:31 +0100112 # ignoring the dummy pipe here intentionally - the flag will take care
113 # of properly terminating the loop
Klement Sekera277b89c2016-10-28 13:20:27 +0200114
115
Klement Sekera87134932017-03-07 11:39:27 +0100116def running_extended_tests():
117 try:
118 s = os.getenv("EXTENDED_TESTS")
119 return True if s.lower() in ("y", "yes", "1") else False
120 except:
121 return False
122 return False
123
124
Klement Sekerad3e671e2017-09-29 12:36:37 +0200125def running_on_centos():
126 try:
127 os_id = os.getenv("OS_ID")
128 return True if "centos" in os_id.lower() else False
129 except:
130 return False
131 return False
132
133
Klement Sekera909a6a12017-08-08 04:33:53 +0200134class KeepAliveReporter(object):
135 """
136 Singleton object which reports test start to parent process
137 """
138 _shared_state = {}
139
140 def __init__(self):
141 self.__dict__ = self._shared_state
142
143 @property
144 def pipe(self):
145 return self._pipe
146
147 @pipe.setter
148 def pipe(self, pipe):
149 if hasattr(self, '_pipe'):
150 raise Exception("Internal error - pipe should only be set once.")
151 self._pipe = pipe
152
153 def send_keep_alive(self, test):
154 """
155 Write current test tmpdir & desc to keep-alive pipe to signal liveness
156 """
Klement Sekera3f6ff192017-08-11 06:56:05 +0200157 if self.pipe is None:
158 # if not running forked..
159 return
160
Klement Sekera909a6a12017-08-08 04:33:53 +0200161 if isclass(test):
162 desc = test.__name__
163 else:
164 desc = test.shortDescription()
165 if not desc:
166 desc = str(test)
167
Dave Wallacee2efd122017-09-30 22:04:21 -0400168 self.pipe.send((desc, test.vpp_bin, test.tempdir, test.vpp.pid))
Klement Sekera909a6a12017-08-08 04:33:53 +0200169
170
Damjan Marionf56b77a2016-10-03 19:44:57 +0200171class VppTestCase(unittest.TestCase):
Matej Klotton86d87c42016-11-11 11:38:55 +0100172 """This subclass is a base class for VPP test cases that are implemented as
173 classes. It provides methods to create and run test case.
Klement Sekeraf62ae122016-10-11 11:47:09 +0200174 """
175
176 @property
177 def packet_infos(self):
178 """List of packet infos"""
179 return self._packet_infos
180
Klement Sekeradab231a2016-12-21 08:50:14 +0100181 @classmethod
182 def get_packet_count_for_if_idx(cls, dst_if_index):
183 """Get the number of packet info for specified destination if index"""
184 if dst_if_index in cls._packet_count_for_dst_if_idx:
185 return cls._packet_count_for_dst_if_idx[dst_if_index]
186 else:
187 return 0
Klement Sekeraf62ae122016-10-11 11:47:09 +0200188
189 @classmethod
190 def instance(cls):
191 """Return the instance of this testcase"""
192 return cls.test_instance
193
Damjan Marionf56b77a2016-10-03 19:44:57 +0200194 @classmethod
Klement Sekera277b89c2016-10-28 13:20:27 +0200195 def set_debug_flags(cls, d):
196 cls.debug_core = False
197 cls.debug_gdb = False
198 cls.debug_gdbserver = False
199 if d is None:
200 return
201 dl = d.lower()
202 if dl == "core":
Klement Sekera277b89c2016-10-28 13:20:27 +0200203 cls.debug_core = True
204 elif dl == "gdb":
205 cls.debug_gdb = True
206 elif dl == "gdbserver":
207 cls.debug_gdbserver = True
208 else:
209 raise Exception("Unrecognized DEBUG option: '%s'" % d)
210
211 @classmethod
Damjan Marionf56b77a2016-10-03 19:44:57 +0200212 def setUpConstants(cls):
Klement Sekeraf62ae122016-10-11 11:47:09 +0200213 """ Set-up the test case class based on environment variables """
214 try:
Klement Sekera277b89c2016-10-28 13:20:27 +0200215 s = os.getenv("STEP")
216 cls.step = True if s.lower() in ("y", "yes", "1") else False
Klement Sekeraf62ae122016-10-11 11:47:09 +0200217 except:
Klement Sekera277b89c2016-10-28 13:20:27 +0200218 cls.step = False
219 try:
220 d = os.getenv("DEBUG")
221 except:
222 d = None
Klement Sekeraa3d933c2017-11-06 09:46:00 +0100223 try:
224 c = os.getenv("CACHE_OUTPUT", "1")
225 cls.cache_vpp_output = \
Klement Sekerae1783992017-11-07 03:19:16 +0100226 False if c.lower() in ("n", "no", "0") else True
Klement Sekeraa3d933c2017-11-06 09:46:00 +0100227 except:
228 cls.cache_vpp_output = True
Klement Sekera277b89c2016-10-28 13:20:27 +0200229 cls.set_debug_flags(d)
Damjan Marionf56b77a2016-10-03 19:44:57 +0200230 cls.vpp_bin = os.getenv('VPP_TEST_BIN', "vpp")
Pierre Pfistercd8e3182016-10-07 16:30:03 +0100231 cls.plugin_path = os.getenv('VPP_TEST_PLUGIN_PATH')
Klement Sekera47e275b2017-03-21 08:21:25 +0100232 cls.extern_plugin_path = os.getenv('EXTERN_PLUGINS')
233 plugin_path = None
234 if cls.plugin_path is not None:
235 if cls.extern_plugin_path is not None:
236 plugin_path = "%s:%s" % (
237 cls.plugin_path, cls.extern_plugin_path)
Klement Sekera6abbc282017-03-24 05:47:15 +0100238 else:
239 plugin_path = cls.plugin_path
Klement Sekera47e275b2017-03-21 08:21:25 +0100240 elif cls.extern_plugin_path is not None:
241 plugin_path = cls.extern_plugin_path
Klement Sekera01bbbe92016-11-02 09:25:05 +0100242 debug_cli = ""
243 if cls.step or cls.debug_gdb or cls.debug_gdbserver:
244 debug_cli = "cli-listen localhost:5002"
Klement Sekera80a7f0a2017-03-02 11:27:11 +0100245 coredump_size = None
246 try:
247 size = os.getenv("COREDUMP_SIZE")
248 if size is not None:
249 coredump_size = "coredump-size %s" % size
250 except:
251 pass
252 if coredump_size is None:
Dave Wallacee2efd122017-09-30 22:04:21 -0400253 coredump_size = "coredump-size unlimited"
Klement Sekera80a7f0a2017-03-02 11:27:11 +0100254 cls.vpp_cmdline = [cls.vpp_bin, "unix",
Dave Wallacee2efd122017-09-30 22:04:21 -0400255 "{", "nodaemon", debug_cli, "full-coredump",
256 coredump_size, "}", "api-trace", "{", "on", "}",
Damjan Marion374e2c52017-03-09 20:38:15 +0100257 "api-segment", "{", "prefix", cls.shm_prefix, "}",
258 "plugins", "{", "plugin", "dpdk_plugin.so", "{",
Klement Sekera75e7d132017-09-20 08:26:30 +0200259 "disable", "}", "}",
260 "punt", "{", "socket", cls.punt_socket_path, "}"]
Klement Sekera47e275b2017-03-21 08:21:25 +0100261 if plugin_path is not None:
262 cls.vpp_cmdline.extend(["plugin_path", plugin_path])
Klement Sekera277b89c2016-10-28 13:20:27 +0200263 cls.logger.info("vpp_cmdline: %s" % cls.vpp_cmdline)
264
265 @classmethod
266 def wait_for_enter(cls):
267 if cls.debug_gdbserver:
268 print(double_line_delim)
269 print("Spawned GDB server with PID: %d" % cls.vpp.pid)
270 elif cls.debug_gdb:
271 print(double_line_delim)
272 print("Spawned VPP with PID: %d" % cls.vpp.pid)
273 else:
274 cls.logger.debug("Spawned VPP with PID: %d" % cls.vpp.pid)
275 return
276 print(single_line_delim)
277 print("You can debug the VPP using e.g.:")
278 if cls.debug_gdbserver:
279 print("gdb " + cls.vpp_bin + " -ex 'target remote localhost:7777'")
280 print("Now is the time to attach a gdb by running the above "
281 "command, set up breakpoints etc. and then resume VPP from "
282 "within gdb by issuing the 'continue' command")
283 elif cls.debug_gdb:
284 print("gdb " + cls.vpp_bin + " -ex 'attach %s'" % cls.vpp.pid)
285 print("Now is the time to attach a gdb by running the above "
286 "command and set up breakpoints etc.")
287 print(single_line_delim)
288 raw_input("Press ENTER to continue running the testcase...")
289
290 @classmethod
291 def run_vpp(cls):
292 cmdline = cls.vpp_cmdline
293
294 if cls.debug_gdbserver:
Klement Sekera931be3a2016-11-03 05:36:01 +0100295 gdbserver = '/usr/bin/gdbserver'
296 if not os.path.isfile(gdbserver) or \
297 not os.access(gdbserver, os.X_OK):
298 raise Exception("gdbserver binary '%s' does not exist or is "
299 "not executable" % gdbserver)
300
301 cmdline = [gdbserver, 'localhost:7777'] + cls.vpp_cmdline
Klement Sekera277b89c2016-10-28 13:20:27 +0200302 cls.logger.info("Gdbserver cmdline is %s", " ".join(cmdline))
303
Klement Sekera931be3a2016-11-03 05:36:01 +0100304 try:
305 cls.vpp = subprocess.Popen(cmdline,
306 stdout=subprocess.PIPE,
307 stderr=subprocess.PIPE,
308 bufsize=1)
309 except Exception as e:
310 cls.logger.critical("Couldn't start vpp: %s" % e)
311 raise
312
Klement Sekera277b89c2016-10-28 13:20:27 +0200313 cls.wait_for_enter()
Pierre Pfistercd8e3182016-10-07 16:30:03 +0100314
Damjan Marionf56b77a2016-10-03 19:44:57 +0200315 @classmethod
316 def setUpClass(cls):
Klement Sekeraf62ae122016-10-11 11:47:09 +0200317 """
318 Perform class setup before running the testcase
319 Remove shared memory files, start vpp and connect the vpp-api
320 """
Klement Sekeraacb9b8e2017-02-14 02:55:31 +0100321 gc.collect() # run garbage collection first
Klement Sekera75e7d132017-09-20 08:26:30 +0200322 random.seed(1)
Klement Sekera277b89c2016-10-28 13:20:27 +0200323 cls.logger = getLogger(cls.__name__)
Klement Sekeraf62ae122016-10-11 11:47:09 +0200324 cls.tempdir = tempfile.mkdtemp(
Klement Sekeraf413bef2017-08-15 07:09:02 +0200325 prefix='vpp-unittest-%s-' % cls.__name__)
Klement Sekera027dbd52017-04-11 06:01:53 +0200326 cls.file_handler = FileHandler("%s/log.txt" % cls.tempdir)
327 cls.file_handler.setFormatter(
Klement Sekeraacb9b8e2017-02-14 02:55:31 +0100328 Formatter(fmt='%(asctime)s,%(msecs)03d %(message)s',
329 datefmt="%H:%M:%S"))
Klement Sekera027dbd52017-04-11 06:01:53 +0200330 cls.file_handler.setLevel(DEBUG)
331 cls.logger.addHandler(cls.file_handler)
Klement Sekeraf62ae122016-10-11 11:47:09 +0200332 cls.shm_prefix = cls.tempdir.split("/")[-1]
Klement Sekera75e7d132017-09-20 08:26:30 +0200333 cls.punt_socket_path = '%s/%s' % (cls.tempdir, vpp_uds_socket_name)
Klement Sekeraf62ae122016-10-11 11:47:09 +0200334 os.chdir(cls.tempdir)
Klement Sekera277b89c2016-10-28 13:20:27 +0200335 cls.logger.info("Temporary dir is %s, shm prefix is %s",
336 cls.tempdir, cls.shm_prefix)
Damjan Marionf56b77a2016-10-03 19:44:57 +0200337 cls.setUpConstants()
Klement Sekeradab231a2016-12-21 08:50:14 +0100338 cls.reset_packet_infos()
Klement Sekera9225dee2016-12-12 08:36:58 +0100339 cls._captures = []
340 cls._zombie_captures = []
Klement Sekeraf62ae122016-10-11 11:47:09 +0200341 cls.verbose = 0
Klement Sekera085f5c02016-11-24 01:59:16 +0100342 cls.vpp_dead = False
Klement Sekera10db26f2017-01-11 08:16:53 +0100343 cls.registry = VppObjectRegistry()
Klement Sekera3747c752017-04-10 06:30:17 +0200344 cls.vpp_startup_failed = False
Klement Sekera909a6a12017-08-08 04:33:53 +0200345 cls.reporter = KeepAliveReporter()
Klement Sekeraf62ae122016-10-11 11:47:09 +0200346 # need to catch exceptions here because if we raise, then the cleanup
347 # doesn't get called and we might end with a zombie vpp
348 try:
Klement Sekera277b89c2016-10-28 13:20:27 +0200349 cls.run_vpp()
Dave Wallacee2efd122017-09-30 22:04:21 -0400350 cls.reporter.send_keep_alive(cls)
Klement Sekerae4504c62016-12-08 10:16:41 +0100351 cls.vpp_stdout_deque = deque()
Klement Sekerae4504c62016-12-08 10:16:41 +0100352 cls.vpp_stderr_deque = deque()
Klement Sekeraacb9b8e2017-02-14 02:55:31 +0100353 cls.pump_thread_stop_flag = Event()
354 cls.pump_thread_wakeup_pipe = os.pipe()
355 cls.pump_thread = Thread(target=pump_output, args=(cls,))
Klement Sekeraaeeac3b2017-02-14 07:11:52 +0100356 cls.pump_thread.daemon = True
Klement Sekeraacb9b8e2017-02-14 02:55:31 +0100357 cls.pump_thread.start()
Klement Sekera7bb873a2016-11-18 07:38:42 +0100358 cls.vapi = VppPapiProvider(cls.shm_prefix, cls.shm_prefix, cls)
Klement Sekera085f5c02016-11-24 01:59:16 +0100359 if cls.step:
360 hook = StepHook(cls)
361 else:
362 hook = PollHook(cls)
363 cls.vapi.register_hook(hook)
Klement Sekeraacb9b8e2017-02-14 02:55:31 +0100364 cls.sleep(0.1, "after vpp startup, before initial poll")
Klement Sekera3747c752017-04-10 06:30:17 +0200365 try:
366 hook.poll_vpp()
367 except:
368 cls.vpp_startup_failed = True
369 cls.logger.critical(
370 "VPP died shortly after startup, check the"
371 " output to standard error for possible cause")
372 raise
Klement Sekera085f5c02016-11-24 01:59:16 +0100373 try:
374 cls.vapi.connect()
375 except:
376 if cls.debug_gdbserver:
377 print(colorize("You're running VPP inside gdbserver but "
378 "VPP-API connection failed, did you forget "
379 "to 'continue' VPP from within gdb?", RED))
380 raise
Klement Sekeraf62ae122016-10-11 11:47:09 +0200381 except:
Klement Sekera0529a742016-12-02 07:05:24 +0100382 t, v, tb = sys.exc_info()
Klement Sekera085f5c02016-11-24 01:59:16 +0100383 try:
384 cls.quit()
385 except:
386 pass
Klement Sekera0529a742016-12-02 07:05:24 +0100387 raise t, v, tb
Damjan Marionf56b77a2016-10-03 19:44:57 +0200388
Damjan Marionf56b77a2016-10-03 19:44:57 +0200389 @classmethod
390 def quit(cls):
Klement Sekeraf62ae122016-10-11 11:47:09 +0200391 """
392 Disconnect vpp-api, kill vpp and cleanup shared memory files
393 """
Klement Sekera277b89c2016-10-28 13:20:27 +0200394 if (cls.debug_gdbserver or cls.debug_gdb) and hasattr(cls, 'vpp'):
395 cls.vpp.poll()
396 if cls.vpp.returncode is None:
397 print(double_line_delim)
398 print("VPP or GDB server is still running")
399 print(single_line_delim)
Klement Sekerada505f62017-01-04 12:58:53 +0100400 raw_input("When done debugging, press ENTER to kill the "
401 "process and finish running the testcase...")
Klement Sekera277b89c2016-10-28 13:20:27 +0200402
Klement Sekeraacb9b8e2017-02-14 02:55:31 +0100403 os.write(cls.pump_thread_wakeup_pipe[1], 'ding dong wake up')
404 cls.pump_thread_stop_flag.set()
405 if hasattr(cls, 'pump_thread'):
406 cls.logger.debug("Waiting for pump thread to stop")
407 cls.pump_thread.join()
408 if hasattr(cls, 'vpp_stderr_reader_thread'):
409 cls.logger.debug("Waiting for stdderr pump to stop")
410 cls.vpp_stderr_reader_thread.join()
411
Klement Sekeraf62ae122016-10-11 11:47:09 +0200412 if hasattr(cls, 'vpp'):
Klement Sekera0529a742016-12-02 07:05:24 +0100413 if hasattr(cls, 'vapi'):
414 cls.vapi.disconnect()
Klement Sekeraacb9b8e2017-02-14 02:55:31 +0100415 del cls.vapi
Klement Sekeraf62ae122016-10-11 11:47:09 +0200416 cls.vpp.poll()
417 if cls.vpp.returncode is None:
Klement Sekeraacb9b8e2017-02-14 02:55:31 +0100418 cls.logger.debug("Sending TERM to vpp")
Klement Sekeraf62ae122016-10-11 11:47:09 +0200419 cls.vpp.terminate()
Klement Sekeraacb9b8e2017-02-14 02:55:31 +0100420 cls.logger.debug("Waiting for vpp to die")
421 cls.vpp.communicate()
Klement Sekeraf62ae122016-10-11 11:47:09 +0200422 del cls.vpp
Damjan Marionf56b77a2016-10-03 19:44:57 +0200423
Klement Sekera3747c752017-04-10 06:30:17 +0200424 if cls.vpp_startup_failed:
425 stdout_log = cls.logger.info
426 stderr_log = cls.logger.critical
427 else:
428 stdout_log = cls.logger.info
429 stderr_log = cls.logger.info
430
Klement Sekerae4504c62016-12-08 10:16:41 +0100431 if hasattr(cls, 'vpp_stdout_deque'):
Klement Sekera3747c752017-04-10 06:30:17 +0200432 stdout_log(single_line_delim)
433 stdout_log('VPP output to stdout while running %s:', cls.__name__)
434 stdout_log(single_line_delim)
Klement Sekerae4504c62016-12-08 10:16:41 +0100435 vpp_output = "".join(cls.vpp_stdout_deque)
Klement Sekera027dbd52017-04-11 06:01:53 +0200436 with open(cls.tempdir + '/vpp_stdout.txt', 'w') as f:
437 f.write(vpp_output)
Klement Sekera3747c752017-04-10 06:30:17 +0200438 stdout_log('\n%s', vpp_output)
439 stdout_log(single_line_delim)
Klement Sekera277b89c2016-10-28 13:20:27 +0200440
Klement Sekerae4504c62016-12-08 10:16:41 +0100441 if hasattr(cls, 'vpp_stderr_deque'):
Klement Sekera3747c752017-04-10 06:30:17 +0200442 stderr_log(single_line_delim)
443 stderr_log('VPP output to stderr while running %s:', cls.__name__)
444 stderr_log(single_line_delim)
Klement Sekerae4504c62016-12-08 10:16:41 +0100445 vpp_output = "".join(cls.vpp_stderr_deque)
Klement Sekera027dbd52017-04-11 06:01:53 +0200446 with open(cls.tempdir + '/vpp_stderr.txt', 'w') as f:
447 f.write(vpp_output)
Klement Sekera3747c752017-04-10 06:30:17 +0200448 stderr_log('\n%s', vpp_output)
449 stderr_log(single_line_delim)
Klement Sekera277b89c2016-10-28 13:20:27 +0200450
Damjan Marionf56b77a2016-10-03 19:44:57 +0200451 @classmethod
452 def tearDownClass(cls):
Klement Sekeraf62ae122016-10-11 11:47:09 +0200453 """ Perform final cleanup after running all tests in this test-case """
Damjan Marionf56b77a2016-10-03 19:44:57 +0200454 cls.quit()
Klement Sekera027dbd52017-04-11 06:01:53 +0200455 cls.file_handler.close()
Damjan Marionf56b77a2016-10-03 19:44:57 +0200456
Damjan Marionf56b77a2016-10-03 19:44:57 +0200457 def tearDown(self):
Klement Sekeraf62ae122016-10-11 11:47:09 +0200458 """ Show various debug prints after each test """
Klement Sekerab91017a2017-02-09 06:04:36 +0100459 self.logger.debug("--- tearDown() for %s.%s(%s) called ---" %
460 (self.__class__.__name__, self._testMethodName,
461 self._testMethodDoc))
Klement Sekeraf62ae122016-10-11 11:47:09 +0200462 if not self.vpp_dead:
Jan49c0fca2016-10-26 15:44:27 +0200463 self.logger.debug(self.vapi.cli("show trace"))
Neale Ranns88fc83e2017-04-05 08:11:14 -0700464 self.logger.info(self.vapi.ppcli("show interface"))
Jan49c0fca2016-10-26 15:44:27 +0200465 self.logger.info(self.vapi.ppcli("show hardware"))
466 self.logger.info(self.vapi.ppcli("show error"))
467 self.logger.info(self.vapi.ppcli("show run"))
Klement Sekera10db26f2017-01-11 08:16:53 +0100468 self.registry.remove_vpp_config(self.logger)
Dave Wallace90c55722017-02-16 11:25:26 -0500469 # Save/Dump VPP api trace log
470 api_trace = "vpp_api_trace.%s.log" % self._testMethodName
471 tmp_api_trace = "/tmp/%s" % api_trace
472 vpp_api_trace_log = "%s/%s" % (self.tempdir, api_trace)
473 self.logger.info(self.vapi.ppcli("api trace save %s" % api_trace))
474 self.logger.info("Moving %s to %s\n" % (tmp_api_trace,
475 vpp_api_trace_log))
476 os.rename(tmp_api_trace, vpp_api_trace_log)
477 self.logger.info(self.vapi.ppcli("api trace dump %s" %
478 vpp_api_trace_log))
Klement Sekera1b686402017-03-02 11:29:19 +0100479 else:
480 self.registry.unregister_all(self.logger)
Damjan Marionf56b77a2016-10-03 19:44:57 +0200481
Damjan Marionf56b77a2016-10-03 19:44:57 +0200482 def setUp(self):
Klement Sekeraf62ae122016-10-11 11:47:09 +0200483 """ Clear trace before running each test"""
Klement Sekera909a6a12017-08-08 04:33:53 +0200484 self.reporter.send_keep_alive(self)
Klement Sekerab91017a2017-02-09 06:04:36 +0100485 self.logger.debug("--- setUp() for %s.%s(%s) called ---" %
486 (self.__class__.__name__, self._testMethodName,
487 self._testMethodDoc))
Klement Sekera0c1519b2016-12-08 05:03:32 +0100488 if self.vpp_dead:
489 raise Exception("VPP is dead when setting up the test")
Klement Sekeraacb9b8e2017-02-14 02:55:31 +0100490 self.sleep(.1, "during setUp")
Klement Sekerae4504c62016-12-08 10:16:41 +0100491 self.vpp_stdout_deque.append(
492 "--- test setUp() for %s.%s(%s) starts here ---\n" %
493 (self.__class__.__name__, self._testMethodName,
494 self._testMethodDoc))
495 self.vpp_stderr_deque.append(
496 "--- test setUp() for %s.%s(%s) starts here ---\n" %
497 (self.__class__.__name__, self._testMethodName,
498 self._testMethodDoc))
Klement Sekeraf62ae122016-10-11 11:47:09 +0200499 self.vapi.cli("clear trace")
500 # store the test instance inside the test class - so that objects
501 # holding the class can access instance methods (like assertEqual)
502 type(self).test_instance = self
Damjan Marionf56b77a2016-10-03 19:44:57 +0200503
Damjan Marionf56b77a2016-10-03 19:44:57 +0200504 @classmethod
Klement Sekera75e7d132017-09-20 08:26:30 +0200505 def pg_enable_capture(cls, interfaces=None):
Klement Sekeraf62ae122016-10-11 11:47:09 +0200506 """
507 Enable capture on packet-generator interfaces
Damjan Marionf56b77a2016-10-03 19:44:57 +0200508
Klement Sekera75e7d132017-09-20 08:26:30 +0200509 :param interfaces: iterable interface indexes (if None,
510 use self.pg_interfaces)
Damjan Marionf56b77a2016-10-03 19:44:57 +0200511
Klement Sekeraf62ae122016-10-11 11:47:09 +0200512 """
Klement Sekera75e7d132017-09-20 08:26:30 +0200513 if interfaces is None:
514 interfaces = cls.pg_interfaces
Klement Sekeraf62ae122016-10-11 11:47:09 +0200515 for i in interfaces:
516 i.enable_capture()
Damjan Marionf56b77a2016-10-03 19:44:57 +0200517
Damjan Marionf56b77a2016-10-03 19:44:57 +0200518 @classmethod
Klement Sekera9225dee2016-12-12 08:36:58 +0100519 def register_capture(cls, cap_name):
520 """ Register a capture in the testclass """
521 # add to the list of captures with current timestamp
522 cls._captures.append((time.time(), cap_name))
523 # filter out from zombies
524 cls._zombie_captures = [(stamp, name)
525 for (stamp, name) in cls._zombie_captures
526 if name != cap_name]
527
528 @classmethod
529 def pg_start(cls):
530 """ Remove any zombie captures and enable the packet generator """
531 # how long before capture is allowed to be deleted - otherwise vpp
532 # crashes - 100ms seems enough (this shouldn't be needed at all)
533 capture_ttl = 0.1
534 now = time.time()
535 for stamp, cap_name in cls._zombie_captures:
536 wait = stamp + capture_ttl - now
537 if wait > 0:
Klement Sekeraacb9b8e2017-02-14 02:55:31 +0100538 cls.sleep(wait, "before deleting capture %s" % cap_name)
Klement Sekera9225dee2016-12-12 08:36:58 +0100539 now = time.time()
540 cls.logger.debug("Removing zombie capture %s" % cap_name)
541 cls.vapi.cli('packet-generator delete %s' % cap_name)
542
Klement Sekeraf62ae122016-10-11 11:47:09 +0200543 cls.vapi.cli("trace add pg-input 50") # 50 is maximum
544 cls.vapi.cli('packet-generator enable')
Klement Sekera9225dee2016-12-12 08:36:58 +0100545 cls._zombie_captures = cls._captures
546 cls._captures = []
Damjan Marionf56b77a2016-10-03 19:44:57 +0200547
Damjan Marionf56b77a2016-10-03 19:44:57 +0200548 @classmethod
Klement Sekeraf62ae122016-10-11 11:47:09 +0200549 def create_pg_interfaces(cls, interfaces):
550 """
Matej Klotton8d8a1da2016-12-22 11:06:56 +0100551 Create packet-generator interfaces.
Damjan Marionf56b77a2016-10-03 19:44:57 +0200552
Matej Klotton8d8a1da2016-12-22 11:06:56 +0100553 :param interfaces: iterable indexes of the interfaces.
554 :returns: List of created interfaces.
Damjan Marionf56b77a2016-10-03 19:44:57 +0200555
Klement Sekeraf62ae122016-10-11 11:47:09 +0200556 """
557 result = []
558 for i in interfaces:
559 intf = VppPGInterface(cls, i)
560 setattr(cls, intf.name, intf)
561 result.append(intf)
562 cls.pg_interfaces = result
563 return result
564
Matej Klotton0178d522016-11-04 11:11:44 +0100565 @classmethod
566 def create_loopback_interfaces(cls, interfaces):
567 """
Matej Klotton8d8a1da2016-12-22 11:06:56 +0100568 Create loopback interfaces.
Matej Klotton0178d522016-11-04 11:11:44 +0100569
Matej Klotton8d8a1da2016-12-22 11:06:56 +0100570 :param interfaces: iterable indexes of the interfaces.
571 :returns: List of created interfaces.
Matej Klotton0178d522016-11-04 11:11:44 +0100572 """
573 result = []
574 for i in interfaces:
575 intf = VppLoInterface(cls, i)
576 setattr(cls, intf.name, intf)
577 result.append(intf)
578 cls.lo_interfaces = result
579 return result
580
Damjan Marionf56b77a2016-10-03 19:44:57 +0200581 @staticmethod
Klement Sekera75e7d132017-09-20 08:26:30 +0200582 def extend_packet(packet, size, padding=' '):
Klement Sekeraf62ae122016-10-11 11:47:09 +0200583 """
Klement Sekera75e7d132017-09-20 08:26:30 +0200584 Extend packet to given size by padding with spaces or custom padding
Klement Sekeraf62ae122016-10-11 11:47:09 +0200585 NOTE: Currently works only when Raw layer is present.
586
587 :param packet: packet
588 :param size: target size
Klement Sekera75e7d132017-09-20 08:26:30 +0200589 :param padding: padding used to extend the payload
Klement Sekeraf62ae122016-10-11 11:47:09 +0200590
591 """
Damjan Marionf56b77a2016-10-03 19:44:57 +0200592 packet_len = len(packet) + 4
593 extend = size - packet_len
594 if extend > 0:
Klement Sekera75e7d132017-09-20 08:26:30 +0200595 num = (extend / len(padding)) + 1
596 packet[Raw].load += (padding * num)[:extend]
Damjan Marionf56b77a2016-10-03 19:44:57 +0200597
Klement Sekeradab231a2016-12-21 08:50:14 +0100598 @classmethod
599 def reset_packet_infos(cls):
600 """ Reset the list of packet info objects and packet counts to zero """
601 cls._packet_infos = {}
602 cls._packet_count_for_dst_if_idx = {}
Klement Sekeraf62ae122016-10-11 11:47:09 +0200603
Klement Sekeradab231a2016-12-21 08:50:14 +0100604 @classmethod
605 def create_packet_info(cls, src_if, dst_if):
Klement Sekeraf62ae122016-10-11 11:47:09 +0200606 """
607 Create packet info object containing the source and destination indexes
608 and add it to the testcase's packet info list
609
Klement Sekeradab231a2016-12-21 08:50:14 +0100610 :param VppInterface src_if: source interface
611 :param VppInterface dst_if: destination interface
Klement Sekeraf62ae122016-10-11 11:47:09 +0200612
613 :returns: _PacketInfo object
614
615 """
Damjan Marionf56b77a2016-10-03 19:44:57 +0200616 info = _PacketInfo()
Klement Sekeradab231a2016-12-21 08:50:14 +0100617 info.index = len(cls._packet_infos)
618 info.src = src_if.sw_if_index
619 info.dst = dst_if.sw_if_index
620 if isinstance(dst_if, VppSubInterface):
621 dst_idx = dst_if.parent.sw_if_index
622 else:
623 dst_idx = dst_if.sw_if_index
624 if dst_idx in cls._packet_count_for_dst_if_idx:
625 cls._packet_count_for_dst_if_idx[dst_idx] += 1
626 else:
627 cls._packet_count_for_dst_if_idx[dst_idx] = 1
628 cls._packet_infos[info.index] = info
Damjan Marionf56b77a2016-10-03 19:44:57 +0200629 return info
Damjan Marionf56b77a2016-10-03 19:44:57 +0200630
Damjan Marionf56b77a2016-10-03 19:44:57 +0200631 @staticmethod
632 def info_to_payload(info):
Klement Sekeraf62ae122016-10-11 11:47:09 +0200633 """
634 Convert _PacketInfo object to packet payload
635
636 :param info: _PacketInfo object
637
638 :returns: string containing serialized data from packet info
639 """
Pavel Kotucek59dda062017-03-02 15:22:47 +0100640 return "%d %d %d %d %d" % (info.index, info.src, info.dst,
641 info.ip, info.proto)
Damjan Marionf56b77a2016-10-03 19:44:57 +0200642
Damjan Marionf56b77a2016-10-03 19:44:57 +0200643 @staticmethod
644 def payload_to_info(payload):
Klement Sekeraf62ae122016-10-11 11:47:09 +0200645 """
646 Convert packet payload to _PacketInfo object
647
648 :param payload: packet payload
649
650 :returns: _PacketInfo object containing de-serialized data from payload
651
652 """
Damjan Marionf56b77a2016-10-03 19:44:57 +0200653 numbers = payload.split()
654 info = _PacketInfo()
655 info.index = int(numbers[0])
656 info.src = int(numbers[1])
657 info.dst = int(numbers[2])
Pavel Kotucek59dda062017-03-02 15:22:47 +0100658 info.ip = int(numbers[3])
659 info.proto = int(numbers[4])
Damjan Marionf56b77a2016-10-03 19:44:57 +0200660 return info
Damjan Marionf56b77a2016-10-03 19:44:57 +0200661
Damjan Marionf56b77a2016-10-03 19:44:57 +0200662 def get_next_packet_info(self, info):
Klement Sekeraf62ae122016-10-11 11:47:09 +0200663 """
664 Iterate over the packet info list stored in the testcase
665 Start iteration with first element if info is None
666 Continue based on index in info if info is specified
667
668 :param info: info or None
669 :returns: next info in list or None if no more infos
670 """
Damjan Marionf56b77a2016-10-03 19:44:57 +0200671 if info is None:
672 next_index = 0
673 else:
674 next_index = info.index + 1
Klement Sekeradab231a2016-12-21 08:50:14 +0100675 if next_index == len(self._packet_infos):
Damjan Marionf56b77a2016-10-03 19:44:57 +0200676 return None
677 else:
Klement Sekeradab231a2016-12-21 08:50:14 +0100678 return self._packet_infos[next_index]
Damjan Marionf56b77a2016-10-03 19:44:57 +0200679
Klement Sekeraf62ae122016-10-11 11:47:09 +0200680 def get_next_packet_info_for_interface(self, src_index, info):
681 """
682 Search the packet info list for the next packet info with same source
683 interface index
684
685 :param src_index: source interface index to search for
686 :param info: packet info - where to start the search
687 :returns: packet info or None
688
689 """
Damjan Marionf56b77a2016-10-03 19:44:57 +0200690 while True:
691 info = self.get_next_packet_info(info)
692 if info is None:
693 return None
Klement Sekeraf62ae122016-10-11 11:47:09 +0200694 if info.src == src_index:
Damjan Marionf56b77a2016-10-03 19:44:57 +0200695 return info
Damjan Marionf56b77a2016-10-03 19:44:57 +0200696
Klement Sekeraf62ae122016-10-11 11:47:09 +0200697 def get_next_packet_info_for_interface2(self, src_index, dst_index, info):
698 """
699 Search the packet info list for the next packet info with same source
700 and destination interface indexes
701
702 :param src_index: source interface index to search for
703 :param dst_index: destination interface index to search for
704 :param info: packet info - where to start the search
705 :returns: packet info or None
706
707 """
Damjan Marionf56b77a2016-10-03 19:44:57 +0200708 while True:
Klement Sekeraf62ae122016-10-11 11:47:09 +0200709 info = self.get_next_packet_info_for_interface(src_index, info)
Damjan Marionf56b77a2016-10-03 19:44:57 +0200710 if info is None:
711 return None
Klement Sekeraf62ae122016-10-11 11:47:09 +0200712 if info.dst == dst_index:
Damjan Marionf56b77a2016-10-03 19:44:57 +0200713 return info
Damjan Marionf56b77a2016-10-03 19:44:57 +0200714
Klement Sekera0e3c0de2016-09-29 14:43:44 +0200715 def assert_equal(self, real_value, expected_value, name_or_class=None):
716 if name_or_class is None:
Klement Sekera239790f2017-02-16 10:53:53 +0100717 self.assertEqual(real_value, expected_value)
Klement Sekera0e3c0de2016-09-29 14:43:44 +0200718 return
719 try:
720 msg = "Invalid %s: %d('%s') does not match expected value %d('%s')"
721 msg = msg % (getdoc(name_or_class).strip(),
722 real_value, str(name_or_class(real_value)),
723 expected_value, str(name_or_class(expected_value)))
724 except:
725 msg = "Invalid %s: %s does not match expected value %s" % (
726 name_or_class, real_value, expected_value)
727
728 self.assertEqual(real_value, expected_value, msg)
729
Klement Sekerab17dd962017-01-09 07:43:48 +0100730 def assert_in_range(self,
731 real_value,
732 expected_min,
733 expected_max,
734 name=None):
Klement Sekera0e3c0de2016-09-29 14:43:44 +0200735 if name is None:
736 msg = None
737 else:
738 msg = "Invalid %s: %s out of range <%s,%s>" % (
739 name, real_value, expected_min, expected_max)
740 self.assertTrue(expected_min <= real_value <= expected_max, msg)
741
Klement Sekeraacb9b8e2017-02-14 02:55:31 +0100742 @classmethod
743 def sleep(cls, timeout, remark=None):
744 if hasattr(cls, 'logger'):
Klement Sekera3cfa5582017-04-19 07:10:58 +0000745 cls.logger.debug("Starting sleep for %ss (%s)" % (timeout, remark))
746 before = time.time()
Klement Sekeraa57a9702017-02-02 06:58:07 +0100747 time.sleep(timeout)
Klement Sekera3cfa5582017-04-19 07:10:58 +0000748 after = time.time()
749 if after - before > 2 * timeout:
Klement Sekera60c12232017-07-18 10:33:06 +0200750 cls.logger.error("unexpected time.sleep() result - "
751 "slept for %ss instead of ~%ss!" % (
752 after - before, timeout))
Klement Sekera3cfa5582017-04-19 07:10:58 +0000753 if hasattr(cls, 'logger'):
754 cls.logger.debug(
755 "Finished sleep (%s) - slept %ss (wanted %ss)" % (
756 remark, after - before, timeout))
Klement Sekeraa57a9702017-02-02 06:58:07 +0100757
Neale Ranns52fae862018-01-08 04:41:42 -0800758 def send_and_assert_no_replies(self, intf, pkts, remark=""):
759 self.vapi.cli("clear trace")
760 intf.add_stream(pkts)
761 self.pg_enable_capture(self.pg_interfaces)
762 self.pg_start()
763 timeout = 1
764 for i in self.pg_interfaces:
765 i.get_capture(0, timeout=timeout)
766 i.assert_nothing_captured(remark=remark)
767 timeout = 0.1
768
769 def send_and_expect(self, input, pkts, output):
770 self.vapi.cli("clear trace")
771 input.add_stream(pkts)
772 self.pg_enable_capture(self.pg_interfaces)
773 self.pg_start()
774 rx = output.get_capture(len(pkts))
775 return rx
776
Damjan Marionf56b77a2016-10-03 19:44:57 +0200777
Klement Sekera87134932017-03-07 11:39:27 +0100778class TestCasePrinter(object):
779 _shared_state = {}
780
781 def __init__(self):
782 self.__dict__ = self._shared_state
783 if not hasattr(self, "_test_case_set"):
784 self._test_case_set = set()
785
786 def print_test_case_heading_if_first_time(self, case):
787 if case.__class__ not in self._test_case_set:
788 print(double_line_delim)
789 print(colorize(getdoc(case.__class__).splitlines()[0], YELLOW))
790 print(double_line_delim)
791 self._test_case_set.add(case.__class__)
792
793
Damjan Marionf56b77a2016-10-03 19:44:57 +0200794class VppTestResult(unittest.TestResult):
Klement Sekeraf62ae122016-10-11 11:47:09 +0200795 """
796 @property result_string
797 String variable to store the test case result string.
798 @property errors
799 List variable containing 2-tuples of TestCase instances and strings
800 holding formatted tracebacks. Each tuple represents a test which
801 raised an unexpected exception.
802 @property failures
803 List variable containing 2-tuples of TestCase instances and strings
804 holding formatted tracebacks. Each tuple represents a test where
805 a failure was explicitly signalled using the TestCase.assert*()
806 methods.
807 """
808
Damjan Marionf56b77a2016-10-03 19:44:57 +0200809 def __init__(self, stream, descriptions, verbosity):
Klement Sekeraf62ae122016-10-11 11:47:09 +0200810 """
Klement Sekerada505f62017-01-04 12:58:53 +0100811 :param stream File descriptor to store where to report test results.
812 Set to the standard error stream by default.
813 :param descriptions Boolean variable to store information if to use
814 test case descriptions.
Klement Sekeraf62ae122016-10-11 11:47:09 +0200815 :param verbosity Integer variable to store required verbosity level.
816 """
Damjan Marionf56b77a2016-10-03 19:44:57 +0200817 unittest.TestResult.__init__(self, stream, descriptions, verbosity)
818 self.stream = stream
819 self.descriptions = descriptions
820 self.verbosity = verbosity
821 self.result_string = None
Klement Sekera87134932017-03-07 11:39:27 +0100822 self.printer = TestCasePrinter()
Damjan Marionf56b77a2016-10-03 19:44:57 +0200823
Damjan Marionf56b77a2016-10-03 19:44:57 +0200824 def addSuccess(self, test):
Klement Sekeraf62ae122016-10-11 11:47:09 +0200825 """
826 Record a test succeeded result
827
828 :param test:
829
830 """
Klement Sekerab91017a2017-02-09 06:04:36 +0100831 if hasattr(test, 'logger'):
832 test.logger.debug("--- addSuccess() %s.%s(%s) called"
833 % (test.__class__.__name__,
834 test._testMethodName,
835 test._testMethodDoc))
Damjan Marionf56b77a2016-10-03 19:44:57 +0200836 unittest.TestResult.addSuccess(self, test)
Klement Sekera277b89c2016-10-28 13:20:27 +0200837 self.result_string = colorize("OK", GREEN)
Damjan Marionf56b77a2016-10-03 19:44:57 +0200838
Klement Sekeraf62ae122016-10-11 11:47:09 +0200839 def addSkip(self, test, reason):
840 """
841 Record a test skipped.
842
843 :param test:
844 :param reason:
845
846 """
Klement Sekerab91017a2017-02-09 06:04:36 +0100847 if hasattr(test, 'logger'):
848 test.logger.debug("--- addSkip() %s.%s(%s) called, reason is %s"
849 % (test.__class__.__name__,
850 test._testMethodName,
851 test._testMethodDoc,
852 reason))
Klement Sekeraf62ae122016-10-11 11:47:09 +0200853 unittest.TestResult.addSkip(self, test, reason)
Klement Sekera277b89c2016-10-28 13:20:27 +0200854 self.result_string = colorize("SKIP", YELLOW)
Klement Sekeraf62ae122016-10-11 11:47:09 +0200855
Klement Sekeraf413bef2017-08-15 07:09:02 +0200856 def symlink_failed(self, test):
857 logger = None
858 if hasattr(test, 'logger'):
859 logger = test.logger
860 if hasattr(test, 'tempdir'):
861 try:
862 failed_dir = os.getenv('VPP_TEST_FAILED_DIR')
863 link_path = '%s/%s-FAILED' % (failed_dir,
864 test.tempdir.split("/")[-1])
865 if logger:
866 logger.debug("creating a link to the failed test")
867 logger.debug("os.symlink(%s, %s)" %
868 (test.tempdir, link_path))
869 os.symlink(test.tempdir, link_path)
870 except Exception as e:
871 if logger:
872 logger.error(e)
873
Klement Sekeradf2b9802017-10-05 10:26:03 +0200874 def send_failure_through_pipe(self, test):
875 if hasattr(self, 'test_framework_failed_pipe'):
876 pipe = self.test_framework_failed_pipe
877 if pipe:
878 pipe.send(test.__class__)
879
Damjan Marionf56b77a2016-10-03 19:44:57 +0200880 def addFailure(self, test, err):
Klement Sekeraf62ae122016-10-11 11:47:09 +0200881 """
882 Record a test failed result
883
884 :param test:
885 :param err: error message
886
887 """
Klement Sekerab91017a2017-02-09 06:04:36 +0100888 if hasattr(test, 'logger'):
889 test.logger.debug("--- addFailure() %s.%s(%s) called, err is %s"
890 % (test.__class__.__name__,
891 test._testMethodName,
892 test._testMethodDoc, err))
893 test.logger.debug("formatted exception is:\n%s" %
894 "".join(format_exception(*err)))
Damjan Marionf56b77a2016-10-03 19:44:57 +0200895 unittest.TestResult.addFailure(self, test, err)
Klement Sekeraf62ae122016-10-11 11:47:09 +0200896 if hasattr(test, 'tempdir'):
Klement Sekera277b89c2016-10-28 13:20:27 +0200897 self.result_string = colorize("FAIL", RED) + \
Klement Sekeraf62ae122016-10-11 11:47:09 +0200898 ' [ temp dir used by test case: ' + test.tempdir + ' ]'
Klement Sekeraf413bef2017-08-15 07:09:02 +0200899 self.symlink_failed(test)
Klement Sekeraf62ae122016-10-11 11:47:09 +0200900 else:
Klement Sekera277b89c2016-10-28 13:20:27 +0200901 self.result_string = colorize("FAIL", RED) + ' [no temp dir]'
Damjan Marionf56b77a2016-10-03 19:44:57 +0200902
Klement Sekeradf2b9802017-10-05 10:26:03 +0200903 self.send_failure_through_pipe(test)
904
Damjan Marionf56b77a2016-10-03 19:44:57 +0200905 def addError(self, test, err):
Klement Sekeraf62ae122016-10-11 11:47:09 +0200906 """
907 Record a test error result
Damjan Marionf56b77a2016-10-03 19:44:57 +0200908
Klement Sekeraf62ae122016-10-11 11:47:09 +0200909 :param test:
910 :param err: error message
911
912 """
Klement Sekerab91017a2017-02-09 06:04:36 +0100913 if hasattr(test, 'logger'):
914 test.logger.debug("--- addError() %s.%s(%s) called, err is %s"
915 % (test.__class__.__name__,
916 test._testMethodName,
917 test._testMethodDoc, err))
918 test.logger.debug("formatted exception is:\n%s" %
919 "".join(format_exception(*err)))
Klement Sekeraf62ae122016-10-11 11:47:09 +0200920 unittest.TestResult.addError(self, test, err)
921 if hasattr(test, 'tempdir'):
Klement Sekera277b89c2016-10-28 13:20:27 +0200922 self.result_string = colorize("ERROR", RED) + \
Klement Sekeraf62ae122016-10-11 11:47:09 +0200923 ' [ temp dir used by test case: ' + test.tempdir + ' ]'
Klement Sekeraf413bef2017-08-15 07:09:02 +0200924 self.symlink_failed(test)
Klement Sekeraf62ae122016-10-11 11:47:09 +0200925 else:
Klement Sekera277b89c2016-10-28 13:20:27 +0200926 self.result_string = colorize("ERROR", RED) + ' [no temp dir]'
Klement Sekeraf62ae122016-10-11 11:47:09 +0200927
Klement Sekeradf2b9802017-10-05 10:26:03 +0200928 self.send_failure_through_pipe(test)
929
Damjan Marionf56b77a2016-10-03 19:44:57 +0200930 def getDescription(self, test):
Klement Sekeraf62ae122016-10-11 11:47:09 +0200931 """
932 Get test description
933
934 :param test:
935 :returns: test description
936
937 """
Damjan Marionf56b77a2016-10-03 19:44:57 +0200938 # TODO: if none print warning not raise exception
939 short_description = test.shortDescription()
940 if self.descriptions and short_description:
941 return short_description
942 else:
943 return str(test)
Damjan Marionf56b77a2016-10-03 19:44:57 +0200944
Damjan Marionf56b77a2016-10-03 19:44:57 +0200945 def startTest(self, test):
Klement Sekeraf62ae122016-10-11 11:47:09 +0200946 """
947 Start a test
948
949 :param test:
950
951 """
Klement Sekera87134932017-03-07 11:39:27 +0100952 self.printer.print_test_case_heading_if_first_time(test)
Damjan Marionf56b77a2016-10-03 19:44:57 +0200953 unittest.TestResult.startTest(self, test)
954 if self.verbosity > 0:
Klement Sekeraf62ae122016-10-11 11:47:09 +0200955 self.stream.writeln(
956 "Starting " + self.getDescription(test) + " ...")
957 self.stream.writeln(single_line_delim)
Damjan Marionf56b77a2016-10-03 19:44:57 +0200958
Damjan Marionf56b77a2016-10-03 19:44:57 +0200959 def stopTest(self, test):
Klement Sekeraf62ae122016-10-11 11:47:09 +0200960 """
961 Stop a test
962
963 :param test:
964
965 """
Damjan Marionf56b77a2016-10-03 19:44:57 +0200966 unittest.TestResult.stopTest(self, test)
967 if self.verbosity > 0:
Klement Sekeraf62ae122016-10-11 11:47:09 +0200968 self.stream.writeln(single_line_delim)
Klement Sekera52e84f32017-01-13 07:25:25 +0100969 self.stream.writeln("%-73s%s" % (self.getDescription(test),
Klement Sekerada505f62017-01-04 12:58:53 +0100970 self.result_string))
Klement Sekeraf62ae122016-10-11 11:47:09 +0200971 self.stream.writeln(single_line_delim)
Damjan Marionf56b77a2016-10-03 19:44:57 +0200972 else:
Klement Sekera52e84f32017-01-13 07:25:25 +0100973 self.stream.writeln("%-73s%s" % (self.getDescription(test),
Klement Sekerada505f62017-01-04 12:58:53 +0100974 self.result_string))
Damjan Marionf56b77a2016-10-03 19:44:57 +0200975
Damjan Marionf56b77a2016-10-03 19:44:57 +0200976 def printErrors(self):
Klement Sekeraf62ae122016-10-11 11:47:09 +0200977 """
978 Print errors from running the test case
979 """
Damjan Marionf56b77a2016-10-03 19:44:57 +0200980 self.stream.writeln()
981 self.printErrorList('ERROR', self.errors)
982 self.printErrorList('FAIL', self.failures)
Damjan Marionf56b77a2016-10-03 19:44:57 +0200983
Damjan Marionf56b77a2016-10-03 19:44:57 +0200984 def printErrorList(self, flavour, errors):
Klement Sekeraf62ae122016-10-11 11:47:09 +0200985 """
986 Print error list to the output stream together with error type
987 and test case description.
988
989 :param flavour: error type
990 :param errors: iterable errors
991
992 """
Damjan Marionf56b77a2016-10-03 19:44:57 +0200993 for test, err in errors:
Klement Sekeraf62ae122016-10-11 11:47:09 +0200994 self.stream.writeln(double_line_delim)
995 self.stream.writeln("%s: %s" %
996 (flavour, self.getDescription(test)))
997 self.stream.writeln(single_line_delim)
Damjan Marionf56b77a2016-10-03 19:44:57 +0200998 self.stream.writeln("%s" % err)
Damjan Marionf56b77a2016-10-03 19:44:57 +0200999
1000
Klement Sekeradf2b9802017-10-05 10:26:03 +02001001class Filter_by_test_option:
1002 def __init__(self, filter_file_name, filter_class_name, filter_func_name):
1003 self.filter_file_name = filter_file_name
1004 self.filter_class_name = filter_class_name
1005 self.filter_func_name = filter_func_name
1006
1007 def __call__(self, file_name, class_name, func_name):
1008 if self.filter_file_name and file_name != self.filter_file_name:
1009 return False
1010 if self.filter_class_name and class_name != self.filter_class_name:
1011 return False
1012 if self.filter_func_name and func_name != self.filter_func_name:
1013 return False
1014 return True
1015
1016
Damjan Marionf56b77a2016-10-03 19:44:57 +02001017class VppTestRunner(unittest.TextTestRunner):
Klement Sekeraf62ae122016-10-11 11:47:09 +02001018 """
Klement Sekera104543f2017-02-03 07:29:43 +01001019 A basic test runner implementation which prints results to standard error.
Klement Sekeraf62ae122016-10-11 11:47:09 +02001020 """
1021 @property
1022 def resultclass(self):
1023 """Class maintaining the results of the tests"""
1024 return VppTestResult
Damjan Marionf56b77a2016-10-03 19:44:57 +02001025
Klement Sekeradf2b9802017-10-05 10:26:03 +02001026 def __init__(self, keep_alive_pipe=None, failed_pipe=None,
1027 stream=sys.stderr, descriptions=True,
Klement Sekera3f6ff192017-08-11 06:56:05 +02001028 verbosity=1, failfast=False, buffer=False, resultclass=None):
Klement Sekera7a161da2017-01-17 13:42:48 +01001029 # ignore stream setting here, use hard-coded stdout to be in sync
1030 # with prints from VppTestCase methods ...
1031 super(VppTestRunner, self).__init__(sys.stdout, descriptions,
1032 verbosity, failfast, buffer,
1033 resultclass)
Klement Sekera909a6a12017-08-08 04:33:53 +02001034 reporter = KeepAliveReporter()
Klement Sekeradf2b9802017-10-05 10:26:03 +02001035 reporter.pipe = keep_alive_pipe
1036 # this is super-ugly, but very simple to implement and works as long
1037 # as we run only one test at the same time
1038 VppTestResult.test_framework_failed_pipe = failed_pipe
Klement Sekera7a161da2017-01-17 13:42:48 +01001039
Klement Sekera104543f2017-02-03 07:29:43 +01001040 test_option = "TEST"
1041
1042 def parse_test_option(self):
1043 try:
1044 f = os.getenv(self.test_option)
1045 except:
1046 f = None
1047 filter_file_name = None
1048 filter_class_name = None
1049 filter_func_name = None
1050 if f:
1051 if '.' in f:
1052 parts = f.split('.')
1053 if len(parts) > 3:
1054 raise Exception("Unrecognized %s option: %s" %
1055 (self.test_option, f))
1056 if len(parts) > 2:
1057 if parts[2] not in ('*', ''):
1058 filter_func_name = parts[2]
1059 if parts[1] not in ('*', ''):
1060 filter_class_name = parts[1]
1061 if parts[0] not in ('*', ''):
1062 if parts[0].startswith('test_'):
1063 filter_file_name = parts[0]
1064 else:
1065 filter_file_name = 'test_%s' % parts[0]
1066 else:
1067 if f.startswith('test_'):
1068 filter_file_name = f
1069 else:
1070 filter_file_name = 'test_%s' % f
1071 return filter_file_name, filter_class_name, filter_func_name
1072
Klement Sekeradf2b9802017-10-05 10:26:03 +02001073 @staticmethod
1074 def filter_tests(tests, filter_cb):
Klement Sekera104543f2017-02-03 07:29:43 +01001075 result = unittest.suite.TestSuite()
1076 for t in tests:
1077 if isinstance(t, unittest.suite.TestSuite):
1078 # this is a bunch of tests, recursively filter...
Klement Sekeradf2b9802017-10-05 10:26:03 +02001079 x = filter_tests(t, filter_cb)
Klement Sekera104543f2017-02-03 07:29:43 +01001080 if x.countTestCases() > 0:
1081 result.addTest(x)
1082 elif isinstance(t, unittest.TestCase):
1083 # this is a single test
1084 parts = t.id().split('.')
1085 # t.id() for common cases like this:
1086 # test_classifier.TestClassifier.test_acl_ip
1087 # apply filtering only if it is so
1088 if len(parts) == 3:
Klement Sekeradf2b9802017-10-05 10:26:03 +02001089 if not filter_cb(parts[0], parts[1], parts[2]):
Klement Sekera104543f2017-02-03 07:29:43 +01001090 continue
1091 result.addTest(t)
1092 else:
1093 # unexpected object, don't touch it
1094 result.addTest(t)
1095 return result
1096
Damjan Marionf56b77a2016-10-03 19:44:57 +02001097 def run(self, test):
Klement Sekeraf62ae122016-10-11 11:47:09 +02001098 """
1099 Run the tests
1100
1101 :param test:
1102
1103 """
Klement Sekera3658adc2017-06-07 08:19:47 +02001104 faulthandler.enable() # emit stack trace to stderr if killed by signal
Klement Sekeraf62ae122016-10-11 11:47:09 +02001105 print("Running tests using custom test runner") # debug message
Klement Sekera104543f2017-02-03 07:29:43 +01001106 filter_file, filter_class, filter_func = self.parse_test_option()
1107 print("Active filters: file=%s, class=%s, function=%s" % (
1108 filter_file, filter_class, filter_func))
Klement Sekeradf2b9802017-10-05 10:26:03 +02001109 filter_cb = Filter_by_test_option(
1110 filter_file, filter_class, filter_func)
1111 filtered = self.filter_tests(test, filter_cb)
Klement Sekera104543f2017-02-03 07:29:43 +01001112 print("%s out of %s tests match specified filters" % (
1113 filtered.countTestCases(), test.countTestCases()))
Klement Sekera3747c752017-04-10 06:30:17 +02001114 if not running_extended_tests():
1115 print("Not running extended tests (some tests will be skipped)")
Klement Sekera104543f2017-02-03 07:29:43 +01001116 return super(VppTestRunner, self).run(filtered)
Neale Ranns812ed392017-10-16 04:20:13 -07001117
1118
1119class Worker(Thread):
1120 def __init__(self, args, logger):
1121 self.logger = logger
1122 self.args = args
1123 self.result = None
1124 super(Worker, self).__init__()
1125
1126 def run(self):
1127 executable = self.args[0]
1128 self.logger.debug("Running executable w/args `%s'" % self.args)
1129 env = os.environ.copy()
1130 env["CK_LOG_FILE_NAME"] = "-"
1131 self.process = subprocess.Popen(
1132 self.args, shell=False, env=env, preexec_fn=os.setpgrp,
1133 stdout=subprocess.PIPE, stderr=subprocess.PIPE)
1134 out, err = self.process.communicate()
1135 self.logger.debug("Finished running `%s'" % executable)
1136 self.logger.info("Return code is `%s'" % self.process.returncode)
1137 self.logger.info(single_line_delim)
1138 self.logger.info("Executable `%s' wrote to stdout:" % executable)
1139 self.logger.info(single_line_delim)
1140 self.logger.info(out)
1141 self.logger.info(single_line_delim)
1142 self.logger.info("Executable `%s' wrote to stderr:" % executable)
1143 self.logger.info(single_line_delim)
1144 self.logger.error(err)
1145 self.logger.info(single_line_delim)
1146 self.result = self.process.returncode