2020-12-24 09:16:09 +00:00
|
|
|
#!/usr/bin/env python3
|
|
|
|
|
|
|
|
import unittest
|
|
|
|
|
|
|
|
from scapy.layers.inet import IP, UDP
|
|
|
|
from scapy.layers.inet6 import IPv6, Raw
|
2023-08-31 00:47:44 -04:00
|
|
|
from scapy.layers.l2 import Ether, ARP
|
2020-12-24 09:16:09 +00:00
|
|
|
|
2021-06-07 09:34:07 +00:00
|
|
|
from util import reassemble4
|
2020-12-24 09:16:09 +00:00
|
|
|
from vpp_object import VppObject
|
2023-08-31 00:47:44 -04:00
|
|
|
from framework import VppTestCase
|
|
|
|
from asfframework import VppTestRunner
|
2021-06-07 09:34:07 +00:00
|
|
|
from vpp_ipip_tun_interface import VppIpIpTunInterface
|
2022-04-26 19:02:15 +02:00
|
|
|
from template_ipsec import (
|
|
|
|
TemplateIpsec,
|
|
|
|
IpsecTun4,
|
|
|
|
)
|
|
|
|
from template_ipsec import (
|
|
|
|
TemplateIpsec,
|
|
|
|
IpsecTun4,
|
|
|
|
)
|
2021-06-07 09:34:07 +00:00
|
|
|
from test_ipsec_tun_if_esp import TemplateIpsecItf4
|
2020-12-24 09:16:09 +00:00
|
|
|
|
|
|
|
|
|
|
|
class VppLcpPair(VppObject):
|
|
|
|
def __init__(self, test, phy, host):
|
|
|
|
self._test = test
|
|
|
|
self.phy = phy
|
|
|
|
self.host = host
|
|
|
|
|
|
|
|
def add_vpp_config(self):
|
2022-04-26 19:02:15 +02:00
|
|
|
self._test.vapi.cli("test lcp add phy %s host %s" % (self.phy, self.host))
|
2020-12-24 09:16:09 +00:00
|
|
|
self._test.registry.register(self, self._test.logger)
|
|
|
|
return self
|
|
|
|
|
|
|
|
def remove_vpp_config(self):
|
2022-04-26 19:02:15 +02:00
|
|
|
self._test.vapi.cli("test lcp del phy %s host %s" % (self.phy, self.host))
|
2020-12-24 09:16:09 +00:00
|
|
|
|
|
|
|
def object_id(self):
|
2022-04-26 19:02:15 +02:00
|
|
|
return "lcp:%d:%d" % (self.phy.sw_if_index, self.host.sw_if_index)
|
2020-12-24 09:16:09 +00:00
|
|
|
|
|
|
|
def query_vpp_config(self):
|
2022-04-26 19:02:15 +02:00
|
|
|
pairs = list(self._test.vapi.vpp.details_iter(self._test.vapi.lcp_itf_pair_get))
|
2020-12-24 09:16:09 +00:00
|
|
|
|
|
|
|
for p in pairs:
|
2022-04-26 19:02:15 +02:00
|
|
|
if (
|
|
|
|
p.phy_sw_if_index == self.phy.sw_if_index
|
|
|
|
and p.host_sw_if_index == self.host.sw_if_index
|
|
|
|
):
|
2020-12-24 09:16:09 +00:00
|
|
|
return True
|
|
|
|
return False
|
|
|
|
|
|
|
|
|
|
|
|
class TestLinuxCP(VppTestCase):
|
2022-04-26 19:02:15 +02:00
|
|
|
"""Linux Control Plane"""
|
|
|
|
|
|
|
|
extra_vpp_plugin_config = [
|
|
|
|
"plugin",
|
|
|
|
"linux_cp_plugin.so",
|
|
|
|
"{",
|
|
|
|
"enable",
|
|
|
|
"}",
|
|
|
|
"plugin",
|
|
|
|
"linux_cp_unittest_plugin.so",
|
|
|
|
"{",
|
|
|
|
"enable",
|
|
|
|
"}",
|
|
|
|
]
|
2020-12-24 09:16:09 +00:00
|
|
|
|
|
|
|
@classmethod
|
|
|
|
def setUpClass(cls):
|
|
|
|
super(TestLinuxCP, cls).setUpClass()
|
|
|
|
|
|
|
|
@classmethod
|
|
|
|
def tearDownClass(cls):
|
|
|
|
super(TestLinuxCP, cls).tearDownClass()
|
|
|
|
|
|
|
|
def setUp(self):
|
|
|
|
super(TestLinuxCP, self).setUp()
|
|
|
|
|
2021-06-07 09:34:07 +00:00
|
|
|
# create 4 pg interfaces so we can create two pairs
|
2020-12-24 09:16:09 +00:00
|
|
|
self.create_pg_interfaces(range(4))
|
|
|
|
|
2021-06-07 09:34:07 +00:00
|
|
|
# create on ip4 and one ip6 pg tun
|
|
|
|
self.pg_interfaces += self.create_pg_ip4_interfaces(range(4, 5))
|
|
|
|
self.pg_interfaces += self.create_pg_ip6_interfaces(range(5, 6))
|
|
|
|
|
2020-12-24 09:16:09 +00:00
|
|
|
for i in self.pg_interfaces:
|
|
|
|
i.admin_up()
|
|
|
|
|
|
|
|
def tearDown(self):
|
|
|
|
for i in self.pg_interfaces:
|
|
|
|
i.admin_down()
|
|
|
|
super(TestLinuxCP, self).tearDown()
|
|
|
|
|
|
|
|
def test_linux_cp_tap(self):
|
2022-04-26 19:02:15 +02:00
|
|
|
"""Linux CP TAP"""
|
2020-12-24 09:16:09 +00:00
|
|
|
|
|
|
|
#
|
|
|
|
# Setup
|
|
|
|
#
|
|
|
|
|
|
|
|
arp_opts = {"who-has": 1, "is-at": 2}
|
|
|
|
|
|
|
|
# create two pairs, wihch a bunch of hots on the phys
|
|
|
|
hosts = [self.pg0, self.pg1]
|
|
|
|
phys = [self.pg2, self.pg3]
|
|
|
|
N_HOSTS = 4
|
|
|
|
|
|
|
|
for phy in phys:
|
|
|
|
phy.config_ip4()
|
|
|
|
phy.generate_remote_hosts(4)
|
|
|
|
phy.configure_ipv4_neighbors()
|
|
|
|
|
|
|
|
pair1 = VppLcpPair(self, phys[0], hosts[0]).add_vpp_config()
|
|
|
|
pair2 = VppLcpPair(self, phys[1], hosts[1]).add_vpp_config()
|
|
|
|
|
|
|
|
self.logger.info(self.vapi.cli("sh lcp adj verbose"))
|
|
|
|
self.logger.info(self.vapi.cli("sh lcp"))
|
|
|
|
|
|
|
|
#
|
|
|
|
# Traffic Tests
|
|
|
|
#
|
|
|
|
|
|
|
|
# hosts to phys
|
|
|
|
for phy, host in zip(phys, hosts):
|
|
|
|
for j in range(N_HOSTS):
|
2022-04-26 19:02:15 +02:00
|
|
|
p = (
|
|
|
|
Ether(src=phy.local_mac, dst=phy.remote_hosts[j].mac)
|
|
|
|
/ IP(src=phy.local_ip4, dst=phy.remote_hosts[j].ip4)
|
|
|
|
/ UDP(sport=1234, dport=1234)
|
|
|
|
/ Raw()
|
|
|
|
)
|
2020-12-24 09:16:09 +00:00
|
|
|
|
|
|
|
rxs = self.send_and_expect(host, [p], phy)
|
|
|
|
|
|
|
|
# verify packet is unchanged
|
|
|
|
for rx in rxs:
|
|
|
|
self.assertEqual(p.show2(True), rx.show2(True))
|
|
|
|
|
|
|
|
# ARPs x-connect to phy
|
2022-04-26 19:02:15 +02:00
|
|
|
p = Ether(dst="ff:ff:ff:ff:ff:ff", src=phy.remote_hosts[j].mac) / ARP(
|
|
|
|
op="who-has",
|
|
|
|
hwdst=phy.remote_hosts[j].mac,
|
|
|
|
hwsrc=phy.local_mac,
|
|
|
|
psrc=phy.local_ip4,
|
|
|
|
pdst=phy.remote_hosts[j].ip4,
|
|
|
|
)
|
2020-12-24 09:16:09 +00:00
|
|
|
|
|
|
|
rxs = self.send_and_expect(host, [p], phy)
|
|
|
|
|
|
|
|
# verify packet is unchanged
|
|
|
|
for rx in rxs:
|
|
|
|
self.assertEqual(p.show2(True), rx.show2(True))
|
|
|
|
|
|
|
|
# phy to host
|
|
|
|
for phy, host in zip(phys, hosts):
|
|
|
|
for j in range(N_HOSTS):
|
2022-04-26 19:02:15 +02:00
|
|
|
p = (
|
|
|
|
Ether(dst=phy.local_mac, src=phy.remote_hosts[j].mac)
|
|
|
|
/ IP(dst=phy.local_ip4, src=phy.remote_hosts[j].ip4)
|
|
|
|
/ UDP(sport=1234, dport=1234)
|
|
|
|
/ Raw()
|
|
|
|
)
|
2020-12-24 09:16:09 +00:00
|
|
|
|
|
|
|
rxs = self.send_and_expect(phy, [p], host)
|
|
|
|
|
|
|
|
# verify packet is unchanged
|
|
|
|
for rx in rxs:
|
|
|
|
self.assertEqual(p.show2(True), rx.show2(True))
|
|
|
|
|
|
|
|
# ARPs rx'd on the phy are sent to the host
|
2022-04-26 19:02:15 +02:00
|
|
|
p = Ether(dst="ff:ff:ff:ff:ff:ff", src=phy.remote_hosts[j].mac) / ARP(
|
|
|
|
op="is-at",
|
|
|
|
hwsrc=phy.remote_hosts[j].mac,
|
|
|
|
hwdst=phy.local_mac,
|
|
|
|
pdst=phy.local_ip4,
|
|
|
|
psrc=phy.remote_hosts[j].ip4,
|
|
|
|
)
|
2020-12-24 09:16:09 +00:00
|
|
|
|
|
|
|
rxs = self.send_and_expect(phy, [p], host)
|
|
|
|
|
|
|
|
# verify packet is unchanged
|
|
|
|
for rx in rxs:
|
|
|
|
self.assertEqual(p.show2(True), rx.show2(True))
|
|
|
|
|
|
|
|
# cleanup
|
|
|
|
for phy in phys:
|
|
|
|
phy.unconfig_ip4()
|
|
|
|
|
2021-06-07 09:34:07 +00:00
|
|
|
def test_linux_cp_tun(self):
|
2022-04-26 19:02:15 +02:00
|
|
|
"""Linux CP TUN"""
|
2021-06-07 09:34:07 +00:00
|
|
|
|
|
|
|
#
|
|
|
|
# Setup
|
|
|
|
#
|
|
|
|
N_PKTS = 31
|
|
|
|
|
|
|
|
# create two pairs, wihch a bunch of hots on the phys
|
|
|
|
hosts = [self.pg4, self.pg5]
|
|
|
|
phy = self.pg2
|
|
|
|
|
|
|
|
phy.config_ip4()
|
|
|
|
phy.config_ip6()
|
|
|
|
phy.resolve_arp()
|
|
|
|
phy.resolve_ndp()
|
|
|
|
|
|
|
|
tun4 = VppIpIpTunInterface(
|
2022-04-26 19:02:15 +02:00
|
|
|
self, phy, phy.local_ip4, phy.remote_ip4
|
|
|
|
).add_vpp_config()
|
2021-06-07 09:34:07 +00:00
|
|
|
tun6 = VppIpIpTunInterface(
|
2022-04-26 19:02:15 +02:00
|
|
|
self, phy, phy.local_ip6, phy.remote_ip6
|
|
|
|
).add_vpp_config()
|
2021-06-07 09:34:07 +00:00
|
|
|
tuns = [tun4, tun6]
|
|
|
|
|
|
|
|
tun4.admin_up()
|
|
|
|
tun4.config_ip4()
|
|
|
|
tun6.admin_up()
|
|
|
|
tun6.config_ip6()
|
|
|
|
|
|
|
|
pair1 = VppLcpPair(self, tuns[0], hosts[0]).add_vpp_config()
|
|
|
|
pair2 = VppLcpPair(self, tuns[1], hosts[1]).add_vpp_config()
|
|
|
|
|
|
|
|
self.logger.info(self.vapi.cli("sh lcp adj verbose"))
|
|
|
|
self.logger.info(self.vapi.cli("sh lcp"))
|
|
|
|
self.logger.info(self.vapi.cli("sh ip punt redirect"))
|
|
|
|
|
|
|
|
#
|
|
|
|
# Traffic Tests
|
|
|
|
#
|
|
|
|
|
|
|
|
# host to phy for v4
|
2022-04-26 19:02:15 +02:00
|
|
|
p = IP(src=tun4.local_ip4, dst="2.2.2.2") / UDP(sport=1234, dport=1234) / Raw()
|
2021-06-07 09:34:07 +00:00
|
|
|
|
|
|
|
rxs = self.send_and_expect(self.pg4, p * N_PKTS, phy)
|
|
|
|
|
|
|
|
# verify inner packet is unchanged and has the tunnel encap
|
|
|
|
for rx in rxs:
|
|
|
|
self.assertEqual(rx[Ether].dst, phy.remote_mac)
|
|
|
|
self.assertEqual(rx[IP].dst, phy.remote_ip4)
|
|
|
|
self.assertEqual(rx[IP].src, phy.local_ip4)
|
|
|
|
inner = IP(rx[IP].payload)
|
|
|
|
self.assertEqual(inner.src, tun4.local_ip4)
|
|
|
|
self.assertEqual(inner.dst, "2.2.2.2")
|
|
|
|
|
|
|
|
# host to phy for v6
|
2022-04-26 19:02:15 +02:00
|
|
|
p = IPv6(src=tun6.local_ip6, dst="2::2") / UDP(sport=1234, dport=1234) / Raw()
|
2021-06-07 09:34:07 +00:00
|
|
|
|
|
|
|
rxs = self.send_and_expect(self.pg5, p * N_PKTS, phy)
|
|
|
|
|
|
|
|
# verify inner packet is unchanged and has the tunnel encap
|
|
|
|
for rx in rxs:
|
|
|
|
self.assertEqual(rx[IPv6].dst, phy.remote_ip6)
|
|
|
|
self.assertEqual(rx[IPv6].src, phy.local_ip6)
|
|
|
|
inner = IPv6(rx[IPv6].payload)
|
|
|
|
self.assertEqual(inner.src, tun6.local_ip6)
|
|
|
|
self.assertEqual(inner.dst, "2::2")
|
|
|
|
|
|
|
|
# phy to host v4
|
2022-04-26 19:02:15 +02:00
|
|
|
p = (
|
|
|
|
Ether(dst=phy.local_mac, src=phy.remote_mac)
|
|
|
|
/ IP(dst=phy.local_ip4, src=phy.remote_ip4)
|
|
|
|
/ IP(dst=tun4.local_ip4, src=tun4.remote_ip4)
|
|
|
|
/ UDP(sport=1234, dport=1234)
|
|
|
|
/ Raw()
|
|
|
|
)
|
2021-06-07 09:34:07 +00:00
|
|
|
|
|
|
|
rxs = self.send_and_expect(phy, p * N_PKTS, self.pg4)
|
|
|
|
for rx in rxs:
|
|
|
|
rx = IP(rx)
|
|
|
|
self.assertEqual(rx[IP].dst, tun4.local_ip4)
|
|
|
|
self.assertEqual(rx[IP].src, tun4.remote_ip4)
|
|
|
|
|
|
|
|
# phy to host v6
|
2022-04-26 19:02:15 +02:00
|
|
|
p = (
|
|
|
|
Ether(dst=phy.local_mac, src=phy.remote_mac)
|
|
|
|
/ IPv6(dst=phy.local_ip6, src=phy.remote_ip6)
|
|
|
|
/ IPv6(dst=tun6.local_ip6, src=tun6.remote_ip6)
|
|
|
|
/ UDP(sport=1234, dport=1234)
|
|
|
|
/ Raw()
|
|
|
|
)
|
2021-06-07 09:34:07 +00:00
|
|
|
|
|
|
|
rxs = self.send_and_expect(phy, p * N_PKTS, self.pg5)
|
|
|
|
for rx in rxs:
|
|
|
|
rx = IPv6(rx)
|
|
|
|
self.assertEqual(rx[IPv6].dst, tun6.local_ip6)
|
|
|
|
self.assertEqual(rx[IPv6].src, tun6.remote_ip6)
|
|
|
|
|
|
|
|
# cleanup
|
|
|
|
phy.unconfig_ip4()
|
|
|
|
phy.unconfig_ip6()
|
|
|
|
|
|
|
|
tun4.unconfig_ip4()
|
|
|
|
tun6.unconfig_ip6()
|
|
|
|
|
|
|
|
|
2022-04-26 19:02:15 +02:00
|
|
|
class TestLinuxCPIpsec(TemplateIpsec, TemplateIpsecItf4, IpsecTun4):
|
|
|
|
"""IPsec Interface IPv4"""
|
2021-06-07 09:34:07 +00:00
|
|
|
|
2022-04-26 19:02:15 +02:00
|
|
|
extra_vpp_plugin_config = [
|
|
|
|
"plugin",
|
|
|
|
"linux_cp_plugin.so",
|
|
|
|
"{",
|
|
|
|
"enable",
|
|
|
|
"}",
|
|
|
|
"plugin",
|
|
|
|
"linux_cp_unittest_plugin.so",
|
|
|
|
"{",
|
|
|
|
"enable",
|
|
|
|
"}",
|
|
|
|
]
|
2021-06-07 09:34:07 +00:00
|
|
|
|
|
|
|
def setUp(self):
|
|
|
|
super(TestLinuxCPIpsec, self).setUp()
|
|
|
|
|
|
|
|
self.tun_if = self.pg0
|
|
|
|
self.pg_interfaces += self.create_pg_ip4_interfaces(range(3, 4))
|
|
|
|
self.pg_interfaces += self.create_pg_ip6_interfaces(range(4, 5))
|
|
|
|
|
|
|
|
def tearDown(self):
|
|
|
|
super(TestLinuxCPIpsec, self).tearDown()
|
|
|
|
|
|
|
|
def verify_encrypted(self, p, sa, rxs):
|
|
|
|
decrypt_pkts = []
|
|
|
|
for rx in rxs:
|
|
|
|
if p.nat_header:
|
|
|
|
self.assertEqual(rx[UDP].dport, 4500)
|
|
|
|
self.assert_packet_checksums_valid(rx)
|
|
|
|
self.assertEqual(len(rx) - len(Ether()), rx[IP].len)
|
|
|
|
try:
|
|
|
|
rx_ip = rx[IP]
|
|
|
|
decrypt_pkt = p.vpp_tun_sa.decrypt(rx_ip)
|
|
|
|
if not decrypt_pkt.haslayer(IP):
|
|
|
|
decrypt_pkt = IP(decrypt_pkt[Raw].load)
|
|
|
|
if rx_ip.proto == socket.IPPROTO_ESP:
|
|
|
|
self.verify_esp_padding(sa, rx_ip[ESP].data, decrypt_pkt)
|
|
|
|
decrypt_pkts.append(decrypt_pkt)
|
|
|
|
self.assert_equal(decrypt_pkt.src, p.tun_if.local_ip4)
|
|
|
|
self.assert_equal(decrypt_pkt.dst, p.tun_if.remote_ip4)
|
|
|
|
except:
|
|
|
|
self.logger.debug(ppp("Unexpected packet:", rx))
|
|
|
|
try:
|
|
|
|
self.logger.debug(ppp("Decrypted packet:", decrypt_pkt))
|
|
|
|
except:
|
|
|
|
pass
|
|
|
|
raise
|
|
|
|
pkts = reassemble4(decrypt_pkts)
|
|
|
|
for pkt in pkts:
|
|
|
|
self.assert_packet_checksums_valid(pkt)
|
|
|
|
|
|
|
|
def verify_decrypted(self, p, rxs):
|
|
|
|
for rx in rxs:
|
|
|
|
rx = IP(rx)
|
|
|
|
self.assert_equal(rx[IP].src, p.tun_if.remote_ip4)
|
|
|
|
self.assert_equal(rx[IP].dst, p.tun_if.local_ip4)
|
|
|
|
self.assert_packet_checksums_valid(rx)
|
|
|
|
|
2022-04-26 19:02:15 +02:00
|
|
|
def gen_encrypt_pkts(self, p, sa, sw_intf, src, dst, count=1, payload_size=54):
|
|
|
|
return [
|
|
|
|
Ether(src=sw_intf.remote_mac, dst=sw_intf.local_mac)
|
|
|
|
/ sa.encrypt(
|
|
|
|
IP(src=src, dst=dst)
|
|
|
|
/ UDP(sport=1111, dport=2222)
|
|
|
|
/ Raw(b"X" * payload_size)
|
|
|
|
)
|
|
|
|
for i in range(count)
|
|
|
|
]
|
2021-06-07 09:34:07 +00:00
|
|
|
|
|
|
|
def test_linux_cp_ipsec4_tun(self):
|
2022-04-26 19:02:15 +02:00
|
|
|
"""Linux CP Ipsec TUN"""
|
2021-06-07 09:34:07 +00:00
|
|
|
|
|
|
|
#
|
|
|
|
# Setup
|
|
|
|
#
|
|
|
|
N_PKTS = 31
|
|
|
|
|
|
|
|
# the pg that paris with the tunnel
|
|
|
|
self.host = self.pg3
|
|
|
|
|
|
|
|
# tunnel and protection setup
|
|
|
|
p = self.ipv4_params
|
|
|
|
|
|
|
|
self.config_network(p)
|
2022-04-26 19:02:15 +02:00
|
|
|
self.config_sa_tun(p, self.pg0.local_ip4, self.pg0.remote_ip4)
|
2021-06-07 09:34:07 +00:00
|
|
|
self.config_protect(p)
|
|
|
|
|
|
|
|
pair = VppLcpPair(self, p.tun_if, self.host).add_vpp_config()
|
|
|
|
|
2021-11-09 16:15:48 +00:00
|
|
|
self.logger.info(self.vapi.cli("sh int addr"))
|
2021-06-07 09:34:07 +00:00
|
|
|
self.logger.info(self.vapi.cli("sh lcp"))
|
|
|
|
self.logger.info(self.vapi.cli("sh ip punt redirect"))
|
|
|
|
|
|
|
|
#
|
|
|
|
# Traffic Tests
|
|
|
|
#
|
|
|
|
|
|
|
|
# host to phy for v4
|
2022-04-26 19:02:15 +02:00
|
|
|
pkt = (
|
|
|
|
IP(src=p.tun_if.local_ip4, dst=p.tun_if.remote_ip4)
|
|
|
|
/ UDP(sport=1234, dport=1234)
|
|
|
|
/ Raw()
|
|
|
|
)
|
2021-06-07 09:34:07 +00:00
|
|
|
|
|
|
|
rxs = self.send_and_expect(self.host, pkt * N_PKTS, self.tun_if)
|
|
|
|
self.verify_encrypted(p, p.vpp_tun_sa, rxs)
|
|
|
|
|
|
|
|
# phy to host for v4
|
2022-04-26 19:02:15 +02:00
|
|
|
pkts = self.gen_encrypt_pkts(
|
|
|
|
p,
|
|
|
|
p.scapy_tun_sa,
|
|
|
|
self.tun_if,
|
|
|
|
src=p.tun_if.remote_ip4,
|
|
|
|
dst=p.tun_if.local_ip4,
|
|
|
|
count=N_PKTS,
|
|
|
|
)
|
2021-11-09 16:15:48 +00:00
|
|
|
rxs = self.send_and_expect(self.tun_if, pkts, self.host)
|
|
|
|
self.verify_decrypted(p, rxs)
|
2021-06-07 09:34:07 +00:00
|
|
|
|
|
|
|
# cleanup
|
|
|
|
pair.remove_vpp_config()
|
|
|
|
self.unconfig_protect(p)
|
|
|
|
self.unconfig_sa(p)
|
|
|
|
self.unconfig_network(p)
|
|
|
|
|
2020-12-24 09:16:09 +00:00
|
|
|
|
2022-04-26 19:02:15 +02:00
|
|
|
if __name__ == "__main__":
|
2020-12-24 09:16:09 +00:00
|
|
|
unittest.main(testRunner=VppTestRunner)
|