Add a reasonably tolerant .pylintrc and fix most pylint errors and warnings. ------------------------------------------------------------------ Your code has been rated at 9.78/10
115 lines
4.3 KiB
Python
115 lines
4.3 KiB
Python
#
|
|
# Copyright (c) 2022 Pim van Pelt
|
|
#
|
|
# Licensed under the Apache License, Version 2.0 (the "License");
|
|
# you may not use this file except in compliance with the License.
|
|
# You may obtain a copy of the License at:
|
|
# http://www.apache.org/licenses/LICENSE-2.0
|
|
# Unless required by applicable law or agreed to in writing, software
|
|
# distributed under the License is distributed on an "AS IS" BASIS,
|
|
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
# See the License for the specific language governing permissions and
|
|
# limitations under the License.
|
|
#
|
|
import ipaddress
|
|
|
|
|
|
def get_all_addresses_except_ifname(yaml, except_ifname):
|
|
"""Return a list of all ipaddress.ip_interface() instances in the entire config,
|
|
except for those that belong to 'ifname'.
|
|
"""
|
|
ret = []
|
|
if "interfaces" in yaml:
|
|
for ifname, iface in yaml["interfaces"].items():
|
|
if ifname == except_ifname:
|
|
continue
|
|
|
|
if "addresses" in iface:
|
|
for addr in iface["addresses"]:
|
|
ret.append(ipaddress.ip_interface(addr))
|
|
if "sub-interfaces" in iface:
|
|
for subid, sub_iface in iface["sub-interfaces"].items():
|
|
sub_ifname = f"{ifname}.{int(subid)}"
|
|
if sub_ifname == except_ifname:
|
|
continue
|
|
|
|
if "addresses" in sub_iface:
|
|
for addr in sub_iface["addresses"]:
|
|
ret.append(ipaddress.ip_interface(addr))
|
|
if "loopbacks" in yaml:
|
|
for ifname, iface in yaml["loopbacks"].items():
|
|
if ifname == except_ifname:
|
|
continue
|
|
|
|
if "addresses" in iface:
|
|
for addr in iface["addresses"]:
|
|
ret.append(ipaddress.ip_interface(addr))
|
|
if "bridgedomains" in yaml:
|
|
for ifname, iface in yaml["bridgedomains"].items():
|
|
if ifname == except_ifname:
|
|
continue
|
|
|
|
if "addresses" in iface:
|
|
for addr in iface["addresses"]:
|
|
ret.append(ipaddress.ip_interface(addr))
|
|
|
|
return ret
|
|
|
|
|
|
def is_allowed(yaml, ifname, iface_addresses, ip_interface):
|
|
"""Returns True if there is at most one occurence of the ip_interface (an IPv4/IPv6 prefix+len)
|
|
in the entire config. That said, we need the 'iface_addresses' because VPP is a bit fickle in
|
|
this regard.
|
|
|
|
IP addresses from the same prefix/len can be added to a given interface (ie 192.0.2.1/24 and
|
|
192.0.2.2/24), but other than that, any prefix can not occur as a more-specific or less-specific
|
|
of any other interface.
|
|
|
|
So, we will allow:
|
|
- any ip_interface that is of equal network/len of existing one(s) _on the same interface_
|
|
|
|
And, we will reject
|
|
- any ip_interface that is a more specific of any existing one
|
|
- any ip_interface that is a less specific of any existing one
|
|
|
|
Examples:
|
|
vpp# set interface ip address loop0 192.0.2.1/24
|
|
vpp# set interface ip address loop0 192.0.2.2/24
|
|
vpp# set interface ip address loop0 192.0.2.1/29
|
|
set interface ip address: failed to add 192.0.2.1/29 on loop0 which conflicts with 192.0.2.1/24 for interface loop0
|
|
vpp# set interface ip address loop0 192.0.2.3/23
|
|
set interface ip address: failed to add 192.0.2.3/23 on loop0 which conflicts with 192.0.2.1/24 for interface loop0
|
|
"""
|
|
all_other_addresses = get_all_addresses_except_ifname(yaml, ifname)
|
|
|
|
my_ip_network = ipaddress.ip_network(ip_interface, strict=False)
|
|
|
|
for ipi in all_other_addresses:
|
|
if ipi.version != my_ip_network.version:
|
|
continue
|
|
|
|
if ipaddress.ip_network(ipi, strict=False) == my_ip_network:
|
|
return False
|
|
|
|
if ipaddress.ip_network(ipi, strict=False).subnet_of(my_ip_network):
|
|
return False
|
|
|
|
if my_ip_network.subnet_of(ipaddress.ip_network(ipi, strict=False)):
|
|
return False
|
|
|
|
for addr in iface_addresses:
|
|
ipi = ipaddress.ip_interface(addr)
|
|
if ipi.version != my_ip_network.version:
|
|
continue
|
|
|
|
if ipaddress.ip_network(ipi, strict=False) == my_ip_network:
|
|
return True
|
|
|
|
if ipaddress.ip_network(ipi, strict=False).subnet_of(my_ip_network):
|
|
return False
|
|
|
|
if my_ip_network.subnet_of(ipaddress.ip_network(ipi, strict=False)):
|
|
return False
|
|
|
|
return True
|