2019-10-31 13:31:07 -05:00
|
|
|
#!/usr/bin/env python3
|
2018-11-24 21:46:05 -08:00
|
|
|
|
|
|
|
import abc
|
2017-01-31 10:39:16 +01:00
|
|
|
import unittest
|
|
|
|
|
2018-11-23 03:20:21 -08:00
|
|
|
from scapy.fields import BitField, ByteField, FlagsField, IntField
|
|
|
|
from scapy.packet import bind_layers, Packet, Raw
|
2017-01-31 10:39:16 +01:00
|
|
|
from scapy.layers.inet import IP, UDP, Ether
|
2018-11-23 03:20:21 -08:00
|
|
|
from scapy.layers.inet6 import IPv6
|
2017-01-31 10:39:16 +01:00
|
|
|
|
2023-08-31 00:47:44 -04:00
|
|
|
from framework import VppTestCase
|
|
|
|
from asfframework import VppTestRunner
|
|
|
|
from lisp import (
|
2022-04-26 19:02:15 +02:00
|
|
|
VppLocalMapping,
|
|
|
|
VppLispAdjacency,
|
|
|
|
VppLispLocator,
|
|
|
|
VppLispLocatorSet,
|
|
|
|
VppRemoteMapping,
|
|
|
|
LispRemoteLocator,
|
|
|
|
)
|
2020-12-04 17:50:26 -05:00
|
|
|
from util import ppp
|
2024-03-11 10:38:46 +00:00
|
|
|
from config import config
|
2018-11-23 03:20:21 -08:00
|
|
|
|
|
|
|
# From py_lispnetworking.lisp.py: # GNU General Public License v2.0
|
|
|
|
|
|
|
|
|
|
|
|
class LISP_GPE_Header(Packet):
|
|
|
|
name = "LISP GPE Header"
|
|
|
|
fields_desc = [
|
|
|
|
FlagsField("gpe_flags", None, 6, ["N", "L", "E", "V", "I", "P"]),
|
|
|
|
BitField("reserved", 0, 18),
|
|
|
|
ByteField("next_proto", 0),
|
|
|
|
IntField("iid", 0),
|
|
|
|
]
|
2022-04-26 19:02:15 +02:00
|
|
|
|
|
|
|
|
2018-11-23 03:20:21 -08:00
|
|
|
bind_layers(UDP, LISP_GPE_Header, dport=4341)
|
|
|
|
bind_layers(UDP, LISP_GPE_Header, sport=4341)
|
|
|
|
bind_layers(LISP_GPE_Header, IP, next_proto=1)
|
|
|
|
bind_layers(LISP_GPE_Header, IPv6, next_proto=2)
|
|
|
|
bind_layers(LISP_GPE_Header, Ether, next_proto=3)
|
2017-01-31 10:39:16 +01:00
|
|
|
|
|
|
|
|
2020-12-04 17:50:26 -05:00
|
|
|
class ForeignAddressFactory(object):
|
|
|
|
count = 0
|
|
|
|
prefix_len = 24
|
2022-04-26 19:02:15 +02:00
|
|
|
net_template = "10.10.10.{}"
|
|
|
|
net = net_template.format(0) + "/" + str(prefix_len)
|
2020-12-04 17:50:26 -05:00
|
|
|
|
|
|
|
def get_ip4(self):
|
|
|
|
if self.count > 255:
|
|
|
|
raise Exception("Network host address exhaustion")
|
|
|
|
self.count += 1
|
|
|
|
return self.net_template.format(self.count)
|
|
|
|
|
|
|
|
|
2020-12-03 00:42:46 -05:00
|
|
|
class Driver(metaclass=abc.ABCMeta):
|
2022-04-26 19:02:15 +02:00
|
|
|
config_order = [
|
|
|
|
"locator-sets",
|
|
|
|
"locators",
|
|
|
|
"local-mappings",
|
|
|
|
"remote-mappings",
|
|
|
|
"adjacencies",
|
|
|
|
]
|
2017-01-31 10:39:16 +01:00
|
|
|
|
|
|
|
""" Basic class for data driven testing """
|
2022-04-26 19:02:15 +02:00
|
|
|
|
2017-01-31 10:39:16 +01:00
|
|
|
def __init__(self, test, test_cases):
|
|
|
|
self._test_cases = test_cases
|
|
|
|
self._test = test
|
|
|
|
|
|
|
|
@property
|
|
|
|
def test_cases(self):
|
|
|
|
return self._test_cases
|
|
|
|
|
|
|
|
@property
|
|
|
|
def test(self):
|
|
|
|
return self._test
|
|
|
|
|
2022-04-26 19:02:15 +02:00
|
|
|
def create_packet(self, src_if, dst_if, deid, payload=""):
|
2017-01-31 10:39:16 +01:00
|
|
|
"""
|
|
|
|
Create IPv4 packet
|
|
|
|
|
|
|
|
param: src_if
|
|
|
|
param: dst_if
|
|
|
|
"""
|
2022-04-26 19:02:15 +02:00
|
|
|
packet = (
|
|
|
|
Ether(dst=src_if.local_mac, src=src_if.remote_mac)
|
|
|
|
/ IP(src=src_if.remote_ip4, dst=deid)
|
|
|
|
/ Raw(payload)
|
|
|
|
)
|
2017-01-31 10:39:16 +01:00
|
|
|
return packet
|
|
|
|
|
2018-11-24 21:46:05 -08:00
|
|
|
@abc.abstractmethod
|
2017-01-31 10:39:16 +01:00
|
|
|
def run(self):
|
2022-04-26 19:02:15 +02:00
|
|
|
"""testing procedure"""
|
2017-01-31 10:39:16 +01:00
|
|
|
pass
|
|
|
|
|
|
|
|
|
|
|
|
class SimpleDriver(Driver):
|
2022-04-26 19:02:15 +02:00
|
|
|
"""Implements simple test procedure"""
|
|
|
|
|
2017-01-31 10:39:16 +01:00
|
|
|
def __init__(self, test, test_cases):
|
|
|
|
super(SimpleDriver, self).__init__(test, test_cases)
|
|
|
|
|
|
|
|
def verify_capture(self, src_loc, dst_loc, capture):
|
|
|
|
"""
|
|
|
|
Verify captured packet
|
|
|
|
|
|
|
|
:param src_loc: source locator address
|
|
|
|
:param dst_loc: destination locator address
|
|
|
|
:param capture: list of captured packets
|
|
|
|
"""
|
2022-04-26 19:02:15 +02:00
|
|
|
self.test.assertEqual(
|
|
|
|
len(capture),
|
|
|
|
1,
|
|
|
|
"Unexpected number of "
|
|
|
|
"packets! Expected 1 but {} received".format(len(capture)),
|
|
|
|
)
|
2017-01-31 10:39:16 +01:00
|
|
|
packet = capture[0]
|
|
|
|
try:
|
|
|
|
ip_hdr = packet[IP]
|
|
|
|
# assert the values match
|
|
|
|
self.test.assertEqual(ip_hdr.src, src_loc, "IP source address")
|
2022-04-26 19:02:15 +02:00
|
|
|
self.test.assertEqual(ip_hdr.dst, dst_loc, "IP destination address")
|
2017-01-31 10:39:16 +01:00
|
|
|
gpe_hdr = packet[LISP_GPE_Header]
|
2022-04-26 19:02:15 +02:00
|
|
|
self.test.assertEqual(gpe_hdr.next_proto, 1, "next_proto is not ipv4!")
|
2017-01-31 10:39:16 +01:00
|
|
|
ih = gpe_hdr[IP]
|
2022-04-26 19:02:15 +02:00
|
|
|
self.test.assertEqual(
|
|
|
|
ih.src, self.test.pg0.remote_ip4, "unexpected source EID!"
|
|
|
|
)
|
|
|
|
self.test.assertEqual(ih.dst, self.test.deid_ip4, "unexpected dest EID!")
|
2017-01-31 10:39:16 +01:00
|
|
|
except:
|
2022-04-26 19:02:15 +02:00
|
|
|
self.test.logger.error(ppp("Unexpected or invalid packet:", packet))
|
2017-01-31 10:39:16 +01:00
|
|
|
raise
|
|
|
|
|
|
|
|
def configure_tc(self, tc):
|
|
|
|
for config_item in self.config_order:
|
|
|
|
for vpp_object in tc[config_item]:
|
|
|
|
vpp_object.add_vpp_config()
|
|
|
|
|
|
|
|
def run(self, dest):
|
2022-04-26 19:02:15 +02:00
|
|
|
"""Send traffic for each test case and verify that it
|
|
|
|
is encapsulated"""
|
2017-01-31 10:39:16 +01:00
|
|
|
for tc in enumerate(self.test_cases):
|
2022-04-26 19:02:15 +02:00
|
|
|
self.test.logger.info("Running {}".format(tc[1]["name"]))
|
2017-01-31 10:39:16 +01:00
|
|
|
self.configure_tc(tc[1])
|
|
|
|
|
2022-04-26 19:02:15 +02:00
|
|
|
packet = self.create_packet(self.test.pg0, self.test.pg1, dest, "data")
|
2017-01-31 10:39:16 +01:00
|
|
|
self.test.pg0.add_stream(packet)
|
|
|
|
self.test.pg0.enable_capture()
|
|
|
|
self.test.pg1.enable_capture()
|
|
|
|
self.test.pg_start()
|
|
|
|
capture = self.test.pg1.get_capture(1)
|
2022-04-26 19:02:15 +02:00
|
|
|
self.verify_capture(
|
|
|
|
self.test.pg1.local_ip4, self.test.pg1.remote_ip4, capture
|
|
|
|
)
|
2017-01-31 10:39:16 +01:00
|
|
|
self.test.pg0.assert_nothing_captured()
|
|
|
|
|
|
|
|
|
2024-03-11 10:38:46 +00:00
|
|
|
@unittest.skipIf("lisp" in config.excluded_plugins, "Exclude LISP plugin tests")
|
2017-01-31 10:39:16 +01:00
|
|
|
class TestLisp(VppTestCase):
|
2022-04-26 19:02:15 +02:00
|
|
|
"""Basic LISP test"""
|
2017-01-31 10:39:16 +01:00
|
|
|
|
|
|
|
@classmethod
|
|
|
|
def setUpClass(cls):
|
|
|
|
super(TestLisp, cls).setUpClass()
|
|
|
|
cls.faf = ForeignAddressFactory()
|
|
|
|
cls.create_pg_interfaces(range(2)) # create pg0 and pg1
|
|
|
|
for i in cls.pg_interfaces:
|
|
|
|
i.admin_up() # put the interface upsrc_if
|
|
|
|
i.config_ip4() # configure IPv4 address on the interface
|
|
|
|
i.resolve_arp() # resolve ARP, so that we know VPP MAC
|
|
|
|
|
2019-03-12 19:23:27 -07:00
|
|
|
@classmethod
|
|
|
|
def tearDownClass(cls):
|
|
|
|
super(TestLisp, cls).tearDownClass()
|
|
|
|
|
2017-01-31 10:39:16 +01:00
|
|
|
def setUp(self):
|
|
|
|
super(TestLisp, self).setUp()
|
2020-01-30 13:26:43 +01:00
|
|
|
self.vapi.lisp_enable_disable(is_enable=1)
|
2017-01-31 10:39:16 +01:00
|
|
|
|
|
|
|
def test_lisp_basic_encap(self):
|
|
|
|
"""Test case for basic encapsulation"""
|
|
|
|
|
|
|
|
self.deid_ip4_net = self.faf.net
|
|
|
|
self.deid_ip4 = self.faf.get_ip4()
|
2022-04-26 19:02:15 +02:00
|
|
|
self.seid_ip4 = "{!s}/{!s}".format(self.pg0.local_ip4, 32)
|
2020-01-30 13:26:43 +01:00
|
|
|
self.rloc_ip4 = self.pg1.remote_ip4
|
2017-01-31 10:39:16 +01:00
|
|
|
|
|
|
|
test_cases = [
|
|
|
|
{
|
2022-04-26 19:02:15 +02:00
|
|
|
"name": "basic ip4 over ip4",
|
|
|
|
"locator-sets": [VppLispLocatorSet(self, "ls-4o4")],
|
|
|
|
"locators": [VppLispLocator(self, self.pg1.sw_if_index, "ls-4o4")],
|
|
|
|
"local-mappings": [VppLocalMapping(self, self.seid_ip4, "ls-4o4")],
|
|
|
|
"remote-mappings": [
|
|
|
|
VppRemoteMapping(
|
|
|
|
self, self.deid_ip4_net, [LispRemoteLocator(self.rloc_ip4)]
|
|
|
|
)
|
2017-01-31 10:39:16 +01:00
|
|
|
],
|
2022-04-26 19:02:15 +02:00
|
|
|
"adjacencies": [
|
2017-01-31 10:39:16 +01:00
|
|
|
VppLispAdjacency(self, self.seid_ip4, self.deid_ip4_net)
|
2022-04-26 19:02:15 +02:00
|
|
|
],
|
2017-01-31 10:39:16 +01:00
|
|
|
}
|
|
|
|
]
|
|
|
|
self.test_driver = SimpleDriver(self, test_cases)
|
|
|
|
self.test_driver.run(self.deid_ip4)
|
|
|
|
|
|
|
|
|
2024-03-11 10:38:46 +00:00
|
|
|
@unittest.skipIf("lisp" in config.excluded_plugins, "Exclude LISP plugin tests")
|
2019-11-12 17:16:47 +00:00
|
|
|
class TestLispUT(VppTestCase):
|
2022-04-26 19:02:15 +02:00
|
|
|
"""Lisp UT"""
|
2019-11-12 17:16:47 +00:00
|
|
|
|
|
|
|
@classmethod
|
|
|
|
def setUpClass(cls):
|
|
|
|
super(TestLispUT, cls).setUpClass()
|
|
|
|
|
|
|
|
@classmethod
|
|
|
|
def tearDownClass(cls):
|
|
|
|
super(TestLispUT, cls).tearDownClass()
|
|
|
|
|
|
|
|
def test_fib(self):
|
2022-04-26 19:02:15 +02:00
|
|
|
"""LISP Unit Tests"""
|
2019-11-12 17:16:47 +00:00
|
|
|
error = self.vapi.cli("test lisp cp")
|
|
|
|
|
|
|
|
if error:
|
|
|
|
self.logger.critical(error)
|
|
|
|
self.assertNotIn("Failed", error)
|
|
|
|
|
|
|
|
|
2022-04-26 19:02:15 +02:00
|
|
|
if __name__ == "__main__":
|
2017-01-31 10:39:16 +01:00
|
|
|
unittest.main(testRunner=VppTestRunner)
|