virtual-dsm/generate-dhcpd-conf

81 lines
2.2 KiB
Plaintext
Raw Normal View History

2019-03-11 00:35:19 -07:00
#!/usr/bin/env python3
import argparse
import ipaddress
import json
import re
2019-03-11 22:32:16 -07:00
import socket
2019-03-11 00:35:19 -07:00
import subprocess
from typing import List, Iterable
2019-03-11 00:35:19 -07:00
DEFAULT_ROUTE = 'default'
DEFAULT_DNS_IPS = ('8.8.8.8', '8.8.4.4')
2019-03-11 00:35:19 -07:00
DHCP_CONF_TEMPLATE = """
start {host_addr}
end {host_addr}
# avoid dhcpd complaining that we have
# too many addresses
maxleases 1
interface {dhcp_intf}
2019-03-11 22:32:16 -07:00
option dns {dns}
option router {gateway}
option subnet {subnet}
option hostname {hostname}
2019-03-11 00:35:19 -07:00
"""
def default_route(routes):
"""Returns the host's default route"""
for route in routes:
if route['dst'] == DEFAULT_ROUTE:
return route
raise ValueError('no default route')
def addr_of(addrs, dev : str) -> ipaddress.IPv4Interface:
"""Finds and returns the IP address of `dev`"""
for addr in addrs:
if addr['ifname'] != dev:
continue
if len(addr['addr_info']) != 1:
raise ValueError('only exactly one address on dev is supported')
info = addr['addr_info'][0]
return ipaddress.IPv4Interface((info['local'], info['prefixlen']))
raise ValueError('dev {0} not found'.format(dev))
def generate_conf(intf_name : str, dns : Iterable[str]) -> str:
2019-03-11 00:35:19 -07:00
"""Generates a dhcpd config. `intf_name` is the interface to listen on."""
with subprocess.Popen(['ip', '-json', 'route'],
stdout=subprocess.PIPE) as proc:
routes = json.load(proc.stdout)
with subprocess.Popen(['ip', '-json', 'addr'],
stdout=subprocess.PIPE) as proc:
addrs = json.load(proc.stdout)
droute = default_route(routes)
host_addr = addr_of(addrs, droute['dev'])
return DHCP_CONF_TEMPLATE.format(
dhcp_intf = intf_name,
dns = ' '.join(dns),
2019-03-11 00:35:19 -07:00
gateway = droute['gateway'],
2019-03-11 22:32:16 -07:00
host_addr = host_addr.ip,
hostname = socket.gethostname(),
2019-03-11 00:35:19 -07:00
subnet = host_addr.network.netmask,
)
if __name__ == '__main__':
parser = argparse.ArgumentParser()
parser.add_argument('intf_name')
parser.add_argument('dns_ips', nargs='*')
2019-03-11 00:35:19 -07:00
args = parser.parse_args()
dns_ips = args.dns_ips
if not dns_ips:
dns_ips = DEFAULT_DNS_IPS
print(generate_conf(args.intf_name, dns_ips))