-
Notifications
You must be signed in to change notification settings - Fork 0
/
generate-dhcpd-conf
executable file
·80 lines (65 loc) · 2.21 KB
/
generate-dhcpd-conf
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
#!/usr/bin/env python3
import argparse
import ipaddress
import json
import re
import socket
import subprocess
from typing import List, Iterable
DEFAULT_ROUTE = 'default'
DEFAULT_DNS_IPS = ('8.8.8.8', '8.8.4.4')
DHCP_CONF_TEMPLATE = """
start {host_addr}
end {host_addr}
# avoid dhcpd complaining that we have
# too many addresses
max_leases 1
interface {dhcp_intf}
option dns {dns}
option router {gateway}
option subnet {subnet}
option hostname {hostname}
"""
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:
"""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),
gateway = droute['gateway'],
host_addr = host_addr.ip,
hostname = socket.gethostname(),
subnet = host_addr.network.netmask,
)
if __name__ == '__main__':
parser = argparse.ArgumentParser()
parser.add_argument('intf_name')
parser.add_argument('dns_ips', nargs='*')
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))