vpp/test/vpp_ipsec.py
Arthur de Kerhor ad95b06181 ipsec: add per-SA error counters
Error counters are added on a per-node basis. In Ipsec, it is
useful to also track the errors that occured per SA.

Type: feature
Change-Id: Iabcdcb439f67ad3c6c202b36ffc44ab39abac1bc
Signed-off-by: Arthur de Kerhor <arthurdekerhor@gmail.com>
2023-03-23 08:58:55 +00:00

473 lines
14 KiB
Python

from vpp_object import VppObject
from ipaddress import ip_address
from vpp_papi import VppEnum
from vpp_interface import VppInterface
try:
text_type = unicode
except NameError:
text_type = str
def mk_counter():
return {"packets": 0, "bytes": 0}
class VppIpsecSpd(VppObject):
"""
VPP SPD DB
"""
def __init__(self, test, id):
self.test = test
self.id = id
def add_vpp_config(self):
self.test.vapi.ipsec_spd_add_del(self.id)
self.test.registry.register(self, self.test.logger)
def remove_vpp_config(self):
self.test.vapi.ipsec_spd_add_del(self.id, is_add=0)
def object_id(self):
return "ipsec-spd-%d" % self.id
def query_vpp_config(self):
spds = self.test.vapi.ipsec_spds_dump()
for spd in spds:
if spd.spd_id == self.id:
return True
return False
class VppIpsecSpdItfBinding(VppObject):
"""
VPP SPD DB to interface binding
(i.e. this SPD is used on this interface)
"""
def __init__(self, test, spd, itf):
self.test = test
self.spd = spd
self.itf = itf
def add_vpp_config(self):
self.test.vapi.ipsec_interface_add_del_spd(self.spd.id, self.itf.sw_if_index)
self.test.registry.register(self, self.test.logger)
def remove_vpp_config(self):
self.test.vapi.ipsec_interface_add_del_spd(
self.spd.id, self.itf.sw_if_index, is_add=0
)
def object_id(self):
return "bind-%s-to-%s" % (self.spd.id, self.itf)
def query_vpp_config(self):
bs = self.test.vapi.ipsec_spd_interface_dump()
for b in bs:
if b.sw_if_index == self.itf.sw_if_index:
return True
return False
class VppIpsecSpdEntry(VppObject):
"""
VPP SPD DB Entry
"""
def __init__(
self,
test,
spd,
sa_id,
local_start,
local_stop,
remote_start,
remote_stop,
proto=socket.IPPROTO_RAW,
priority=100,
policy=None,
is_outbound=1,
remote_port_start=0,
remote_port_stop=65535,
local_port_start=0,
local_port_stop=65535,
):
self.test = test
self.spd = spd
self.sa_id = sa_id
self.local_start = ip_address(text_type(local_start))
self.local_stop = ip_address(text_type(local_stop))
self.remote_start = ip_address(text_type(remote_start))
self.remote_stop = ip_address(text_type(remote_stop))
self.proto = proto
self.is_outbound = is_outbound
self.priority = priority
if not policy:
self.policy = VppEnum.vl_api_ipsec_spd_action_t.IPSEC_API_SPD_ACTION_BYPASS
else:
self.policy = policy
self.is_ipv6 = 0 if self.local_start.version == 4 else 1
self.local_port_start = local_port_start
self.local_port_stop = local_port_stop
self.remote_port_start = remote_port_start
self.remote_port_stop = remote_port_stop
def add_vpp_config(self):
rv = self.test.vapi.ipsec_spd_entry_add_del(
self.spd.id,
self.sa_id,
self.local_start,
self.local_stop,
self.remote_start,
self.remote_stop,
protocol=self.proto,
is_ipv6=self.is_ipv6,
is_outbound=self.is_outbound,
priority=self.priority,
policy=self.policy,
local_port_start=self.local_port_start,
local_port_stop=self.local_port_stop,
remote_port_start=self.remote_port_start,
remote_port_stop=self.remote_port_stop,
)
self.stat_index = rv.stat_index
self.test.registry.register(self, self.test.logger)
return self
def remove_vpp_config(self):
self.test.vapi.ipsec_spd_entry_add_del(
self.spd.id,
self.sa_id,
self.local_start,
self.local_stop,
self.remote_start,
self.remote_stop,
protocol=self.proto,
is_ipv6=self.is_ipv6,
is_outbound=self.is_outbound,
priority=self.priority,
policy=self.policy,
local_port_start=self.local_port_start,
local_port_stop=self.local_port_stop,
remote_port_start=self.remote_port_start,
remote_port_stop=self.remote_port_stop,
is_add=0,
)
def object_id(self):
return "spd-entry-%d-%d-%d-%d-%d-%d" % (
self.spd.id,
self.priority,
self.policy,
self.is_outbound,
self.is_ipv6,
self.remote_port_start,
)
def query_vpp_config(self):
ss = self.test.vapi.ipsec_spd_dump(self.spd.id)
for s in ss:
if (
s.entry.sa_id == self.sa_id
and s.entry.is_outbound == self.is_outbound
and s.entry.priority == self.priority
and s.entry.policy == self.policy
and s.entry.remote_address_start == self.remote_start
and s.entry.remote_port_start == self.remote_port_start
):
return True
return False
def get_stats(self, worker=None):
c = self.test.statistics.get_counter("/net/ipsec/policy")
if worker is None:
total = mk_counter()
for t in c:
total["packets"] += t[self.stat_index]["packets"]
return total
else:
# +1 to skip main thread
return c[worker + 1][self.stat_index]
class VppIpsecSA(VppObject):
"""
VPP SAD Entry
"""
DEFAULT_UDP_PORT = 4500
def __init__(
self,
test,
id,
spi,
integ_alg,
integ_key,
crypto_alg,
crypto_key,
proto,
tun_src=None,
tun_dst=None,
flags=None,
salt=0,
tun_flags=None,
dscp=None,
udp_src=None,
udp_dst=None,
hop_limit=None,
):
e = VppEnum.vl_api_ipsec_sad_flags_t
self.test = test
self.id = id
self.spi = spi
self.integ_alg = integ_alg
self.integ_key = integ_key
self.crypto_alg = crypto_alg
self.crypto_key = crypto_key
self.proto = proto
self.salt = salt
self.table_id = 0
self.tun_src = tun_src
self.tun_dst = tun_dst
if not flags:
self.flags = e.IPSEC_API_SAD_FLAG_NONE
else:
self.flags = flags
if tun_src:
self.tun_src = ip_address(text_type(tun_src))
self.flags = self.flags | e.IPSEC_API_SAD_FLAG_IS_TUNNEL
if tun_dst:
self.tun_dst = ip_address(text_type(tun_dst))
self.udp_src = udp_src
self.udp_dst = udp_dst
self.tun_flags = (
VppEnum.vl_api_tunnel_encap_decap_flags_t.TUNNEL_API_ENCAP_DECAP_FLAG_NONE
)
if tun_flags:
self.tun_flags = tun_flags
self.dscp = VppEnum.vl_api_ip_dscp_t.IP_API_DSCP_CS0
if dscp:
self.dscp = dscp
self.hop_limit = 255
if hop_limit:
self.hop_limit = hop_limit
def tunnel_encode(self):
return {
"src": (self.tun_src if self.tun_src else []),
"dst": (self.tun_dst if self.tun_dst else []),
"encap_decap_flags": self.tun_flags,
"dscp": self.dscp,
"hop_limit": self.hop_limit,
"table_id": self.table_id,
}
def add_vpp_config(self):
entry = {
"sad_id": self.id,
"spi": self.spi,
"integrity_algorithm": self.integ_alg,
"integrity_key": {
"length": len(self.integ_key),
"data": self.integ_key,
},
"crypto_algorithm": self.crypto_alg,
"crypto_key": {
"data": self.crypto_key,
"length": len(self.crypto_key),
},
"protocol": self.proto,
"tunnel": self.tunnel_encode(),
"flags": self.flags,
"salt": self.salt,
}
# don't explicitly send the defaults, let papi fill them in
if self.udp_src:
entry["udp_src_port"] = self.udp_src
if self.udp_dst:
entry["udp_dst_port"] = self.udp_dst
r = self.test.vapi.ipsec_sad_entry_add(entry=entry)
self.stat_index = r.stat_index
self.test.registry.register(self, self.test.logger)
return self
def update_vpp_config(
self, udp_src=None, udp_dst=None, is_tun=False, tun_src=None, tun_dst=None
):
if is_tun:
if tun_src:
self.tun_src = ip_address(text_type(tun_src))
if tun_dst:
self.tun_dst = ip_address(text_type(tun_dst))
if udp_src:
self.udp_src = udp_src
if udp_dst:
self.udp_dst = udp_dst
self.test.vapi.ipsec_sad_entry_update(
sad_id=self.id,
is_tun=is_tun,
tunnel=self.tunnel_encode(),
udp_src_port=udp_src,
udp_dst_port=udp_dst,
)
def remove_vpp_config(self):
self.test.vapi.ipsec_sad_entry_del(id=self.id)
def object_id(self):
return "ipsec-sa-%d" % self.id
def query_vpp_config(self):
e = VppEnum.vl_api_ipsec_sad_flags_t
bs = self.test.vapi.ipsec_sa_v3_dump()
for b in bs:
if b.entry.sad_id == self.id:
# if udp encap is configured then the ports should match
# those configured or the default
if self.flags & e.IPSEC_API_SAD_FLAG_UDP_ENCAP:
if not b.entry.flags & e.IPSEC_API_SAD_FLAG_UDP_ENCAP:
return False
if self.udp_src:
if self.udp_src != b.entry.udp_src_port:
return False
else:
if self.DEFAULT_UDP_PORT != b.entry.udp_src_port:
return False
if self.udp_dst:
if self.udp_dst != b.entry.udp_dst_port:
return False
else:
if self.DEFAULT_UDP_PORT != b.entry.udp_dst_port:
return False
return True
return False
def get_stats(self, worker=None):
c = self.test.statistics.get_counter("/net/ipsec/sa")
if worker is None:
total = mk_counter()
for t in c:
total["packets"] += t[self.stat_index]["packets"]
return total
else:
# +1 to skip main thread
return c[worker + 1][self.stat_index]
def get_err(self, name, worker=None):
c = self.test.statistics.get_counter("/net/ipsec/sa/err/" + name)
if worker is None:
total = 0
for t in c:
total += t[self.stat_index]
return total
else:
# +1 to skip main thread
return c[worker + 1][self.stat_index]
class VppIpsecTunProtect(VppObject):
"""
VPP IPSEC tunnel protection
"""
def __init__(self, test, itf, sa_out, sas_in, nh=None):
self.test = test
self.itf = itf
self.sas_in = []
for sa in sas_in:
self.sas_in.append(sa.id)
self.sa_out = sa_out.id
self.nh = nh
if not self.nh:
self.nh = "0.0.0.0"
def update_vpp_config(self, sa_out, sas_in):
self.sas_in = []
for sa in sas_in:
self.sas_in.append(sa.id)
self.sa_out = sa_out.id
self.test.vapi.ipsec_tunnel_protect_update(
tunnel={
"sw_if_index": self.itf._sw_if_index,
"n_sa_in": len(self.sas_in),
"sa_out": self.sa_out,
"sa_in": self.sas_in,
"nh": self.nh,
}
)
def object_id(self):
return "ipsec-tun-protect-%s-%s" % (self.itf, self.nh)
def add_vpp_config(self):
self.test.vapi.ipsec_tunnel_protect_update(
tunnel={
"sw_if_index": self.itf._sw_if_index,
"n_sa_in": len(self.sas_in),
"sa_out": self.sa_out,
"sa_in": self.sas_in,
"nh": self.nh,
}
)
self.test.registry.register(self, self.test.logger)
def remove_vpp_config(self):
self.test.vapi.ipsec_tunnel_protect_del(
sw_if_index=self.itf.sw_if_index, nh=self.nh
)
def query_vpp_config(self):
bs = self.test.vapi.ipsec_tunnel_protect_dump(sw_if_index=self.itf.sw_if_index)
for b in bs:
if b.tun.sw_if_index == self.itf.sw_if_index and self.nh == str(b.tun.nh):
return True
return False
class VppIpsecInterface(VppInterface):
"""
VPP IPSec interface
"""
def __init__(self, test, mode=None, instance=0xFFFFFFFF):
super(VppIpsecInterface, self).__init__(test)
self.mode = mode
if not self.mode:
self.mode = VppEnum.vl_api_tunnel_mode_t.TUNNEL_API_MODE_P2P
self.instance = instance
def add_vpp_config(self):
r = self.test.vapi.ipsec_itf_create(
itf={
"user_instance": self.instance,
"mode": self.mode,
}
)
self.set_sw_if_index(r.sw_if_index)
self.test.registry.register(self, self.test.logger)
ts = self.test.vapi.ipsec_itf_dump(sw_if_index=self._sw_if_index)
self.instance = ts[0].itf.user_instance
return self
def remove_vpp_config(self):
self.test.vapi.ipsec_itf_delete(sw_if_index=self._sw_if_index)
def query_vpp_config(self):
ts = self.test.vapi.ipsec_itf_dump(sw_if_index=0xFFFFFFFF)
for t in ts:
if t.itf.sw_if_index == self._sw_if_index:
return True
return False
def __str__(self):
return self.object_id()
def object_id(self):
return "ipsec%d" % self.instance