I use DynDns to map my ISP provided dynamic IP to a static hostname. For some reason, the DynDns provided update clients don’t always work for me and often leave my hostname pointing to a dead or (worse) someone else’s IP. I decided to take matter into my own hands and write a script that would email me my IP whenever my DHCP lease expired and my ISP issued me a fresh one. This would ensure I know how to reach back home, even if my hostname was pointing to an old IP.
While going through the DynDns API, I realized it was trivial to update the hostname as well, essentially replicating the functionality of the aforementioned client(s). So I decided to add that as well.
I know this functionality can be replicated via curl + sendmail, but Python is my tool of choice, so just live with it. Without further ado, here’s the script, with an explanation afterwards.
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 | #!/usr/bin/env python # encoding: utf-8 """ Script to email IP whenever it changes. Also updates DynDns hostname. Version 1.0 Created by Kunal Dua on 2010-05-10 http://www.kunaldua.com/blog/?p=360 This program is free software; you may redistribute it and/or modify it under the same terms as Python itself. """ def send_mail(subject, content): import smtplib from email.mime.text import MIMEText SERVER = "smtpserver" PORT = 587 #Use 25 if this doesn't work USER = "username" PASS = "password" FROM = "IPBot <mail@example.com>" TO = "user+folder@example.com" SUBJECT = subject TEXT = content message = MIMEText(TEXT) message['Subject'] = SUBJECT message['From'] = FROM message['To'] = TO server = smtplib.SMTP(SERVER, PORT) server.login (USER, PASS) server.sendmail(FROM, TO, message.as_string()) server.quit() def update_dyndns(theip): USERNAME = 'username' PASSWORD = 'password' HOSTNAME = 'example.dyndns.org' theurl = 'https://%s:%s@members.dyndns.org/nic/update?hostname=%s&myip=%s&wildcard=NOCHG&mx=NOCHG&backmx=NOCHG' % (USERNAME, PASSWORD, HOSTNAME, theip) import urllib conn = urllib.urlopen(theurl) #print conn.read() conn.close() if __name__ == '__main__': import urllib2, re conn = urllib2.urlopen('http://checkip.dyndns.com/') data = conn.read() conn.close() m = re.search('([0-9]*)(\.)([0-9]*)(\.)([0-9]*)(\.)([0-9]*)', data) currip = m.group(0) lastfile = "lastip.txt" allfile = "history.txt" theipfile = open(lastfile,"r") lastip = theipfile.read() theipfile.close() if lastip == currip: #print "no change needed" exit() else: histfile = open(allfile, "a") import datetime thenow = datetime.datetime.now().ctime() histfile.write("%s %s\n" % (thenow, currip)) histfile.close() theipfile = open(lastfile,"w") theipfile.write(currip) theipfile.close() send_mail(currip, '') update_dyndns(currip) |
- Lines 17-22 and 39-41 replace with your email and DynDns settings respectively.
- Line 22 – My email provider supports redirecting mails to a folder by simply adding the name of the folder before @ sign. For example user+ipupdates@example.com will deliver mail in folder ipupdates of user@example.com. If your email provider supports this, it’s a useful trick to prevent these mails from cluttering up your inbox. If not, simply enter your regular email address.
- Lines 60-61 initialize 2 files that I use. One is to store the current IP (or the last known IP) and the other is a history of all IP changes. The former is used to compare if the IP has changed since the script was last run and thus if an email needs to be sent + DynDns updated. The latter is not really needed for the script to function properly, and is used to maintain a log of all IP changes – because you can!
- Line 78 – By default, the subject of the mail is the IP and the body/ text is blank. Feel free to obfuscate your IP if you feel paranoid about sending it in clear text or write sweet nothings to yourself in the body.
- Note: Before you run this script for the first time, create an empty file called lastip.txt in the same directory as the script or the script will fail. I know I could write a trivial check for this, but I leave that as an exercise for the reader.
Recommended frequency of running this job via cron/ launchd is 10 minutes.
Update: (May 27) I am pretty sure the DynDns API is broken in some way because I can’t get it to update even with this script. The email part is working pretty good for me though!
Related posts:
Pingback: Tweets that mention Get email updates when your IP changes/ Python DynDns update client « Kunal Dua -- Topsy.com
Pingback: Deep Kalra, Kunal Bajaj, PN Vi… | smsyellowpages.net
Pingback: How To Set Up A Home Server | www.avehot.com
I will recommend not to wait until you get enough amount of cash to order all you need! You should take the mortgage loans or just college loan and feel comfortable
Pingback: Apache? Dyndns? Configurations? | High Speed Routers
I tried this, but there seems to be a problem with urllib with the colon in the address, I was wondering if you know of a way to get around that. It seems to take whatever is after the colon as the port.
Thanks for this! I had to make a few changes. Specifically, to get the authorization right, I removed the username:password@ section, and instead it looks like:
def update_dyndns(theip):
theurl = ‘https://members.dyndns.org/nic/update?hostname=%s&myip=%s&wildcard=NOCHG&mx=NOCHG&backmx=NOCHG‘ % (HOSTNAME, theip)
#create the credential file
auth_handler = urllib2.HTTPBasicAuthHandler()
auth_handler.add_password(realm=’DynDNS API Access’,
uri=’https://members.dyndns.org/nic/update’,
user=USERNAME,
passwd=PASSWORD)
opener = urllib2.build_opener(auth_handler)
# …and install it globally so it can be used with urlopen.
urllib2.install_opener(opener)
req = urllib2.Request(theurl)
req.add_header(‘User-agent’, ‘your-user-agent-here’)
conn = urllib2.urlopen(req)
#print conn.read()
return conn
Great write-up, but that wouldn’t really to make use of my router ip, any hints?
It’s actually a cool and useful piece of information. I’m satisfied that you simply shared this helpful information with us. Please keep us up to date like this. Thanks for sharing.
Aw, this was a very nice post. In concept I wish to put in writing like this additionally – taking time and precise effort to make a very good article… but what can I say… I procrastinate alot and certainly not appear to get something done.
A formidable share, I simply given this onto a colleague who was doing just a little analysis on this. And he in truth purchased me breakfast because I discovered it for him.. smile. So let me reword that: Thnx for the deal with! But yeah Thnkx for spending the time to debate this, I feel strongly about it and love reading more on this topic. If possible, as you turn into expertise, would you thoughts updating your blog with extra particulars? It’s extremely helpful for me. Large thumb up for this weblog put up!
I’m not positive where you’re getting your information, however good topic. I needs to spend some time finding out much more or understanding more. Thanks for fantastic information I used to be searching for this info for my mission.