use the new shiny BlockipError
[toast/tdyndns.git] / bin / tdyndns_update
1 #!/usr/bin/python
2 import sys
3 import re
4 import argparse
5 from subprocess import Popen, PIPE, call
6 import ipaddr
7
8
9 class NsupdateError(Exception):
10         def __init__(self, returncode):
11                 self.returncode = returncode
12
13
14 class BlockipError(Exception):
15         def __init__(self, returncode):
16                 self.returncode = returncode
17
18
19 def ipfamily_by_ip(ip):
20         if isinstance(ip, ipaddr.IPv4Address):
21                 return 'A'
22         elif isinstance(ip, ipaddr.IPv6Address):
23                 return 'AAAA'
24         assert False
25
26
27 def nsupdate_add(fqdn, ttl, ip):
28         """
29         :param fqdn: Fully qualified domain name
30         :param ip_family: A or AAAA
31         :raises an NsupdateError in case of errors."""
32         command = "update add {fqdn} {ttl} IN {ip_family} {ip}\n\n".format(fqdn=fqdn, ttl=ttl, ip_family=ipfamily_by_ip(ip), ip=ip)
33         p = Popen(['nsupdate', '-l'], stdin=PIPE)
34         p.communicate(command)
35         if p.returncode != 0:
36                 raise NsupdateError(p.returncode)
37
38 def nsupdate_delete(fqdn, ip_family):
39         """
40         :param fqdn: Fully qualified domain name
41         :param ip_family: A or AAAA
42         :raises an NsupdateError in case of errors."""
43         command = "update delete {fqdn} {ip_family}\n\n".format(fqdn=fqdn, ip_family=ip_family)
44         p = Popen(['nsupdate', '-l'], stdin=PIPE)
45         p.communicate(command)
46         if p.returncode != 0:
47                 raise NsupdateError(p.returncode)
48
49
50 def blockip_whitelist_add(ip):
51         """
52         :param ip: ipv4 address
53         :raises a BlockipError in case of errors."""
54         command = "iptables -I blockip -s {ip} -j ACCEPT".format(ip=ip)
55         p = call(command, shell=True)
56         if p != 0:
57                 raise BlockipError(p)
58
59
60 def blockip_whitelist_delete(ip):
61         """
62         :param ip: ipv4 address
63         :raises a BlockipError in case of errors."""
64         command = "iptables -D blockip -s {ip} -j ACCEPT".format(ip=ip)
65         p = call(command, shell=True)
66         if p != 0:
67                 raise BlockipError(p)
68
69
70 def main(args):
71         try:
72                 if args.delete:
73                         if args.ip is None:
74                                 nsupdate_delete(args.fqdn, 'A')
75                                 nsupdate_delete(args.fqdn, 'AAAA')
76                         else:
77                                 nsupdate_delete(args.fqdn, ipfamily_by_ip(args.ip))
78                                 if ipfamily_by_ip(args.ip) == 'A':
79                                         blockip_whitelist_delete(args.ip)
80                 else:
81                         nsupdate_delete(args.fqdn, ipfamily_by_ip(args.ip))
82                         nsupdate_add(args.fqdn, args.ttl, args.ip)
83                         if ipfamily_by_ip(args.ip) == 'A':
84                                 blockip_whitelist_add(args.ip)
85         except NsupdateError as e:
86                 sys.exit(e.returncode)
87
88
89
90 if __name__ == '__main__':
91         parser = argparse.ArgumentParser(description='Add or delete a domain name from dyndns (simplifies call to nsupdate).')
92         parser.add_argument('-d', '--delete', action='store_true', help='delete instead of add')
93         parser.add_argument('-i', '--ip', help='IP address (either IPv4 or IPv6)')
94         parser.add_argument('-t', '--ttl', type=int, default=600, help='TTL (default: 600)')
95         parser.add_argument('fqdn', help='fully qualified domain name to add or delete, e.g. myserver.dyn.example.com')
96         args = parser.parse_args()
97
98         # check ip
99         if not args.delete and not args.ip:
100                 parser.error('The IP address is mandatory')
101         if args.ip:
102                 try:
103                         args.ip = ipaddr.IPAddress(args.ip) # throws an exception if the IP address is not valid
104                 except ValueError:
105                         parser.error('The IP address is not valid')
106
107         # check fqdn
108         if re.match(r'[-0-9a-z]+(\.[-0-9a-z]+)*$', args.fqdn) is None:
109                 parser.error('The fqdn has an invalid format.')
110
111         main(args)