kuryr/kuryr/lib/binding/drivers/utils.py

117 lines
3.7 KiB
Python

# 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
import pyroute2
from pyroute2.netlink.rtnl import ifinfmsg
from kuryr.lib import constants
_IPDB_CACHE = None
FIXED_IP_KEY = 'fixed_ips'
IP_ADDRESS_KEY = 'ip_address'
MAC_ADDRESS_KEY = 'mac_address'
SUBNET_ID_KEY = 'subnet_id'
def get_veth_pair_names(port_id):
ifname = constants.VETH_PREFIX + port_id
ifname = ifname[:constants.NIC_NAME_LEN]
peer_name = constants.CONTAINER_VETH_PREFIX + port_id
peer_name = peer_name[:constants.NIC_NAME_LEN]
return ifname, peer_name
def get_ipdb():
"""Returns the already cached or a newly created IPDB instance.
IPDB reads the Linux specific file when it's instantiated. This behaviour
prevents Mac OSX users from running unit tests. This function makes the
loading IPDB lazyily and therefore it can be mocked after the import of
modules that import this module.
:returns: The already cached or newly created ``pyroute2.IPDB`` instance
"""
global _IPDB_CACHE
if not _IPDB_CACHE:
_IPDB_CACHE = pyroute2.IPDB()
return _IPDB_CACHE
def get_mtu_from_network(network=None):
"""Get Maximum Transfer Unit from neutron network.
:parm network: neutron network
:returns: mtu on the neutron network
"""
if network is None:
mtu = constants.DEFAULT_NETWORK_MTU
else:
mtu = network.get('mtu', constants.DEFAULT_NETWORK_MTU)
return mtu
def remove_device(ifname):
"""Removes the device with name ifname.
:param ifname: the name of the device to remove
:returns: the index the device identified by ifname had if it
exists, otherwise None
:raises: pyroute2.NetlinkError
"""
ip = get_ipdb()
dev_index = ip.interfaces.get(ifname, {}).get('index', None)
if dev_index:
with ip.interfaces[ifname] as iface:
iface.remove()
return dev_index
def is_up(interface):
flags = interface['flags']
if not flags:
return False
return (flags & ifinfmsg.IFF_UP) == 1
def _configure_container_iface(iface, subnets, fixed_ips, mtu=None,
hwaddr=None):
"""Configures the interface that is placed in the container net ns
:param iface: the pyroute IPDB interface object to configure
:param subnets: an iterable of all the Neutron subnets which the
endpoint is trying to join
:param fixed_ips: an iterable of fixed IPs to be set for the iface
:param mtu: Maximum Transfer Unit to set for the iface
:param hwaddr: Hardware address to set for the iface
"""
subnets_dict = {subnet['id']: subnet for subnet in subnets}
# We assume containers always work with fixed ips, dhcp does not really
# make a lot of sense
for fixed_ip in fixed_ips:
if IP_ADDRESS_KEY in fixed_ip and (SUBNET_ID_KEY in fixed_ip):
subnet_id = fixed_ip[SUBNET_ID_KEY]
subnet = subnets_dict[subnet_id]
cidr = ipaddress.ip_network(str(subnet['cidr']))
iface.add_ip(fixed_ip[IP_ADDRESS_KEY], cidr.prefixlen)
if mtu is not None:
iface.set_mtu(mtu)
if hwaddr is not None:
iface.set_address(hwaddr)
if not is_up(iface):
iface.up()