2019-10-31 13:31:07 -05:00
|
|
|
#!/usr/bin/env python3
|
2018-02-08 21:45:08 +01:00
|
|
|
""" Classifier-based L2 ACL Test Case HLD:
|
|
|
|
"""
|
|
|
|
|
|
|
|
import unittest
|
|
|
|
import random
|
|
|
|
import binascii
|
|
|
|
import socket
|
|
|
|
|
|
|
|
|
|
|
|
from scapy.packet import Raw
|
2018-07-03 13:58:24 +02:00
|
|
|
from scapy.data import ETH_P_IP
|
2018-02-08 21:45:08 +01:00
|
|
|
from scapy.layers.l2 import Ether
|
|
|
|
from scapy.layers.inet import IP, TCP, UDP, ICMP
|
|
|
|
from scapy.layers.inet6 import IPv6, ICMPv6EchoRequest
|
|
|
|
from scapy.layers.inet6 import IPv6ExtHdrFragment
|
|
|
|
from framework import VppTestCase, VppTestRunner
|
|
|
|
from util import Host, ppp
|
2019-08-27 10:16:44 +02:00
|
|
|
from template_classifier import TestClassifier
|
2018-02-08 21:45:08 +01:00
|
|
|
|
|
|
|
|
2019-08-27 10:16:44 +02:00
|
|
|
class TestClassifyAcl(TestClassifier):
|
2022-04-26 19:02:15 +02:00
|
|
|
"""Classifier-based L2 input and output ACL Test Case"""
|
2018-02-08 21:45:08 +01:00
|
|
|
|
|
|
|
# traffic types
|
|
|
|
IP = 0
|
|
|
|
ICMP = 1
|
|
|
|
|
|
|
|
# IP version
|
|
|
|
IPRANDOM = -1
|
|
|
|
IPV4 = 0
|
|
|
|
IPV6 = 1
|
|
|
|
|
|
|
|
# rule types
|
|
|
|
DENY = 0
|
|
|
|
PERMIT = 1
|
|
|
|
|
|
|
|
# supported protocols
|
|
|
|
proto = [[6, 17], [1, 58]]
|
2022-04-26 19:02:15 +02:00
|
|
|
proto_map = {1: "ICMP", 58: "ICMPv6EchoRequest", 6: "TCP", 17: "UDP"}
|
2018-02-08 21:45:08 +01:00
|
|
|
ICMPv4 = 0
|
|
|
|
ICMPv6 = 1
|
|
|
|
TCP = 0
|
|
|
|
UDP = 1
|
|
|
|
PROTO_ALL = 0
|
|
|
|
|
|
|
|
# port ranges
|
|
|
|
PORTS_ALL = -1
|
|
|
|
PORTS_RANGE = 0
|
|
|
|
PORTS_RANGE_2 = 1
|
|
|
|
udp_sport_from = 10
|
|
|
|
udp_sport_to = udp_sport_from + 5
|
|
|
|
udp_dport_from = 20000
|
|
|
|
udp_dport_to = udp_dport_from + 5000
|
|
|
|
tcp_sport_from = 30
|
|
|
|
tcp_sport_to = tcp_sport_from + 5
|
|
|
|
tcp_dport_from = 40000
|
|
|
|
tcp_dport_to = tcp_dport_from + 5000
|
|
|
|
|
|
|
|
udp_sport_from_2 = 90
|
|
|
|
udp_sport_to_2 = udp_sport_from_2 + 5
|
|
|
|
udp_dport_from_2 = 30000
|
|
|
|
udp_dport_to_2 = udp_dport_from_2 + 5000
|
|
|
|
tcp_sport_from_2 = 130
|
|
|
|
tcp_sport_to_2 = tcp_sport_from_2 + 5
|
|
|
|
tcp_dport_from_2 = 20000
|
|
|
|
tcp_dport_to_2 = tcp_dport_from_2 + 5000
|
|
|
|
|
|
|
|
icmp4_type = 8 # echo request
|
|
|
|
icmp4_code = 3
|
|
|
|
icmp6_type = 128 # echo request
|
|
|
|
icmp6_code = 3
|
|
|
|
|
|
|
|
icmp4_type_2 = 8
|
|
|
|
icmp4_code_from_2 = 5
|
|
|
|
icmp4_code_to_2 = 20
|
|
|
|
icmp6_type_2 = 128
|
|
|
|
icmp6_code_from_2 = 8
|
|
|
|
icmp6_code_to_2 = 42
|
|
|
|
|
|
|
|
# Test variables
|
|
|
|
bd_id = 1
|
|
|
|
|
|
|
|
@classmethod
|
|
|
|
def setUpClass(cls):
|
|
|
|
"""
|
|
|
|
Perform standard class setup (defined by class method setUpClass in
|
|
|
|
class VppTestCase) before running the test case, set test case related
|
|
|
|
variables and configure VPP.
|
|
|
|
"""
|
|
|
|
super(TestClassifyAcl, cls).setUpClass()
|
2019-08-27 10:16:44 +02:00
|
|
|
cls.af = None
|
2018-02-08 21:45:08 +01:00
|
|
|
|
|
|
|
try:
|
|
|
|
# Create 2 pg interfaces
|
|
|
|
cls.create_pg_interfaces(range(2))
|
|
|
|
|
|
|
|
# Packet flows mapping pg0 -> pg1, pg2 etc.
|
|
|
|
cls.flows = dict()
|
|
|
|
cls.flows[cls.pg0] = [cls.pg1]
|
|
|
|
|
|
|
|
# Packet sizes
|
|
|
|
cls.pg_if_packet_sizes = [64, 512, 1518, 9018]
|
|
|
|
|
|
|
|
# Create BD with MAC learning and unknown unicast flooding disabled
|
|
|
|
# and put interfaces to this BD
|
2022-09-16 13:20:07 +02:00
|
|
|
cls.vapi.bridge_domain_add_del_v2(
|
|
|
|
bd_id=cls.bd_id, uu_flood=1, learn=1, flood=1, forward=1, is_add=1
|
|
|
|
)
|
2018-02-08 21:45:08 +01:00
|
|
|
for pg_if in cls.pg_interfaces:
|
2019-03-11 19:23:25 +01:00
|
|
|
cls.vapi.sw_interface_set_l2_bridge(
|
2022-04-26 19:02:15 +02:00
|
|
|
rx_sw_if_index=pg_if.sw_if_index, bd_id=cls.bd_id
|
|
|
|
)
|
2018-02-08 21:45:08 +01:00
|
|
|
|
|
|
|
# Set up all interfaces
|
|
|
|
for i in cls.pg_interfaces:
|
|
|
|
i.admin_up()
|
|
|
|
|
|
|
|
# Mapping between packet-generator index and lists of test hosts
|
|
|
|
cls.hosts_by_pg_idx = dict()
|
|
|
|
for pg_if in cls.pg_interfaces:
|
|
|
|
cls.hosts_by_pg_idx[pg_if.sw_if_index] = []
|
|
|
|
|
|
|
|
# Create list of deleted hosts
|
|
|
|
cls.deleted_hosts_by_pg_idx = dict()
|
|
|
|
for pg_if in cls.pg_interfaces:
|
|
|
|
cls.deleted_hosts_by_pg_idx[pg_if.sw_if_index] = []
|
|
|
|
|
|
|
|
# warm-up the mac address tables
|
|
|
|
# self.warmup_test()
|
|
|
|
|
2018-07-03 13:58:24 +02:00
|
|
|
# Holder of the active classify table key
|
2022-04-26 19:02:15 +02:00
|
|
|
cls.acl_active_table = ""
|
2018-07-03 13:58:24 +02:00
|
|
|
|
2018-02-08 21:45:08 +01:00
|
|
|
except Exception:
|
|
|
|
super(TestClassifyAcl, cls).tearDownClass()
|
|
|
|
raise
|
|
|
|
|
2019-03-12 19:23:27 -07:00
|
|
|
@classmethod
|
|
|
|
def tearDownClass(cls):
|
|
|
|
super(TestClassifyAcl, cls).tearDownClass()
|
|
|
|
|
2018-02-08 21:45:08 +01:00
|
|
|
def setUp(self):
|
|
|
|
super(TestClassifyAcl, self).setUp()
|
|
|
|
self.acl_tbl_idx = {}
|
|
|
|
|
|
|
|
def tearDown(self):
|
|
|
|
"""
|
|
|
|
Show various debug prints after each test.
|
|
|
|
"""
|
|
|
|
if not self.vpp_dead:
|
2022-04-26 19:02:15 +02:00
|
|
|
if self.acl_active_table == "mac_inout":
|
2018-07-03 13:58:24 +02:00
|
|
|
self.output_acl_set_interface(
|
2022-04-26 19:02:15 +02:00
|
|
|
self.pg1, self.acl_tbl_idx.get(self.acl_active_table), 0
|
|
|
|
)
|
2018-07-03 13:58:24 +02:00
|
|
|
self.input_acl_set_interface(
|
2022-04-26 19:02:15 +02:00
|
|
|
self.pg0, self.acl_tbl_idx.get(self.acl_active_table), 0
|
|
|
|
)
|
|
|
|
self.acl_active_table = ""
|
|
|
|
elif self.acl_active_table == "mac_out":
|
2018-07-03 13:58:24 +02:00
|
|
|
self.output_acl_set_interface(
|
2022-04-26 19:02:15 +02:00
|
|
|
self.pg1, self.acl_tbl_idx.get(self.acl_active_table), 0
|
|
|
|
)
|
|
|
|
self.acl_active_table = ""
|
|
|
|
elif self.acl_active_table == "mac_in":
|
2018-07-03 13:58:24 +02:00
|
|
|
self.input_acl_set_interface(
|
2022-04-26 19:02:15 +02:00
|
|
|
self.pg0, self.acl_tbl_idx.get(self.acl_active_table), 0
|
|
|
|
)
|
|
|
|
self.acl_active_table = ""
|
2018-02-08 21:45:08 +01:00
|
|
|
|
2018-07-03 13:58:24 +02:00
|
|
|
super(TestClassifyAcl, self).tearDown()
|
2018-02-08 21:45:08 +01:00
|
|
|
|
2022-04-26 19:02:15 +02:00
|
|
|
def create_classify_session(
|
|
|
|
self, intf, table_index, match, hit_next_index=0xFFFFFFFF, is_add=1
|
|
|
|
):
|
2018-02-08 21:45:08 +01:00
|
|
|
"""Create Classify Session
|
|
|
|
|
|
|
|
:param VppInterface intf: Interface to apply classify session.
|
|
|
|
:param int table_index: table index to identify classify table.
|
|
|
|
:param str match: matched value for interested traffic.
|
|
|
|
:param int is_add: option to configure classify session.
|
|
|
|
- create(1) or delete(0)
|
|
|
|
"""
|
2019-08-27 10:16:44 +02:00
|
|
|
mask_match, mask_match_len = self._resolve_mask_match(match)
|
2018-02-08 21:45:08 +01:00
|
|
|
r = self.vapi.classify_add_del_session(
|
2019-08-27 10:16:44 +02:00
|
|
|
is_add=is_add,
|
|
|
|
table_index=table_index,
|
|
|
|
match=mask_match,
|
|
|
|
match_len=mask_match_len,
|
2022-04-26 19:02:15 +02:00
|
|
|
hit_next_index=hit_next_index,
|
|
|
|
)
|
|
|
|
self.assertIsNotNone(r, "No response msg for add_del_session")
|
2018-02-08 21:45:08 +01:00
|
|
|
|
|
|
|
def create_hosts(self, count, start=0):
|
|
|
|
"""
|
|
|
|
Create required number of host MAC addresses and distribute them among
|
|
|
|
interfaces. Create host IPv4 address for every host MAC address.
|
|
|
|
|
|
|
|
:param int count: Number of hosts to create MAC/IPv4 addresses for.
|
|
|
|
:param int start: Number to start numbering from.
|
|
|
|
"""
|
|
|
|
n_int = len(self.pg_interfaces)
|
2019-10-21 20:39:45 +02:00
|
|
|
macs_per_if = count // n_int
|
2018-02-08 21:45:08 +01:00
|
|
|
i = -1
|
|
|
|
for pg_if in self.pg_interfaces:
|
|
|
|
i += 1
|
|
|
|
start_nr = macs_per_if * i + start
|
2022-04-26 19:02:15 +02:00
|
|
|
end_nr = (
|
|
|
|
count + start if i == (n_int - 1) else macs_per_if * (i + 1) + start
|
|
|
|
)
|
2018-02-08 21:45:08 +01:00
|
|
|
hosts = self.hosts_by_pg_idx[pg_if.sw_if_index]
|
|
|
|
for j in range(start_nr, end_nr):
|
|
|
|
host = Host(
|
|
|
|
"00:00:00:ff:%02x:%02x" % (pg_if.sw_if_index, j),
|
|
|
|
"172.17.1%02x.%u" % (pg_if.sw_if_index, j),
|
2022-04-26 19:02:15 +02:00
|
|
|
"2017:dead:%02x::%u" % (pg_if.sw_if_index, j),
|
|
|
|
)
|
2018-02-08 21:45:08 +01:00
|
|
|
hosts.append(host)
|
|
|
|
|
|
|
|
def create_upper_layer(self, packet_index, proto, ports=0):
|
|
|
|
p = self.proto_map[proto]
|
2022-04-26 19:02:15 +02:00
|
|
|
if p == "UDP":
|
2018-02-08 21:45:08 +01:00
|
|
|
if ports == 0:
|
2022-04-26 19:02:15 +02:00
|
|
|
return UDP(
|
|
|
|
sport=random.randint(self.udp_sport_from, self.udp_sport_to),
|
|
|
|
dport=random.randint(self.udp_dport_from, self.udp_dport_to),
|
|
|
|
)
|
2018-02-08 21:45:08 +01:00
|
|
|
else:
|
|
|
|
return UDP(sport=ports, dport=ports)
|
2022-04-26 19:02:15 +02:00
|
|
|
elif p == "TCP":
|
2018-02-08 21:45:08 +01:00
|
|
|
if ports == 0:
|
2022-04-26 19:02:15 +02:00
|
|
|
return TCP(
|
|
|
|
sport=random.randint(self.tcp_sport_from, self.tcp_sport_to),
|
|
|
|
dport=random.randint(self.tcp_dport_from, self.tcp_dport_to),
|
|
|
|
)
|
2018-02-08 21:45:08 +01:00
|
|
|
else:
|
|
|
|
return TCP(sport=ports, dport=ports)
|
2022-04-26 19:02:15 +02:00
|
|
|
return ""
|
|
|
|
|
|
|
|
def create_stream(
|
|
|
|
self,
|
|
|
|
src_if,
|
|
|
|
packet_sizes,
|
|
|
|
traffic_type=0,
|
|
|
|
ipv6=0,
|
|
|
|
proto=-1,
|
|
|
|
ports=0,
|
|
|
|
fragments=False,
|
|
|
|
pkt_raw=True,
|
|
|
|
etype=-1,
|
|
|
|
):
|
2018-02-08 21:45:08 +01:00
|
|
|
"""
|
|
|
|
Create input packet stream for defined interface using hosts or
|
|
|
|
deleted_hosts list.
|
|
|
|
|
|
|
|
:param object src_if: Interface to create packet stream for.
|
|
|
|
:param list packet_sizes: List of required packet sizes.
|
|
|
|
:param traffic_type: 1: ICMP packet, 2: IPv6 with EH, 0: otherwise.
|
|
|
|
:return: Stream of packets.
|
|
|
|
"""
|
|
|
|
pkts = []
|
|
|
|
if self.flows.__contains__(src_if):
|
|
|
|
src_hosts = self.hosts_by_pg_idx[src_if.sw_if_index]
|
|
|
|
for dst_if in self.flows[src_if]:
|
|
|
|
dst_hosts = self.hosts_by_pg_idx[dst_if.sw_if_index]
|
|
|
|
n_int = len(dst_hosts) * len(src_hosts)
|
|
|
|
for i in range(0, n_int):
|
2019-10-21 20:39:45 +02:00
|
|
|
dst_host = dst_hosts[i // len(src_hosts)]
|
2018-02-08 21:45:08 +01:00
|
|
|
src_host = src_hosts[i % len(src_hosts)]
|
|
|
|
pkt_info = self.create_packet_info(src_if, dst_if)
|
|
|
|
if ipv6 == 1:
|
|
|
|
pkt_info.ip = 1
|
|
|
|
elif ipv6 == 0:
|
|
|
|
pkt_info.ip = 0
|
|
|
|
else:
|
|
|
|
pkt_info.ip = random.choice([0, 1])
|
|
|
|
if proto == -1:
|
|
|
|
pkt_info.proto = random.choice(self.proto[self.IP])
|
|
|
|
else:
|
|
|
|
pkt_info.proto = proto
|
|
|
|
payload = self.info_to_payload(pkt_info)
|
|
|
|
p = Ether(dst=dst_host.mac, src=src_host.mac)
|
|
|
|
if etype > 0:
|
2022-04-26 19:02:15 +02:00
|
|
|
p = Ether(dst=dst_host.mac, src=src_host.mac, type=etype)
|
2018-02-08 21:45:08 +01:00
|
|
|
if pkt_info.ip:
|
|
|
|
p /= IPv6(dst=dst_host.ip6, src=src_host.ip6)
|
|
|
|
if fragments:
|
|
|
|
p /= IPv6ExtHdrFragment(offset=64, m=1)
|
|
|
|
else:
|
|
|
|
if fragments:
|
2022-04-26 19:02:15 +02:00
|
|
|
p /= IP(
|
|
|
|
src=src_host.ip4, dst=dst_host.ip4, flags=1, frag=64
|
|
|
|
)
|
2018-02-08 21:45:08 +01:00
|
|
|
else:
|
|
|
|
p /= IP(src=src_host.ip4, dst=dst_host.ip4)
|
|
|
|
if traffic_type == self.ICMP:
|
|
|
|
if pkt_info.ip:
|
2022-04-26 19:02:15 +02:00
|
|
|
p /= ICMPv6EchoRequest(
|
|
|
|
type=self.icmp6_type, code=self.icmp6_code
|
|
|
|
)
|
2018-02-08 21:45:08 +01:00
|
|
|
else:
|
2022-04-26 19:02:15 +02:00
|
|
|
p /= ICMP(type=self.icmp4_type, code=self.icmp4_code)
|
2018-02-08 21:45:08 +01:00
|
|
|
else:
|
|
|
|
p /= self.create_upper_layer(i, pkt_info.proto, ports)
|
|
|
|
if pkt_raw:
|
|
|
|
p /= Raw(payload)
|
|
|
|
pkt_info.data = p.copy()
|
|
|
|
if pkt_raw:
|
|
|
|
size = random.choice(packet_sizes)
|
|
|
|
self.extend_packet(p, size)
|
|
|
|
pkts.append(p)
|
|
|
|
return pkts
|
|
|
|
|
2022-04-26 19:02:15 +02:00
|
|
|
def verify_capture(self, pg_if, capture, traffic_type=0, ip_type=0, etype=-1):
|
2018-02-08 21:45:08 +01:00
|
|
|
"""
|
|
|
|
Verify captured input packet stream for defined interface.
|
|
|
|
|
|
|
|
:param object pg_if: Interface to verify captured packet stream for.
|
|
|
|
:param list capture: Captured packet stream.
|
|
|
|
:param traffic_type: 1: ICMP packet, 2: IPv6 with EH, 0: otherwise.
|
|
|
|
"""
|
|
|
|
last_info = dict()
|
|
|
|
for i in self.pg_interfaces:
|
|
|
|
last_info[i.sw_if_index] = None
|
|
|
|
dst_sw_if_index = pg_if.sw_if_index
|
|
|
|
for packet in capture:
|
|
|
|
if etype > 0:
|
|
|
|
if packet[Ether].type != etype:
|
2022-04-26 19:02:15 +02:00
|
|
|
self.logger.error(ppp("Unexpected ethertype in packet:", packet))
|
2018-02-08 21:45:08 +01:00
|
|
|
else:
|
|
|
|
continue
|
|
|
|
try:
|
|
|
|
# Raw data for ICMPv6 are stored in ICMPv6EchoRequest.data
|
|
|
|
if traffic_type == self.ICMP and ip_type == self.IPV6:
|
2022-04-26 19:02:15 +02:00
|
|
|
payload_info = self.payload_to_info(packet[ICMPv6EchoRequest].data)
|
2018-02-08 21:45:08 +01:00
|
|
|
payload = packet[ICMPv6EchoRequest]
|
|
|
|
else:
|
2019-03-06 11:58:06 -08:00
|
|
|
payload_info = self.payload_to_info(packet[Raw])
|
2018-02-08 21:45:08 +01:00
|
|
|
payload = packet[self.proto_map[payload_info.proto]]
|
|
|
|
except:
|
2022-04-26 19:02:15 +02:00
|
|
|
self.logger.error(
|
|
|
|
ppp("Unexpected or invalid packet (outside network):", packet)
|
|
|
|
)
|
2018-02-08 21:45:08 +01:00
|
|
|
raise
|
|
|
|
|
|
|
|
if ip_type != 0:
|
|
|
|
self.assertEqual(payload_info.ip, ip_type)
|
|
|
|
if traffic_type == self.ICMP:
|
|
|
|
try:
|
|
|
|
if payload_info.ip == 0:
|
|
|
|
self.assertEqual(payload.type, self.icmp4_type)
|
|
|
|
self.assertEqual(payload.code, self.icmp4_code)
|
|
|
|
else:
|
|
|
|
self.assertEqual(payload.type, self.icmp6_type)
|
|
|
|
self.assertEqual(payload.code, self.icmp6_code)
|
|
|
|
except:
|
2022-04-26 19:02:15 +02:00
|
|
|
self.logger.error(
|
|
|
|
ppp("Unexpected or invalid packet (outside network):", packet)
|
|
|
|
)
|
2018-02-08 21:45:08 +01:00
|
|
|
raise
|
|
|
|
else:
|
|
|
|
try:
|
|
|
|
ip_version = IPv6 if payload_info.ip == 1 else IP
|
|
|
|
|
|
|
|
ip = packet[ip_version]
|
|
|
|
packet_index = payload_info.index
|
|
|
|
|
|
|
|
self.assertEqual(payload_info.dst, dst_sw_if_index)
|
2022-04-26 19:02:15 +02:00
|
|
|
self.logger.debug(
|
|
|
|
"Got packet on port %s: src=%u (id=%u)"
|
|
|
|
% (pg_if.name, payload_info.src, packet_index)
|
|
|
|
)
|
2018-02-08 21:45:08 +01:00
|
|
|
next_info = self.get_next_packet_info_for_interface2(
|
2022-04-26 19:02:15 +02:00
|
|
|
payload_info.src, dst_sw_if_index, last_info[payload_info.src]
|
|
|
|
)
|
2018-02-08 21:45:08 +01:00
|
|
|
last_info[payload_info.src] = next_info
|
|
|
|
self.assertTrue(next_info is not None)
|
|
|
|
self.assertEqual(packet_index, next_info.index)
|
|
|
|
saved_packet = next_info.data
|
|
|
|
# Check standard fields
|
|
|
|
self.assertEqual(ip.src, saved_packet[ip_version].src)
|
|
|
|
self.assertEqual(ip.dst, saved_packet[ip_version].dst)
|
|
|
|
p = self.proto_map[payload_info.proto]
|
2022-04-26 19:02:15 +02:00
|
|
|
if p == "TCP":
|
2018-02-08 21:45:08 +01:00
|
|
|
tcp = packet[TCP]
|
2022-04-26 19:02:15 +02:00
|
|
|
self.assertEqual(tcp.sport, saved_packet[TCP].sport)
|
|
|
|
self.assertEqual(tcp.dport, saved_packet[TCP].dport)
|
|
|
|
elif p == "UDP":
|
2018-02-08 21:45:08 +01:00
|
|
|
udp = packet[UDP]
|
2022-04-26 19:02:15 +02:00
|
|
|
self.assertEqual(udp.sport, saved_packet[UDP].sport)
|
|
|
|
self.assertEqual(udp.dport, saved_packet[UDP].dport)
|
2018-02-08 21:45:08 +01:00
|
|
|
except:
|
2022-04-26 19:02:15 +02:00
|
|
|
self.logger.error(ppp("Unexpected or invalid packet:", packet))
|
2018-02-08 21:45:08 +01:00
|
|
|
raise
|
|
|
|
for i in self.pg_interfaces:
|
|
|
|
remaining_packet = self.get_next_packet_info_for_interface2(
|
2022-04-26 19:02:15 +02:00
|
|
|
i, dst_sw_if_index, last_info[i.sw_if_index]
|
|
|
|
)
|
2018-02-08 21:45:08 +01:00
|
|
|
self.assertTrue(
|
|
|
|
remaining_packet is None,
|
2022-04-26 19:02:15 +02:00
|
|
|
"Port %u: Packet expected from source %u didn't arrive"
|
|
|
|
% (dst_sw_if_index, i.sw_if_index),
|
|
|
|
)
|
2018-02-08 21:45:08 +01:00
|
|
|
|
|
|
|
def run_traffic_no_check(self):
|
|
|
|
# Test
|
|
|
|
# Create incoming packet streams for packet-generator interfaces
|
|
|
|
for i in self.pg_interfaces:
|
|
|
|
if self.flows.__contains__(i):
|
|
|
|
pkts = self.create_stream(i, self.pg_if_packet_sizes)
|
|
|
|
if len(pkts) > 0:
|
|
|
|
i.add_stream(pkts)
|
|
|
|
|
|
|
|
# Enable packet capture and start packet sending
|
|
|
|
self.pg_enable_capture(self.pg_interfaces)
|
|
|
|
self.pg_start()
|
|
|
|
|
2022-04-26 19:02:15 +02:00
|
|
|
def run_verify_test(
|
|
|
|
self,
|
|
|
|
traffic_type=0,
|
|
|
|
ip_type=0,
|
|
|
|
proto=-1,
|
|
|
|
ports=0,
|
|
|
|
frags=False,
|
|
|
|
pkt_raw=True,
|
|
|
|
etype=-1,
|
|
|
|
):
|
2018-02-08 21:45:08 +01:00
|
|
|
# Test
|
|
|
|
# Create incoming packet streams for packet-generator interfaces
|
|
|
|
pkts_cnt = 0
|
|
|
|
for i in self.pg_interfaces:
|
|
|
|
if self.flows.__contains__(i):
|
2022-04-26 19:02:15 +02:00
|
|
|
pkts = self.create_stream(
|
|
|
|
i,
|
|
|
|
self.pg_if_packet_sizes,
|
|
|
|
traffic_type,
|
|
|
|
ip_type,
|
|
|
|
proto,
|
|
|
|
ports,
|
|
|
|
frags,
|
|
|
|
pkt_raw,
|
|
|
|
etype,
|
|
|
|
)
|
2018-02-08 21:45:08 +01:00
|
|
|
if len(pkts) > 0:
|
|
|
|
i.add_stream(pkts)
|
|
|
|
pkts_cnt += len(pkts)
|
|
|
|
|
|
|
|
# Enable packet capture and start packet sendingself.IPV
|
|
|
|
self.pg_enable_capture(self.pg_interfaces)
|
|
|
|
self.pg_start()
|
|
|
|
|
|
|
|
# Verify
|
|
|
|
# Verify outgoing packet streams per packet-generator interface
|
|
|
|
for src_if in self.pg_interfaces:
|
|
|
|
if self.flows.__contains__(src_if):
|
|
|
|
for dst_if in self.flows[src_if]:
|
|
|
|
capture = dst_if.get_capture(pkts_cnt)
|
2022-04-26 19:02:15 +02:00
|
|
|
self.logger.info("Verifying capture on interface %s" % dst_if.name)
|
|
|
|
self.verify_capture(dst_if, capture, traffic_type, ip_type, etype)
|
2018-02-08 21:45:08 +01:00
|
|
|
|
2022-04-26 19:02:15 +02:00
|
|
|
def run_verify_negat_test(
|
|
|
|
self, traffic_type=0, ip_type=0, proto=-1, ports=0, frags=False, etype=-1
|
|
|
|
):
|
2018-02-08 21:45:08 +01:00
|
|
|
# Test
|
|
|
|
self.reset_packet_infos()
|
|
|
|
for i in self.pg_interfaces:
|
|
|
|
if self.flows.__contains__(i):
|
2022-04-26 19:02:15 +02:00
|
|
|
pkts = self.create_stream(
|
|
|
|
i,
|
|
|
|
self.pg_if_packet_sizes,
|
|
|
|
traffic_type,
|
|
|
|
ip_type,
|
|
|
|
proto,
|
|
|
|
ports,
|
|
|
|
frags,
|
|
|
|
True,
|
|
|
|
etype,
|
|
|
|
)
|
2018-02-08 21:45:08 +01:00
|
|
|
if len(pkts) > 0:
|
|
|
|
i.add_stream(pkts)
|
|
|
|
|
|
|
|
# Enable packet capture and start packet sending
|
|
|
|
self.pg_enable_capture(self.pg_interfaces)
|
|
|
|
self.pg_start()
|
|
|
|
|
|
|
|
# Verify
|
|
|
|
# Verify outgoing packet streams per packet-generator interface
|
|
|
|
for src_if in self.pg_interfaces:
|
|
|
|
if self.flows.__contains__(src_if):
|
|
|
|
for dst_if in self.flows[src_if]:
|
2022-04-26 19:02:15 +02:00
|
|
|
self.logger.info("Verifying capture on interface %s" % dst_if.name)
|
2018-02-08 21:45:08 +01:00
|
|
|
capture = dst_if.get_capture(0)
|
|
|
|
self.assertEqual(len(capture), 0)
|
|
|
|
|
2022-04-26 19:02:15 +02:00
|
|
|
def build_classify_table(
|
|
|
|
self,
|
|
|
|
src_mac="",
|
|
|
|
dst_mac="",
|
|
|
|
ether_type="",
|
|
|
|
etype="",
|
|
|
|
key="mac",
|
|
|
|
hit_next_index=0xFFFFFFFF,
|
|
|
|
):
|
2018-07-03 13:58:24 +02:00
|
|
|
# Basic ACL testing
|
2022-04-26 19:02:15 +02:00
|
|
|
a_mask = self.build_mac_mask(
|
|
|
|
src_mac=src_mac, dst_mac=dst_mac, ether_type=ether_type
|
|
|
|
)
|
2018-07-03 13:58:24 +02:00
|
|
|
self.create_classify_table(key, a_mask)
|
2018-02-08 21:45:08 +01:00
|
|
|
for host in self.hosts_by_pg_idx[self.pg0.sw_if_index]:
|
2022-04-26 19:02:15 +02:00
|
|
|
s_mac = host.mac if src_mac else ""
|
2018-07-03 13:58:24 +02:00
|
|
|
if dst_mac:
|
|
|
|
for dst_if in self.flows[self.pg0]:
|
|
|
|
for dst_host in self.hosts_by_pg_idx[dst_if.sw_if_index]:
|
|
|
|
self.create_classify_session(
|
2022-04-26 19:02:15 +02:00
|
|
|
self.pg0,
|
|
|
|
self.acl_tbl_idx.get(key),
|
|
|
|
self.build_mac_match(
|
|
|
|
src_mac=s_mac, dst_mac=dst_host.mac, ether_type=etype
|
|
|
|
),
|
|
|
|
hit_next_index=hit_next_index,
|
|
|
|
)
|
2018-07-03 13:58:24 +02:00
|
|
|
else:
|
|
|
|
self.create_classify_session(
|
2022-04-26 19:02:15 +02:00
|
|
|
self.pg0,
|
|
|
|
self.acl_tbl_idx.get(key),
|
|
|
|
self.build_mac_match(src_mac=s_mac, dst_mac="", ether_type=etype),
|
|
|
|
hit_next_index=hit_next_index,
|
|
|
|
)
|
2018-02-08 21:45:08 +01:00
|
|
|
|
|
|
|
def test_0000_warmup_test(self):
|
2022-04-26 19:02:15 +02:00
|
|
|
"""Learn the MAC addresses"""
|
2018-02-08 21:45:08 +01:00
|
|
|
self.create_hosts(2)
|
|
|
|
self.run_traffic_no_check()
|
|
|
|
|
2018-07-03 13:58:24 +02:00
|
|
|
def test_0010_inacl_permit_src_mac(self):
|
2022-04-26 19:02:15 +02:00
|
|
|
"""Input L2 ACL test - permit source MAC
|
2018-02-08 21:45:08 +01:00
|
|
|
|
|
|
|
Test scenario for basic IP ACL with source IP
|
|
|
|
- Create IPv4 stream for pg0 -> pg1 interface.
|
|
|
|
- Create ACL with source MAC address.
|
|
|
|
- Send and verify received packets on pg1 interface.
|
|
|
|
"""
|
2022-04-26 19:02:15 +02:00
|
|
|
key = "mac_in"
|
|
|
|
self.build_classify_table(src_mac="ffffffffffff", key=key)
|
2018-07-03 13:58:24 +02:00
|
|
|
self.input_acl_set_interface(self.pg0, self.acl_tbl_idx.get(key))
|
|
|
|
self.acl_active_table = key
|
|
|
|
self.run_verify_test(self.IP, self.IPV4, -1)
|
|
|
|
|
|
|
|
def test_0011_inacl_permit_dst_mac(self):
|
2022-04-26 19:02:15 +02:00
|
|
|
"""Input L2 ACL test - permit destination MAC
|
2018-07-03 13:58:24 +02:00
|
|
|
|
|
|
|
Test scenario for basic IP ACL with source IP
|
|
|
|
- Create IPv4 stream for pg0 -> pg1 interface.
|
|
|
|
- Create ACL with destination MAC address.
|
|
|
|
- Send and verify received packets on pg1 interface.
|
|
|
|
"""
|
2022-04-26 19:02:15 +02:00
|
|
|
key = "mac_in"
|
|
|
|
self.build_classify_table(dst_mac="ffffffffffff", key=key)
|
2018-07-03 13:58:24 +02:00
|
|
|
self.input_acl_set_interface(self.pg0, self.acl_tbl_idx.get(key))
|
|
|
|
self.acl_active_table = key
|
|
|
|
self.run_verify_test(self.IP, self.IPV4, -1)
|
|
|
|
|
|
|
|
def test_0012_inacl_permit_src_dst_mac(self):
|
2022-04-26 19:02:15 +02:00
|
|
|
"""Input L2 ACL test - permit source and destination MAC
|
2018-07-03 13:58:24 +02:00
|
|
|
|
|
|
|
Test scenario for basic IP ACL with source IP
|
|
|
|
- Create IPv4 stream for pg0 -> pg1 interface.
|
|
|
|
- Create ACL with source and destination MAC addresses.
|
|
|
|
- Send and verify received packets on pg1 interface.
|
|
|
|
"""
|
2022-04-26 19:02:15 +02:00
|
|
|
key = "mac_in"
|
2018-07-03 13:58:24 +02:00
|
|
|
self.build_classify_table(
|
2022-04-26 19:02:15 +02:00
|
|
|
src_mac="ffffffffffff", dst_mac="ffffffffffff", key=key
|
|
|
|
)
|
2018-07-03 13:58:24 +02:00
|
|
|
self.input_acl_set_interface(self.pg0, self.acl_tbl_idx.get(key))
|
|
|
|
self.acl_active_table = key
|
|
|
|
self.run_verify_test(self.IP, self.IPV4, -1)
|
|
|
|
|
|
|
|
def test_0013_inacl_permit_ether_type(self):
|
2022-04-26 19:02:15 +02:00
|
|
|
"""Input L2 ACL test - permit ether_type
|
2018-07-03 13:58:24 +02:00
|
|
|
|
|
|
|
Test scenario for basic IP ACL with source IP
|
|
|
|
- Create IPv4 stream for pg0 -> pg1 interface.
|
|
|
|
- Create ACL with destination MAC address.
|
|
|
|
- Send and verify received packets on pg1 interface.
|
|
|
|
"""
|
2022-04-26 19:02:15 +02:00
|
|
|
key = "mac_in"
|
|
|
|
self.build_classify_table(ether_type="ffff", etype=hex(ETH_P_IP)[2:], key=key)
|
2018-07-03 13:58:24 +02:00
|
|
|
self.input_acl_set_interface(self.pg0, self.acl_tbl_idx.get(key))
|
|
|
|
self.acl_active_table = key
|
2018-02-08 21:45:08 +01:00
|
|
|
self.run_verify_test(self.IP, self.IPV4, -1)
|
|
|
|
|
|
|
|
def test_0015_inacl_deny(self):
|
2022-04-26 19:02:15 +02:00
|
|
|
"""Input L2 ACL test - deny
|
2018-02-08 21:45:08 +01:00
|
|
|
|
|
|
|
Test scenario for basic IP ACL with source IP
|
|
|
|
- Create IPv4 stream for pg0 -> pg1 interface.
|
2018-07-03 13:58:24 +02:00
|
|
|
|
2018-02-08 21:45:08 +01:00
|
|
|
- Create ACL with source MAC address.
|
|
|
|
- Send and verify no received packets on pg1 interface.
|
|
|
|
"""
|
2022-04-26 19:02:15 +02:00
|
|
|
key = "mac_in"
|
|
|
|
self.build_classify_table(src_mac="ffffffffffff", hit_next_index=0, key=key)
|
2018-07-03 13:58:24 +02:00
|
|
|
self.input_acl_set_interface(self.pg0, self.acl_tbl_idx.get(key))
|
|
|
|
self.acl_active_table = key
|
2018-02-08 21:45:08 +01:00
|
|
|
self.run_verify_negat_test(self.IP, self.IPV4, -1)
|
|
|
|
|
|
|
|
def test_0020_outacl_permit(self):
|
2022-04-26 19:02:15 +02:00
|
|
|
"""Output L2 ACL test - permit
|
2018-02-08 21:45:08 +01:00
|
|
|
|
|
|
|
Test scenario for basic IP ACL with source IP
|
|
|
|
- Create IPv4 stream for pg0 -> pg1 interface.
|
|
|
|
- Create ACL with source MAC address.
|
|
|
|
- Send and verify received packets on pg1 interface.
|
|
|
|
"""
|
2022-04-26 19:02:15 +02:00
|
|
|
key = "mac_out"
|
|
|
|
self.build_classify_table(src_mac="ffffffffffff", key=key)
|
2018-07-03 13:58:24 +02:00
|
|
|
self.output_acl_set_interface(self.pg1, self.acl_tbl_idx.get(key))
|
|
|
|
self.acl_active_table = key
|
2018-02-08 21:45:08 +01:00
|
|
|
self.run_verify_test(self.IP, self.IPV4, -1)
|
|
|
|
|
|
|
|
def test_0025_outacl_deny(self):
|
2022-04-26 19:02:15 +02:00
|
|
|
"""Output L2 ACL test - deny
|
2018-02-08 21:45:08 +01:00
|
|
|
|
|
|
|
Test scenario for basic IP ACL with source IP
|
|
|
|
- Create IPv4 stream for pg0 -> pg1 interface.
|
|
|
|
- Create ACL with source MAC address.
|
|
|
|
- Send and verify no received packets on pg1 interface.
|
|
|
|
"""
|
2022-04-26 19:02:15 +02:00
|
|
|
key = "mac_out"
|
|
|
|
self.build_classify_table(src_mac="ffffffffffff", hit_next_index=0, key=key)
|
2018-07-03 13:58:24 +02:00
|
|
|
self.output_acl_set_interface(self.pg1, self.acl_tbl_idx.get(key))
|
|
|
|
self.acl_active_table = key
|
2018-02-08 21:45:08 +01:00
|
|
|
self.run_verify_negat_test(self.IP, self.IPV4, -1)
|
|
|
|
|
|
|
|
def test_0030_inoutacl_permit(self):
|
2022-04-26 19:02:15 +02:00
|
|
|
"""Input+Output L2 ACL test - permit
|
2018-02-08 21:45:08 +01:00
|
|
|
|
|
|
|
Test scenario for basic IP ACL with source IP
|
|
|
|
- Create IPv4 stream for pg0 -> pg1 interface.
|
|
|
|
- Create ACLs with source MAC address.
|
|
|
|
- Send and verify received packets on pg1 interface.
|
|
|
|
"""
|
2022-04-26 19:02:15 +02:00
|
|
|
key = "mac_inout"
|
|
|
|
self.build_classify_table(src_mac="ffffffffffff", key=key)
|
2018-07-03 13:58:24 +02:00
|
|
|
self.output_acl_set_interface(self.pg1, self.acl_tbl_idx.get(key))
|
|
|
|
self.input_acl_set_interface(self.pg0, self.acl_tbl_idx.get(key))
|
|
|
|
self.acl_active_table = key
|
2018-02-08 21:45:08 +01:00
|
|
|
self.run_verify_test(self.IP, self.IPV4, -1)
|
|
|
|
|
2020-03-27 06:55:06 +01:00
|
|
|
|
2022-04-26 19:02:15 +02:00
|
|
|
if __name__ == "__main__":
|
2018-02-08 21:45:08 +01:00
|
|
|
unittest.main(testRunner=VppTestRunner)
|