2018-08-08 01:06:40 -07:00
|
|
|
"""
|
|
|
|
IP Types
|
|
|
|
|
|
|
|
"""
|
2018-12-09 08:52:14 -08:00
|
|
|
import logging
|
2018-08-08 01:06:40 -07:00
|
|
|
|
|
|
|
from ipaddress import ip_address
|
2018-09-05 15:42:26 -07:00
|
|
|
from socket import AF_INET, AF_INET6
|
2018-10-16 14:42:50 +02:00
|
|
|
from vpp_papi import VppEnum
|
2019-01-29 11:51:44 -08:00
|
|
|
try:
|
|
|
|
text_type = unicode
|
|
|
|
except NameError:
|
|
|
|
text_type = str
|
2018-08-08 01:06:40 -07:00
|
|
|
|
2018-12-09 08:52:14 -08:00
|
|
|
_log = logging.getLogger(__name__)
|
|
|
|
|
2018-08-08 01:06:40 -07:00
|
|
|
|
2018-09-05 15:42:26 -07:00
|
|
|
class DpoProto:
|
|
|
|
DPO_PROTO_IP4 = 0
|
|
|
|
DPO_PROTO_IP6 = 1
|
|
|
|
DPO_PROTO_MPLS = 2
|
|
|
|
DPO_PROTO_ETHERNET = 3
|
|
|
|
DPO_PROTO_BIER = 4
|
|
|
|
DPO_PROTO_NSH = 5
|
2018-08-08 01:06:40 -07:00
|
|
|
|
|
|
|
|
2018-09-05 15:42:26 -07:00
|
|
|
INVALID_INDEX = 0xffffffff
|
2018-08-08 01:06:40 -07:00
|
|
|
|
|
|
|
|
|
|
|
class VppIpAddressUnion():
|
|
|
|
def __init__(self, addr):
|
|
|
|
self.addr = addr
|
2019-01-29 11:51:44 -08:00
|
|
|
self.ip_addr = ip_address(text_type(self.addr))
|
2018-08-08 01:06:40 -07:00
|
|
|
|
|
|
|
def encode(self):
|
2018-12-09 08:52:14 -08:00
|
|
|
if self.version == 6:
|
2018-05-01 05:17:55 -07:00
|
|
|
return {'ip6': self.ip_addr}
|
2018-08-08 01:06:40 -07:00
|
|
|
else:
|
2018-05-01 05:17:55 -07:00
|
|
|
return {'ip4': self.ip_addr}
|
2018-08-08 01:06:40 -07:00
|
|
|
|
2018-09-05 15:42:26 -07:00
|
|
|
@property
|
|
|
|
def version(self):
|
|
|
|
return self.ip_addr.version
|
|
|
|
|
|
|
|
@property
|
|
|
|
def address(self):
|
|
|
|
return self.addr
|
|
|
|
|
|
|
|
@property
|
|
|
|
def length(self):
|
2018-12-09 08:52:14 -08:00
|
|
|
return self.ip_addr.max_prefixlen
|
2018-09-05 15:42:26 -07:00
|
|
|
|
|
|
|
@property
|
|
|
|
def bytes(self):
|
|
|
|
return self.ip_addr.packed
|
|
|
|
|
|
|
|
def __eq__(self, other):
|
|
|
|
if isinstance(other, self.__class__):
|
|
|
|
return self.ip_addr == other.ip_addr
|
|
|
|
elif hasattr(other, "ip4") and hasattr(other, "ip6"):
|
|
|
|
# vl_api_address_union_t
|
2018-12-09 08:52:14 -08:00
|
|
|
if 4 == self.version:
|
2018-11-27 10:05:23 +01:00
|
|
|
return self.ip_addr.packed == other.ip4
|
2018-09-05 15:42:26 -07:00
|
|
|
else:
|
2018-11-27 10:05:23 +01:00
|
|
|
return self.ip_addr.packed == other.ip6
|
2018-09-05 15:42:26 -07:00
|
|
|
else:
|
2018-12-09 08:52:14 -08:00
|
|
|
_log.error("Comparing VppIpAddressUnions:%s"
|
|
|
|
" with incomparable type: %s",
|
|
|
|
self, other)
|
|
|
|
return NotImplemented
|
2018-09-05 15:42:26 -07:00
|
|
|
|
2018-05-01 05:17:55 -07:00
|
|
|
def __str__(self):
|
|
|
|
return str(self.ip_addr)
|
|
|
|
|
2018-08-08 01:06:40 -07:00
|
|
|
|
|
|
|
class VppIpAddress():
|
|
|
|
def __init__(self, addr):
|
|
|
|
self.addr = VppIpAddressUnion(addr)
|
|
|
|
|
|
|
|
def encode(self):
|
2018-12-09 08:52:14 -08:00
|
|
|
if self.addr.version == 6:
|
2018-08-08 01:06:40 -07:00
|
|
|
return {
|
2018-10-16 14:42:50 +02:00
|
|
|
'af': VppEnum.vl_api_address_family_t.ADDRESS_IP6,
|
2018-08-08 01:06:40 -07:00
|
|
|
'un': self.addr.encode()
|
|
|
|
}
|
|
|
|
else:
|
|
|
|
return {
|
2018-10-16 14:42:50 +02:00
|
|
|
'af': VppEnum.vl_api_address_family_t.ADDRESS_IP4,
|
2018-08-08 01:06:40 -07:00
|
|
|
'un': self.addr.encode()
|
|
|
|
}
|
|
|
|
|
2018-09-05 15:42:26 -07:00
|
|
|
def __eq__(self, other):
|
|
|
|
if isinstance(other, self.__class__):
|
|
|
|
return self.addr == other.addr
|
|
|
|
elif hasattr(other, "af") and hasattr(other, "un"):
|
|
|
|
# a vp_api_address_t
|
2018-12-09 08:52:14 -08:00
|
|
|
if 4 == self.version:
|
2018-10-16 14:42:50 +02:00
|
|
|
return other.af == \
|
|
|
|
VppEnum.vl_api_address_family_t.ADDRESS_IP4 and \
|
2018-09-05 15:42:26 -07:00
|
|
|
other.un == self.addr
|
|
|
|
else:
|
2018-10-16 14:42:50 +02:00
|
|
|
return other.af == \
|
|
|
|
VppEnum.vl_api_address_family_t.ADDRESS_IP6 and \
|
2018-09-05 15:42:26 -07:00
|
|
|
other.un == self.addr
|
|
|
|
else:
|
2018-12-09 08:52:14 -08:00
|
|
|
_log.error(
|
|
|
|
"Comparing VppIpAddress:<%s> %s with incomparable "
|
|
|
|
"type: <%s> %s",
|
|
|
|
self.__class__.__name__, self,
|
|
|
|
other.__class__.__name__, other)
|
|
|
|
return NotImplemented
|
2018-09-05 15:42:26 -07:00
|
|
|
|
|
|
|
def __ne__(self, other):
|
|
|
|
return not (self == other)
|
|
|
|
|
|
|
|
def __str__(self):
|
|
|
|
return self.address
|
|
|
|
|
|
|
|
@property
|
|
|
|
def bytes(self):
|
|
|
|
return self.addr.bytes
|
|
|
|
|
2018-08-08 01:06:40 -07:00
|
|
|
@property
|
|
|
|
def address(self):
|
|
|
|
return self.addr.address
|
|
|
|
|
2018-09-05 15:42:26 -07:00
|
|
|
@property
|
|
|
|
def length(self):
|
|
|
|
return self.addr.length
|
|
|
|
|
|
|
|
@property
|
|
|
|
def version(self):
|
|
|
|
return self.addr.version
|
|
|
|
|
|
|
|
@property
|
|
|
|
def is_ip6(self):
|
|
|
|
return (self.version == 6)
|
|
|
|
|
|
|
|
@property
|
|
|
|
def af(self):
|
|
|
|
if self.version == 6:
|
|
|
|
return AF_INET6
|
|
|
|
else:
|
|
|
|
return AF_INET
|
|
|
|
|
|
|
|
@property
|
|
|
|
def dpo_proto(self):
|
2018-12-09 08:52:14 -08:00
|
|
|
if self.version == 6:
|
2018-09-05 15:42:26 -07:00
|
|
|
return DpoProto.DPO_PROTO_IP6
|
|
|
|
else:
|
|
|
|
return DpoProto.DPO_PROTO_IP4
|
|
|
|
|
2018-08-08 01:06:40 -07:00
|
|
|
|
|
|
|
class VppIpPrefix():
|
|
|
|
def __init__(self, addr, len):
|
|
|
|
self.addr = VppIpAddress(addr)
|
|
|
|
self.len = len
|
|
|
|
|
2018-05-01 05:17:55 -07:00
|
|
|
def __eq__(self, other):
|
|
|
|
if self.address == other.address and self.len == other.len:
|
|
|
|
return True
|
|
|
|
return False
|
|
|
|
|
2018-08-08 01:06:40 -07:00
|
|
|
def encode(self):
|
|
|
|
return {'address': self.addr.encode(),
|
2019-06-06 14:07:55 -04:00
|
|
|
'len': self.len}
|
2018-08-08 01:06:40 -07:00
|
|
|
|
2018-05-01 05:17:55 -07:00
|
|
|
@property
|
|
|
|
def version(self):
|
|
|
|
return self.addr.version
|
|
|
|
|
2018-08-08 01:06:40 -07:00
|
|
|
@property
|
|
|
|
def address(self):
|
|
|
|
return self.addr.address
|
|
|
|
|
2018-10-10 07:22:51 -07:00
|
|
|
@property
|
|
|
|
def bytes(self):
|
|
|
|
return self.addr.bytes
|
|
|
|
|
2018-09-05 15:42:26 -07:00
|
|
|
@property
|
|
|
|
def length(self):
|
|
|
|
return self.len
|
|
|
|
|
2018-10-10 07:22:51 -07:00
|
|
|
@property
|
|
|
|
def is_ip6(self):
|
|
|
|
return self.addr.is_ip6
|
|
|
|
|
2018-09-05 15:42:26 -07:00
|
|
|
def __str__(self):
|
|
|
|
return "%s/%d" % (self.address, self.length)
|
|
|
|
|
|
|
|
def __eq__(self, other):
|
|
|
|
if isinstance(other, self.__class__):
|
|
|
|
return (self.len == other.len and self.addr == other.addr)
|
2019-06-06 14:07:55 -04:00
|
|
|
elif hasattr(other, "address") and hasattr(other, "len"):
|
2018-09-05 15:42:26 -07:00
|
|
|
# vl_api_prefix_t
|
2019-06-06 14:07:55 -04:00
|
|
|
return self.len == other.len and \
|
2018-12-09 08:52:14 -08:00
|
|
|
self.addr == other.address
|
2018-09-05 15:42:26 -07:00
|
|
|
else:
|
2018-12-09 08:52:14 -08:00
|
|
|
_log.error(
|
|
|
|
"Comparing VppIpPrefix:%s with incomparable type: %s" %
|
|
|
|
(self, other))
|
|
|
|
return NotImplemented
|
2018-09-05 15:42:26 -07:00
|
|
|
|
2018-08-08 01:06:40 -07:00
|
|
|
|
|
|
|
class VppIpMPrefix():
|
2018-05-01 05:17:55 -07:00
|
|
|
def __init__(self, saddr, gaddr, glen):
|
2018-08-08 01:06:40 -07:00
|
|
|
self.saddr = saddr
|
|
|
|
self.gaddr = gaddr
|
2018-05-01 05:17:55 -07:00
|
|
|
self.glen = glen
|
|
|
|
self.ip_saddr = VppIpAddressUnion(text_type(self.saddr))
|
|
|
|
self.ip_gaddr = VppIpAddressUnion(text_type(self.gaddr))
|
2018-12-09 08:52:14 -08:00
|
|
|
if self.ip_saddr.version != self.ip_gaddr.version:
|
|
|
|
raise ValueError('Source and group addresses must be of the '
|
|
|
|
'same address family.')
|
2018-08-08 01:06:40 -07:00
|
|
|
|
|
|
|
def encode(self):
|
2018-12-09 08:52:14 -08:00
|
|
|
if 6 == self.ip_saddr.version:
|
2018-08-08 01:06:40 -07:00
|
|
|
prefix = {
|
2018-10-16 14:42:50 +02:00
|
|
|
'af': VppEnum.vl_api_address_family_t.ADDRESS_IP6,
|
2018-05-01 05:17:55 -07:00
|
|
|
'grp_address': {
|
|
|
|
'ip6': self.gaddr
|
|
|
|
},
|
|
|
|
'src_address': {
|
|
|
|
'ip6': self.saddr
|
|
|
|
},
|
|
|
|
'grp_address_length': self.glen,
|
2018-08-08 01:06:40 -07:00
|
|
|
}
|
|
|
|
else:
|
|
|
|
prefix = {
|
2018-10-16 14:42:50 +02:00
|
|
|
'af': VppEnum.vl_api_address_family_t.ADDRESS_IP4,
|
2018-05-01 05:17:55 -07:00
|
|
|
'grp_address': {
|
|
|
|
'ip4': self.gaddr
|
|
|
|
},
|
|
|
|
'src_address': {
|
|
|
|
'ip4': self.saddr
|
|
|
|
},
|
|
|
|
'grp_address_length': self.glen,
|
2018-08-08 01:06:40 -07:00
|
|
|
}
|
|
|
|
return prefix
|
2018-05-01 05:17:55 -07:00
|
|
|
|
|
|
|
@property
|
|
|
|
def length(self):
|
|
|
|
return self.glen
|
|
|
|
|
|
|
|
@property
|
|
|
|
def version(self):
|
|
|
|
return self.ip_gaddr.version
|
|
|
|
|
|
|
|
def __str__(self):
|
|
|
|
return "(%s,%s)/%d" % (self.saddr, self.gaddr, self.glen)
|
|
|
|
|
|
|
|
def __eq__(self, other):
|
|
|
|
if isinstance(other, self.__class__):
|
|
|
|
return (self.glen == other.glen and
|
|
|
|
self.ip_saddr == other.ip_gaddr and
|
|
|
|
self.ip_saddr == other.ip_saddr)
|
|
|
|
elif (hasattr(other, "grp_address_length") and
|
|
|
|
hasattr(other, "grp_address") and
|
|
|
|
hasattr(other, "src_address")):
|
|
|
|
# vl_api_mprefix_t
|
|
|
|
if 4 == self.ip_saddr.version:
|
|
|
|
if self.glen == other.grp_address_length and \
|
|
|
|
self.gaddr == str(other.grp_address.ip4) and \
|
|
|
|
self.saddr == str(other.src_address.ip4):
|
|
|
|
return True
|
|
|
|
return False
|
|
|
|
else:
|
|
|
|
return (self.glen == other.grp_address_length and
|
|
|
|
self.gaddr == other.grp_address.ip6 and
|
|
|
|
self.saddr == other.src_address.ip6)
|
|
|
|
else:
|
|
|
|
raise Exception("Comparing VppIpPrefix:%s with unknown type: %s" %
|
|
|
|
(self, other))
|
|
|
|
return False
|