1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360136113621363136413651366136713681369137013711372137313741375137613771378137913801381138213831384138513861387138813891390139113921393139413951396139713981399140014011402140314041405140614071408140914101411141214131414141514161417141814191420142114221423142414251426142714281429143014311432143314341435143614371438143914401441144214431444144514461447144814491450145114521453145414551456145714581459146014611462146314641465146614671468146914701471147214731474147514761477147814791480148114821483148414851486148714881489149014911492149314941495 |
- import collections
- from functools import cmp_to_key
- import ipaddress
- import re
- from copy import deepcopy
- mac_prefix = "f2"
- vrf_info = {
- 'vrf_external' : {
- 'table' : 1023,
- 'fwmark' : [ '0x1', '0x1023' ],
- },
- }
- default_bond_config = {
- 'bond-mode': '802.3ad',
- 'bond-min-links': '1',
- 'bond-xmit-hash-policy': 'layer3+4'
- }
- default_bridge_config = {
- 'bridge-fd' : '0',
- 'bridge-stp' : 'no',
- 'bridge-ports-condone-regex' : '^[a-zA-Z0-9]+_(v[0-9]{1,4}|eth[0-9])$',
- }
- default_hop_penalty_by_role = {
- 'bbr' : 5,
- 'bras' : 50,
- 'batman_gw' : 50,
- 'batman_ext': 50,
- }
- batman_role_evaluation_order = [ 'bbr', 'batman_gw', 'bras' ]
- default_batman_iface_penalty_by_role = {
- 'default' : 0,
- 'WBBL' : 5,
- 'WBBL_backup' : 10,
- 'VPN_intergw' : 80,
- 'VPN_node' : 100,
- }
- GRE_FFRL_attrs = {
- 'mode' : 'gre',
- 'method' : 'tunnel',
- 'mtu' : '1400',
- 'ttl' : '64',
- }
- loopback_prefix = {
- 'v4' : '10.132.255.',
- 'v6' : '2a03:2260:2342:ffff::',
- }
- DNS_zone_names = {
- 'forward' : 'ffho.net',
- 'rev_v4' : [
- '132.10.in-addr.arpa',
- '30.172.in-addr.arpa',
- ],
- 'rev_v6' : [
- '2.4.3.2.0.6.2.2.3.0.a.2.ip6.arpa',
- ]
- }
- MTU = {
-
-
-
- 'default' : 1500,
-
-
-
-
- 'batman_underlay_iface' : 1560,
-
-
-
-
-
-
-
- 'vxlan_underlay_iface' : 1610,
- }
- class Prefix (object):
- """An internet address with a prefix length.
- The given address is expected to be of format ip/plen in CIDR notation.
- The IP as well as the prefix length and address family will be stored
- in attributes.
- .. code-block:: pycon
- >>> a = Prefix ('10.132.23.42/24')
- >>> str (a.ip)
- '10.132.23.42'
- >>> str (a.af)
- '4'
- >>> str (a.plen)
- '24'
- >>> str (a.netmask)
- '255.255.255.0'
- >>> str (a.network_address)
- '10.132.23.0'
- """
- def __init__ (self, prefix):
- self.prefix = prefix
- self.ip_network = ipaddress.ip_network (u'%s' % prefix, strict = False)
- def __eq__ (self, other):
- if isinstance (other, Prefix):
- return self.ip_network == other.ip_network
- return NotImplemented
- def __lt__ (self, other):
- if isinstance (other, Prefix):
- return self.ip_network < other.ip_network
- return NotImplemented
- def __str__ (self):
- return self.prefix
- @property
- def ip (self):
- return self.prefix.split ('/')[0]
- @property
- def af (self):
- return self.ip_network.version
- @property
- def plen (self):
- return self.ip_network.prefixlen
- @property
- def netmask (self):
- return self.ip_network.netmask
- @property
- def network_address (self):
- return self.ip_network.network_address
- sites = None
- def _get_site_no (sites_config, site_name):
- global sites
- if sites == None:
- sites = {}
- for site in sites_config:
- if site.startswith ("_"):
- continue
- sites[site] = sites_config[site].get ("site_no", -2)
- return sites.get (site_name, -1)
- def gen_batman_iface_mac (site_no, device_no, network):
- net_type_map = {
- 'bat' : "00:00",
- 'dummy' : "00:0d",
- 'int2ext' : "00:0f",
- 'bat-e' : "00:e0",
- 'intergw' : "00:e1",
- 'nodes4' : "00:e4",
- 'nodes6' : "00:e6",
- 'dummy-e' : "00:ed",
- 'ext2int' : "00:ef",
- }
-
- if network in net_type_map:
- last = net_type_map[network]
- elif type (network) == int:
- last = re.sub (r'(\d{2})(\d{2})', '\g<1>:\g<2>', "%04d" % network)
- else:
- last = "ee:ee"
-
- device_no_hex = re.sub (r'([0-9a-fA-F]{2})([0-9a-fA-F]{2})', '\g<1>:\g<2>', "%04x" % int (device_no))
-
- site_no_hex = "%02d" % int (site_no)
- return "%s:%s:%s:%s" % (mac_prefix, device_no_hex, site_no_hex, last)
- def _update_batman_config (node_config, iface, sites_config):
- try:
- node_batman_hop_penalty = int (node_config['batman']['hop-penalty'])
- except (KeyError,ValueError):
- node_batman_hop_penalty = None
- iface_config = node_config['ifaces'][iface]
- iface_type = iface_config.get ('type', 'inet')
- batman_config = {}
- for item in list (iface_config.keys ()):
- value = iface_config.get (item)
- if item.startswith ('batman-'):
- batman_config[item] = value
- iface_config.pop (item)
-
- if iface_type == 'batman':
- if 'batman-hop-penalty' not in batman_config:
-
-
- if node_batman_hop_penalty:
- batman_config['batman-hop-penalty'] = node_batman_hop_penalty
-
-
- else:
- node_roles = node_config.get ('roles', [])
- for role in batman_role_evaluation_order:
- if role in node_roles:
- batman_config['batman-hop-penalty'] = default_hop_penalty_by_role[role]
- break
- if 'batman_ext' in node_roles and iface.endswith('-ext'):
- batman_config['batman-hop-penalty'] = default_hop_penalty_by_role['batman_ext']
-
-
- if 'batman-ifaces' in batman_config and type (batman_config['batman-ifaces']) == list:
- batman_iface_str = " ".join (sorted (batman_config['batman-ifaces']))
- batman_config['batman-ifaces'] = batman_iface_str
-
- elif iface_type == 'batman_iface':
-
-
-
- site = iface_config.get ('site')
- site_no = _get_site_no (sites_config, site)
- device_no = node_config.get ('id')
- network = 1234
-
- match = re.search (r'^vlan(\d+)', iface)
- if match:
- network = int (match.group (1))
- iface_config['hwaddress'] = gen_batman_iface_mac (site_no, device_no, network)
- iface_config['batman'] = batman_config
- def _update_bond_config (config):
- bond_config = default_bond_config.copy ()
- to_pop = []
- for item, value in config.items ():
- if item.startswith ('bond-'):
- bond_config[item] = value
- to_pop.append (item)
- for item in to_pop:
- config.pop (item)
- if bond_config['bond-mode'] not in ['2', 'balance-xor', '4', '802.3ad']:
- bond_config.pop ('bond-xmit-hash-policy')
- config['bond'] = bond_config
- def _update_bridge_config (config):
- bridge_config = default_bridge_config.copy ()
- for item in list (config.keys ()):
- value = config.get (item)
- if not item.startswith ('bridge-'):
- continue
- bridge_config[item] = value
- config.pop (item)
-
- if type (value) == bool:
- bridge_config[item] = "yes" if value else "no"
-
-
- if 'bridge-ports' in bridge_config and type (bridge_config['bridge-ports']) == list:
- bridge_ports_str = " ".join (sorted (bridge_config['bridge-ports']))
- if not bridge_ports_str:
- bridge_ports_str = "none"
- bridge_config['bridge-ports'] = bridge_ports_str
- if config.get ('vlan-mode') == 'tagged':
- bridge_config['bridge-vlan-aware'] = 'yes'
- if config.get ('tagged_vlans'):
- bridge_config['bridge-vids'] = " ".join (map (str, config['tagged_vlans']))
- config['bridge'] = bridge_config
- def _update_bridge_member_config (config):
- bridge_config = {}
- if config.get ('tagged_vlans'):
- bridge_config['bridge-vids'] = " ".join (map (str, config['tagged_vlans']))
- config['bridge'] = bridge_config
- def _update_vlan_config (config):
- vlan_config = {}
- for item in list (config.keys ()):
- value = config.get (item)
- if item.startswith ('vlan-'):
- vlan_config[item] = value
- config.pop (item)
- config['vlan'] = vlan_config
- def _update_veth_config (interface, config):
- veth_peer_name = {
- 'veth_ext2int' : 'veth_int2ext',
- 'veth_int2ext' : 'veth_ext2int'
- }
- if interface not in veth_peer_name:
- return
- if 'link-type' not in config:
- config['link-type'] = 'veth'
- if 'veth-peer-name' not in config:
- config['veth-peer-name'] = veth_peer_name[interface]
- def _set_mtu_to_iface_and_upper (ifaces, iface_name, mtu):
- iface_config = ifaces.get (iface_name)
-
-
-
- set_automtu = True
-
-
- if mtu <= 0:
- set_automtu = False
- mtu = iface_config.get ('mtu', MTU['default'])
-
-
-
- if 'mtu' in iface_config:
- set_automtu = False
-
-
-
-
-
-
-
- elif 'automtu' in iface_config and iface_config['automtu'] >= mtu:
- set_automtu = False
-
- if set_automtu:
- iface_config['automtu'] = mtu
-
-
- vlan_raw_device = None
- if 'vlan-raw-device' in iface_config:
- vlan_raw_device = iface_config['vlan-raw-device']
- elif 'vlan' in iface_config and 'vlan-raw-device' in iface_config['vlan']:
- vlan_raw_device = iface_config['vlan']['vlan-raw-device']
- if vlan_raw_device:
- vlan_raw_device_config = ifaces.get (vlan_raw_device, None)
-
-
-
- if vlan_raw_device_config == None:
- vlan_raw_device_config = {}
- ifaces[vlan_raw_device] = vlan_raw_device_config
-
- if 'mtu' in vlan_raw_device_config:
- return
- if 'automtu' in vlan_raw_device_config and vlan_raw_device_config['automtu'] >= mtu:
- return
- vlan_raw_device_config['automtu'] = mtu
- def _generate_batman_interface_config (node_config, ifaces, sites_config):
-
- roles = node_config.get ('roles', [])
- if 'batman' not in roles:
- return
-
- batman_ext = 'batman_ext' in roles or 'bras' in roles
- device_no = node_config.get ('id', -1)
- for site in node_config.get ('sites', []):
- site_no = _get_site_no (sites_config, site)
-
-
- bat_site_if = "bat-%s" % site
- dummy_site_if = "dummy-%s" % site
- bat_site_if_ext = "bat-%s-ext" % site
- dummy_site_if_ext = "dummy-%s-e" % site
- int2ext_site_if = "i2e-%s" % site
- ext2int_site_if = "e2i-%s" % site
- site_ifaces = {
-
- bat_site_if : {
- 'type' : 'batman',
-
- 'batman-ifaces' : [ dummy_site_if ],
- 'batman-ifaces-ignore-regex': '.*_.*',
- 'hwaddress' : gen_batman_iface_mac (site_no, device_no, 'bat'),
- },
-
- dummy_site_if : {
- 'link-type' : 'dummy',
- 'hwaddress' : gen_batman_iface_mac (site_no, device_no, 'dummy'),
- 'mtu' : MTU['batman_underlay_iface'],
- },
-
- bat_site_if_ext : {
- 'type' : 'batman',
- 'batman-ifaces' : [ dummy_site_if_ext, ext2int_site_if ],
- 'batman-ifaces-ignore-regex': '.*_.*',
- 'hwaddress' : gen_batman_iface_mac (site_no, device_no, 'bat-e'),
- 'ext_only' : True,
- },
-
- dummy_site_if_ext : {
- 'link-type' : 'dummy',
- 'hwaddress' : gen_batman_iface_mac (site_no, device_no, 'dummy-e'),
- 'ext_only' : True,
- 'mtu' : MTU['batman_underlay_iface'],
- },
-
- int2ext_site_if : {
- 'link-type' : 'veth',
- 'veth-peer-name' : ext2int_site_if,
- 'hwaddress' : gen_batman_iface_mac (site_no, device_no, 'int2ext'),
- 'mtu' : MTU['batman_underlay_iface'],
- 'ext_only' : True,
- },
-
- ext2int_site_if : {
- 'link-type' : 'veth',
- 'veth-peer-name' : int2ext_site_if,
- 'hwaddress' : gen_batman_iface_mac (site_no, device_no, 'ext2int'),
- 'mtu' : MTU['batman_underlay_iface'],
- 'ext_only' : True,
- },
- }
- for iface, iface_config_tmpl in site_ifaces.items ():
-
-
- if not batman_ext and iface_config_tmpl.get ('ext_only', False):
- continue
-
- if 'ext_only' in iface_config_tmpl:
- del iface_config_tmpl['ext_only']
-
- if iface not in ifaces:
-
- ifaces[iface] = iface_config_tmpl
-
-
-
- if batman_ext and iface == bat_site_if:
- iface_config_tmpl['batman-ifaces'].append (int2ext_site_if)
-
-
-
- else:
- iface_config = ifaces[iface]
-
- if 'hwaddress' in iface_config_tmpl:
- iface_config['hwaddress'] = iface_config_tmpl['hwaddress']
-
- for attr in iface_config_tmpl:
- if attr not in iface_config:
- iface_config[attr] = iface_config_tmpl[attr]
-
- site_bridge = "br-%s" % site
- if site_bridge in ifaces:
- bridge_config = ifaces.get (site_bridge)
- bridge_ports = bridge_config.get ('bridge-ports', None)
-
- if bridge_ports and bat_site_if not in bridge_ports:
- if type (bridge_ports) == list:
- bridge_ports.append (bat_site_if)
- else:
- bridge_config['bridge-ports'] += ' ' + bat_site_if
-
- if not bridge_ports:
- bridge_config['bridge-ports'] = bat_site_if
-
-
- for iface in list (ifaces.keys ()):
- config = ifaces.get (iface)
- iface_type = config.get ('type', 'inet')
- if iface_type not in ['mesh_breakout', 'batman_iface']:
- continue
- site = config.get ('site')
- site_bridge = "br-%s" % site
- batman_site_if = "bat-%s" % site
- if iface_type == 'mesh_breakout':
-
-
-
- if site_bridge in ifaces:
- bridge_config = ifaces.get (site_bridge)
-
-
- bridge_ports = bridge_config.get ('bridge-ports', None)
- if bridge_ports:
- for dev in (batman_site_if, iface):
- if not dev in bridge_ports:
- if type (bridge_ports) == list:
- bridge_ports.append (dev)
- else:
- bridge_config['bridge-ports'] += ' ' + dev
-
-
- else:
- bridge_config['bridge-ports'] = [ iface, batman_site_if ]
-
- else:
- ifaces[site_bridge] = {
- 'bridge-ports' : [ iface, batman_site_if ],
- }
- elif iface_type == 'batman_iface':
- batman_ifaces = ifaces[batman_site_if]['batman-ifaces']
- if iface not in batman_ifaces:
- if type (batman_ifaces) == list:
- batman_ifaces.append (iface)
- else:
- batman_ifaces += ' ' + iface
-
-
- mtu = config.get('mtu', MTU['batman_underlay_iface'])
- _set_mtu_to_iface_and_upper (ifaces, iface, mtu)
- def _generate_vxlan_interface_config (node_config, ifaces, sites_config):
-
- if 'batman' not in node_config.get ('roles', []):
- return
-
- my_sites = node_config.get ('sites', [])
- if len (my_sites) == 0:
- return
-
-
- device_no = node_config.get ('id', -1)
- for iface in list (ifaces.keys ()):
- iface_config = ifaces.get (iface)
- batman_connect_sites = iface_config.get ('batman_connect_sites', [])
- iface_has_prefixes = len (iface_config.get ('prefixes', {})) != 0
-
- if type (batman_connect_sites) == str:
- batman_connect_sites = [ batman_connect_sites ]
-
- if len (batman_connect_sites) == 0:
- continue
-
-
- _set_mtu_to_iface_and_upper (ifaces, iface, MTU['vxlan_underlay_iface'])
-
- if 'all' in batman_connect_sites:
- batman_connect_sites = my_sites
- for site in batman_connect_sites:
-
- if site not in my_sites:
- continue
-
- vx_iface = ("vx_%s_%s" % (re.sub ('vlan', 'v', iface)[-5:], re.sub (r'[_-]', '', site)))[:15]
- site_no = _get_site_no (sites_config, site)
- bat_iface = "bat-%s" % site
-
-
- if bat_iface not in ifaces:
- continue
-
- if vx_iface in ifaces:
- continue
- try:
- iface_id = int (re.sub ('vlan', '', iface))
-
-
-
-
- iface_id_4digit = "%04d" % iface_id
- octet2 = int (iface_id_4digit[0:2])
- octet3 = int (iface_id_4digit[2:4])
- vni = octet2 * 256 * 256 + octet3 * 256 + site_no
- vtep_config = {
- 'vxlan-id' : vni,
- 'vxlan-physdev' : iface,
- }
-
- if iface_has_prefixes:
- vtep_config['vxlan-svcnodeip'] = "225.%s.%s.%s" % (octet2, octet3, site_no)
- else:
- vtep_config['vxlan-remote-group'] = "ff42:%s::%s" % (iface_id, site_no)
- except ValueError as v:
- vtep_config = {
- 'vxlan-config-error' : str (v),
- }
- iface_id = 9999
- mcast_ip = "225.0.0.%s" % site_no
- vni = site_no
-
- ifaces[vx_iface] = {
- 'vxlan' : vtep_config,
- 'hwaddress' : gen_batman_iface_mac (site_no, device_no, iface_id),
- 'mtu' : MTU['batman_underlay_iface'],
- }
- iface_penalty = get_batman_iface_penalty (iface)
- if iface_penalty:
- ifaces[vx_iface]['batman'] = {
- 'batman-hop-penalty' : iface_penalty
- }
-
-
-
- if not 'batman-ifaces' in ifaces[bat_iface]:
- ifaces[bat_iface]['batman-ifaces'] = [ vx_iface ]
- continue
-
-
- batman_ifaces = ifaces[bat_iface]['batman-ifaces']
- if vx_iface not in batman_ifaces:
- if type (batman_ifaces) == list:
- batman_ifaces.append (vx_iface)
- else:
- batman_ifaces += ' ' + vx_iface
- def _generate_vrfs (ifaces):
- for iface in list (ifaces.keys ()):
- iface_config = ifaces.get (iface)
- vrf = iface_config.get ('vrf', None)
- if vrf and vrf not in ifaces:
- conf = vrf_info.get (vrf, {})
- table = conf.get ('table', 1234)
- fwmark = conf.get ('fwmark', None)
- ifaces[vrf] = {
- 'vrf-table' : table,
- }
-
- if fwmark:
- up = []
-
- if type (fwmark) in (str, int):
- fwmark = [ fwmark ]
-
- for mark in fwmark:
- up.append ("ip rule add fwmark %s table %s" % (mark, table))
- up.append ("ip -6 rule add fwmark %s table %s" % (mark, table))
- ifaces[vrf]['up'] = up
- def _generate_ffrl_gre_tunnels (ifaces):
- for iface, iface_config in ifaces.items ():
-
- if iface_config.get ('type', '') != 'GRE_FFRL':
- continue
-
- for attr, val in GRE_FFRL_attrs.items ():
- if not attr in iface_config:
- iface_config[attr] = val
-
- if not 'local' in iface_config and 'tunnel-physdev' in iface_config:
- try:
- physdev_prefixes = [p.split ('/')[0] for p in ifaces[iface_config['tunnel-physdev']]['prefixes'] if '.' in p]
- if len (physdev_prefixes) == 1:
- iface_config['local'] = physdev_prefixes[0]
- except KeyError:
- pass
- def _generate_loopback_ips (ifaces, node_config, node_id):
-
-
-
- if node_config.get ('primary_ips', False):
- return
- v4_ip = "%s/32" % get_primary_ip (node_config, 'v4').ip
- v6_ip = "%s/128" % get_primary_ip (node_config, 'v6').ip
-
- if 'lo' not in ifaces:
- ifaces['lo'] = { 'prefixes' : [] }
-
- if 'prefixes' not in ifaces['lo']:
- ifaces['lo']['prefixes'] = []
- prefixes = ifaces['lo']['prefixes']
- if v4_ip not in prefixes:
- prefixes.append (v4_ip)
- if v6_ip not in prefixes:
- prefixes.append (v6_ip)
- def _update_interface_desc (node_config, sites_config):
-
- if 'batman' not in node_config.get ('roles', []):
- return
- for iface, iface_config in node_config.get ('ifaces', {}).items ():
- if 'desc' in sites_config:
- continue
-
-
- match = re.search (r'^br-([a-z_-]+)$', iface)
- if match and match.group (1) in sites_config:
- try:
- iface_config['desc'] = sites_config[match.group (1)]['name']
- except KeyError:
- pass
- def get_interface_config (node_config, sites_config, node_id = ""):
-
-
-
-
- node_config = deepcopy (node_config)
-
- ifaces = node_config.get ('ifaces', {})
-
-
-
- _generate_batman_interface_config (node_config, ifaces, sites_config)
-
- _generate_vxlan_interface_config (node_config, ifaces, sites_config)
-
-
- for interface in list (ifaces.keys ()):
- config = ifaces.get (interface)
- iface_type = config.get ('type', 'inet')
-
- if config.get ('enabled', True) == False:
- del ifaces[interface]
- continue
- if 'batman-ifaces' in config or iface_type.startswith ('batman'):
- _update_batman_config (node_config, interface, sites_config)
- if 'bond-slaves' in config:
- _update_bond_config (config)
-
- if 'bridge-ports' in config or interface.startswith ('br-'):
- _update_bridge_config (config)
- if 'bridge-member' in config:
- _update_bridge_member_config (config)
- if 'vlan-raw-device' in config or 'vlan-id' in config:
- _update_vlan_config (config)
- _set_mtu_to_iface_and_upper (ifaces, interface, 0)
-
- if interface.startswith ('veth_'):
- _update_veth_config (interface, config)
-
- _generate_loopback_ips (ifaces, node_config, node_id)
-
- _generate_vrfs (ifaces)
-
- _generate_ffrl_gre_tunnels (ifaces)
-
-
- for interface, config in ifaces.items ():
-
- if interface != 'lo' and ('mtu' not in config):
-
-
- config['mtu'] = config.get ('automtu', MTU['default'])
- for key in [ 'automtu', 'enabled', 'batman_connect_sites', 'bridge-member', 'has_gateway', 'ospf', 'site', 'type', 'tagged_vlans', 'vlan-mode' ]:
- if key in config:
- config.pop (key)
-
- if 'metric' in config and not 'router' in node_config.get ('roles', []):
- config.pop ('metric')
-
-
-
-
- _update_interface_desc (node_config, sites_config)
- return ifaces
- vlan_vxlan_iface_re = re.compile (r'^vlan(\d+)|^vx_v(\d+)_(\w+)')
- def _cmp (x, y):
- if x < y:
- return -1
- elif x == y:
- return 0
- else:
- return 1
- def _iface_sort (iface_a, iface_b):
- a = vlan_vxlan_iface_re.search (iface_a)
- b = vlan_vxlan_iface_re.search (iface_b)
-
- if not a or not b:
- return _cmp (iface_a, iface_b)
-
- vid_a = a.group (1) if a.group (1) else a.group (2)
- vid_b = b.group (1) if b.group (1) else b.group (2)
-
- if (a.group (1) == None) != (b.group (1) == None):
- return _cmp (iface_a, iface_b)
-
-
- if a.group (2) and vid_a == vid_b:
- return _cmp (a.groups (2), b.groups (2))
-
- else:
- return _cmp (int (vid_a), int (vid_b))
- def get_interface_list (ifaces):
- iface_list = []
- for iface in sorted (ifaces.keys (), key = cmp_to_key (_iface_sort)):
- iface_list.append (iface)
- return iface_list
- def gen_bat_hosts (nodes_config, sites_config):
- bat_hosts = {}
- for node_id in sorted (nodes_config.keys ()):
- node_config = nodes_config.get (node_id)
- node_name = node_id.split ('.')[0]
- if 'batman' not in node_config['roles']:
- continue
- ifaces = get_interface_config (node_config, sites_config, node_id)
- for iface in sorted (ifaces):
- iface_config = ifaces.get (iface)
- hwaddress = iface_config.get ('hwaddress', None)
- if hwaddress == None:
- continue
- entry_name = node_name
- match = re.search (r'^dummy-(.+)(-e)?$', iface)
- if match:
- if match.group (2):
- entry_name += "-e"
-
- entry_name += "/%s" % match.group (1)
- else:
- entry_name += "/%s" % re.sub (r'^(vx_.*|i2e|e2i)[_-](.*)$', '\g<1>/\g<2>', iface)
- bat_hosts[hwaddress] = entry_name
- if 'fastd' in node_config.get ('roles', []):
- device_no = node_config.get ('id')
- for site in node_config.get ('sites', []):
- site_no = _get_site_no (sites_config, site)
- for network in ('intergw', 'nodes4', 'nodes6'):
- hwaddress = gen_batman_iface_mac (site_no, device_no, network)
- bat_hosts[hwaddress] = "%s/%s/%s" % (node_name, network, site)
- return bat_hosts
- def get_batman_iface_penalty (iface):
- if iface.startswith ('vlan'):
- vid = int (re.sub ('vlan', '', iface))
- if 2000 <= vid < 2100:
- return default_batman_iface_penalty_by_role.get ('WBBL')
- if 2200 <= vid < 2300:
- return default_batman_iface_penalty_by_role.get ('WBBL_backup')
- if 'intergw' in iface:
- return default_batman_iface_penalty_by_role.get ('VPN_intergw')
- if 'nodes' in iface:
- return default_batman_iface_penalty_by_role.get ('VPN_node')
- return default_batman_iface_penalty_by_role.get ('default', 0)
- def get_ffrl_bgp_config (ifaces, proto):
- _generate_ffrl_gre_tunnels (ifaces)
- sessions = {}
- for iface in sorted (ifaces):
-
- if not iface.startswith ('gre_ffrl_'):
- continue
- iface_config = ifaces.get (iface)
-
- local = None
- neighbor = None
- for prefix in iface_config.get ('prefixes', []):
- if (proto == 'v4' and '.' in prefix) or (proto == 'v6' and ':' in prefix):
- local = prefix.split ('/')[0]
-
- neighbor = str (ipaddress.ip_address (u'%s' % local) - 1)
- break
-
- name = re.sub ('gre_ffrl_', 'ffrl_', iface)
- sessions[name] = {
- 'local' : local,
- 'neighbor' : neighbor,
- 'bgp_local_pref' : iface_config.get ('bgp_local_pref', None),
- }
- return sessions
- def get_node_iface_ips (node_config, iface_name, with_mask = False):
- ips = {
- 'v4' : [],
- 'v6' : [],
- }
- ifaces = node_config.get ('ifaces', {})
- ifaces_names = [ iface_name ]
- if iface_name.startswith ('vrf_'):
-
- ifaces_names = []
- vrf = iface_name
- for iface, iface_config in ifaces.items ():
-
- if iface_config.get ('vrf', None) != vrf:
- continue
-
- if iface.startswith ('veth'):
- continue
- ifaces_names.append (iface)
- try:
- for iface in ifaces_names:
- for prefix in ifaces[iface]['prefixes']:
- ip_ver = 'v6' if ':' in prefix else 'v4'
- if not with_mask:
- prefix = prefix.split ('/')[0]
- ips[ip_ver].append (prefix)
- except KeyError:
- pass
- return ips
- def get_primary_ip (node_config, af):
-
- if 'primary_ips' not in node_config:
- return Prefix ("%s%s" % (loopback_prefix[af], node_config['id']))
- return Prefix (node_config['primary_ips'][af])
- def get_router_id (node_config, node_id):
- return get_primary_ip (node_config, 'v4').ip
- def get_ospf_interface_config (node_config, grains_id):
- ospf_node_config = node_config.get ('ospf', {})
- ospf_interfaces = {}
- for iface, iface_config in node_config.get ('ifaces', {}).items ():
-
- ospf_on = False
-
- ospf_config = {
- 'stub' : True,
- 'cost' : 12345,
-
- }
-
- ospf_config_pillar = iface_config.get ('ospf', {})
-
- if ospf_config_pillar.get ('ignore', False):
- continue
-
- if not iface_config.get ('prefixes'):
- continue
-
- if re.search (r'^vlan90\d$', iface):
- ospf_on = True
- ospf_config['stub'] = True
- ospf_config['cost'] = 10
- ospf_config['desc'] = "Wireless Local Link (WLL)"
-
- elif re.search (r'^(br-?|br\d+\.|vlan)10\d\d$', iface):
- ospf_on = True
- ospf_config['stub'] = False
- ospf_config['cost'] = 100
- ospf_config['desc'] = "Wired Gigabit connection"
-
- elif re.search (r'^vlan12\d\d$', iface):
- ospf_on = True
- ospf_config['stub'] = False
- ospf_config['cost'] = 10
- ospf_config['desc'] = "Wired 10Gb/s connection"
-
- elif re.search (r'^vlan15\d\d$', iface):
- ospf_on = True
- ospf_config['stub'] = False
- ospf_config['cost'] = 200
- ospf_config['desc'] = "VLL connection"
-
- elif re.search (r'^vlan20\d\d$', iface):
- ospf_on = True
- ospf_config['stub'] = False
- ospf_config['cost'] = 1000
- ospf_config['desc'] = "WBBL connection"
-
- elif re.search (r'^vlan22\d\d$', iface):
- ospf_on = True
- ospf_config['stub'] = False
- ospf_config['cost'] = 1000
- ospf_config['desc'] = "WBBL connection"
-
- elif re.search (r'^vlan30\d\d$', iface):
- ospf_on = True
- ospf_config['stub'] = True
- ospf_config['cost'] = 10
-
- elif re.search (r'^vlan39\d\d$', iface):
- ospf_on = True
- ospf_config['stub'] = True
- ospf_config['cost'] = 10
-
- elif iface.startswith ('ovpn-'):
- ospf_on = True
- ospf_config['stub'] = False
- ospf_config['cost'] = 10000
-
- if iface.startswith ('ovpn-cr') and grains_id.startswith ('cr'):
- ospf_config['cost'] = 5000
-
- elif iface.startswith ('ovpn-er-'):
- ospf_config['type'] = 'broadcast'
-
- elif iface.startswith ('wg-'):
- ospf_on = True
- ospf_config['stub'] = False
- ospf_config['cost'] = 10000
-
- if iface.startswith ('wg-cr') and grains_id.startswith ('cr'):
- ospf_config['cost'] = 5000
-
- elif 'ospf' in iface_config:
- ospf_on = True
-
-
- if not ospf_on:
- continue
-
- for attr, val in ospf_config_pillar.items ():
- ospf_config[attr] = val
-
- ospf_interfaces[iface] = ospf_config
- return ospf_interfaces
- def get_te_prefixes (te_node_config, grains_id, proto):
- te_config = {}
- for prefix, prefix_config in te_node_config.get ('prefixes', {}).items ():
- prefix_proto = 'v6' if ':' in prefix else 'v4'
-
-
- if grains_id in prefix_config.get ('nodes', []) and prefix_proto == proto:
- te_config[prefix] = prefix_config
- return te_config
- def generate_DNS_entries (nodes_config, sites_config):
- forward_zone_name = ""
- forward_zone = []
- zones = {
-
-
-
-
- }
-
-
-
- for entry, value in DNS_zone_names.items ():
- if entry == "forward":
- zone = value
- if not zone.startswith ('.'):
- zone = ".%s" % zone
- zones[zone] = forward_zone
- forward_zone_name = zone
- if entry in [ 'rev_v4', 'rev_v6' ]:
- for zone in value:
- if not zone.startswith ('.'):
- zone = ".%s" % zone
- zones[zone] = []
-
-
-
- for node_id in sorted (nodes_config):
- node_config = nodes_config.get (node_id)
- ifaces = get_interface_config (node_config, sites_config, node_id)
- for iface in sorted (ifaces):
- iface_config = ifaces.get (iface)
-
- prefixes = iface_config.get ("prefixes", None)
- if prefixes == None:
- continue
-
- if iface_config.get ('vrf', "") in [ 'vrf_external' ]:
- continue
- for prefix in sorted (prefixes):
- ip = ipaddress.ip_address (u'%s' % prefix.split ('/')[0])
- proto = 'v%s' % ip.version
-
-
-
-
- entry_name = node_id
- if iface != "lo":
- entry_name = "%s.%s" % (iface, node_id)
- elif iface == 'srv' or re.search (r'^(10.132.251|2a03:2260:2342:f251:)', prefix):
- entry_name = re.sub (r'^([^.]+)\.(.+)$', r'\g<1>.srv.\g<2>', entry_name)
-
-
- forward_entry_name = re.sub (forward_zone_name, "", entry_name)
- forward_entry_name = re.sub (forward_zone_name, "", entry_name)
- forward_entry_typ = "A" if ip.version == 4 else "AAAA"
- forward_zone.append ("%s IN %s %s" % (forward_entry_name, forward_entry_typ, ip))
-
-
-
-
- for zone in zones:
- if ip.reverse_pointer.find (zone) > 0:
- PTR_entry = re.sub (zone, "", ip.reverse_pointer)
- zones[zone].append ("%s IN PTR %s." % (PTR_entry, entry_name))
- break
- return zones
- def cidr_to_dotted_mask (prefix):
- return str (ipaddress.ip_network (prefix, strict = False).netmask)
- def is_subprefix (prefix, subprefix):
- p = ipaddress.ip_network (prefix, strict = False)
- s = ipaddress.ip_network (subprefix, strict = False)
- return s.subnet_of (p)
- def get_network_address (prefix, with_prefixlen = False):
- net_h = ipaddress.ip_network (u'%s' % prefix, strict = False)
- network = str (net_h.network_address)
- if with_prefixlen:
- network += "/%s" % net_h.prefixlen
- return network
|