blob: 1ad992c148fc1837c89c6b0a239c463e2ff66a96 [file] [log] [blame]
#!/usr/bin/env python3
# SPDX-FileCopyrightText: © 2021 Open Networking Foundation <support@opennetworking.org>
# SPDX-License-Identifier: Apache-2.0
# service.py
#
import re
import netaddr
from .utils import logger, AttrDict
from .container import ServiceInfoContainer
def getIPaddress(addressWithMask):
return str(netaddr.IPNetwork(addressWithMask).ip)
def dhcpSubnetConfigGenerator():
dhcpSubnetConfigs = list()
serviceInfoContainer = ServiceInfoContainer()
serviceInfoContainer.initialize()
for domainName, domain in serviceInfoContainer.all():
subnetConfig = {
"dns_search": [domainName],
"dns_servers": [domain["dnsServer"]["address"]],
"ntp_servers": [domain["ntpServer"]["address"]],
"tftpd_server": domain["dhcpServer"]["address"],
"range": domain["dhcprange"][0],
"subnet": domain["subnet"],
"routers": domain["router"],
"hosts": list(),
}
for address, host in domain["hosts"].items():
subnetConfig["hosts"].append(
{
"ip_addr": getIPaddress(address),
"mac_addr": host["macaddr"],
"name": host["hostname"],
}
)
subnetConfig["hosts"] = sorted(
subnetConfig["hosts"], key=lambda x: int(x["ip_addr"].split(".")[-1])
)
dhcpSubnetConfigs.append(subnetConfig)
return dhcpSubnetConfigs
def dnsFowardZoneConfigGenerator():
def getDomainNameByIP(ip_address):
"""
getDomainNameByIP will return the corresponding domain name of an IP address
In the ntpServer, dhcpServer, dnsServer we only have the IP addresses of them,
But we can use the dhcp subnet configuration to find the FQDN
"""
dhcpSubnetConfigs = dhcpSubnetConfigGenerator()
for domain in dhcpSubnetConfigs:
domainName = domain["dns_search"][0]
for host in domain["hosts"]:
if ip_address == host["ip_addr"]:
return f"{host['name']}.{domainName}."
dnsForwardZoneConfigs = dict()
serviceInfoContainer = ServiceInfoContainer()
serviceInfoContainer.initialize()
for domainName, domain in serviceInfoContainer.all():
forwardZoneConfig = {
"cname": dict(),
"a": dict(),
"ns": list(),
"srv": dict(),
"txt": dict(),
}
# Get the services set to this Tenant network
ntpServer = domain["ntpServer"] or None
dhcpServer = domain["dhcpServer"] or None
dnsServer = domain["dnsServer"] or None
# If service exists, set the FQDN to CNAME records
if ntpServer:
forwardZoneConfig["cname"]["ntp"] = getDomainNameByIP(ntpServer["address"])
if dhcpServer:
forwardZoneConfig["cname"]["tftp"] = getDomainNameByIP(dhcpServer["address"])
if dnsServer:
forwardZoneConfig["cname"]["ns"] = getDomainNameByIP(dnsServer["address"])
forwardZoneConfig["ns"].append(getDomainNameByIP(dnsServer["address"]))
for address, host in domain["hosts"].items():
# Add exist IP address into dnsReverseZoneConfigs,
hostname = host["hostname"]
forwardZoneConfig["a"][hostname] = address
for address in netaddr.IPSet(domain["dhcprange"]):
# If address exists in ServiceInfoContainer's host dictionary,
# Use the pre-generated hostname as A record's name
hostname = "dhcp%03d" % address.words[-1]
forwardZoneConfig["a"][hostname] = str(address)
dnsForwardZoneConfigs[domainName] = forwardZoneConfig
return dnsForwardZoneConfigs
def dnsReverseZoneConfigGenerator():
def canonicalize_rfc1918_prefix(prefix):
"""
RFC1918 prefixes need to be expanded to their widest canonical range to
group all reverse lookup domains together for reverse DNS with NSD/Unbound.
"""
pnet = netaddr.IPNetwork(prefix)
(o1, o2, o3, o4) = pnet.network.words # Split ipv4 octets
cidr_plen = pnet.prefixlen
if o1 == 10:
o2 = o3 = o4 = 0
cidr_plen = 8
elif (o1 == 172 and o2 >= 16 and o2 <= 31) or (o1 == 192 and o2 == 168):
o3 = o4 = 0
cidr_plen = 16
return "%s/%d" % (".".join(map(str, [o1, o2, o3, o4])), cidr_plen)
serviceInfoContainer = ServiceInfoContainer()
serviceInfoContainer.initialize()
# dnsReverseZoneConfigs contains all reverse zone records.
dnsReverseZoneConfigs = dict()
widedomain = None
for domainName, domain in serviceInfoContainer.all():
# Expand network range to group all tenant domains
widedomain = widedomain or canonicalize_rfc1918_prefix(domain["subnet"])
# Create the basic structure of reverse zone config
# {"10.0.0.0/8": {"ns": list(), "ptr": dict()}}
dnsReverseZoneConfigs.setdefault(widedomain, dict())
dnsReverseZoneConfigs[widedomain].setdefault("ns", list())
dnsReverseZoneConfigs[widedomain].setdefault("ptr", dict())
# Get the DNS services set to this Tenant network
dnsServer = domain["dnsServer"]["name"] if domain["dnsServer"] else None
# If service exists, set the FQDN to CNAME records
if dnsServer:
dnsReverseZoneConfigs[widedomain]["ns"].append(f"{domainName}.{dnsServer}.")
for address, host in domain["hosts"].items():
# Add exist IP address into dnsReverseZoneConfigs,
hostname = host["hostname"]
dnsReverseZoneConfigs[widedomain]["ptr"][
address
] = f"{hostname}.{domainName}."
for address in netaddr.IPSet(domain["dhcprange"]):
# Add DHCP range IP address into dnsReverseZoneConfigs,
# Use the pre-generated hostname as A record's name
hostname = "dhcp%03d" % address.words[3]
dnsReverseZoneConfigs[widedomain]["ptr"][
str(address)
] = f"{hostname}.{domainName}."
dnsReverseZoneConfigs[widedomain]["ptr"] = dict(
sorted(
dnsReverseZoneConfigs[widedomain]["ptr"].items(),
key=lambda x: (int(x[0].split(".")[-2]), int(x[0].split(".")[-1])),
)
)
return dnsReverseZoneConfigs