域名是如何绑定动态IP的?
一般家庭网络的公网IP都是不固定的,而我又想通过域名来访问自己服务器上的应用,也就是说:需要通过将域名绑定到动态IP上来实现这个需求。于是乎,我开始探索实现的技术方案。
通过在网上查阅一系列的资料后,发现阿里云可以做到实现动态域名解析DDNS。于是乎,一顿操作下来,我实现了域名绑定动态IP。这里,我们以Python为例实现。
小伙伴们注意啦:Java版源码已提交到:https://github.com/sunshinelyz/mykit-ddns
好了,说干就干,我们开始吧,走起~~
阿里云DDNS前置条件
域名是在阿里云购买的
地址必须是公网地址,不然加了解析也没有用
通过阿里云提供的SDK,然后自己编写程序新增或者修改域名的解析,达到动态解析域名的目的;主要应用于pppoe拨号的环境,比如家里设置了服务器,但是外网地址经常变化的场景;再比如公司的pppoe网关,需要建立vpn的场景。
安装阿里云SDK
需要安装两个SDK库,一个是阿里云核心SDK库,一个是阿里云域名SDK库;
阿里云核心SDK库
pip install aliyun-python-sdk-core
阿里云域名SDK库
pip install aliyun-python-sdk-domain
阿里云DNSSDK库
pip install aliyun-python-sdk-alidns
设计思路
获取阿里云的accessKeyId和accessSecret
获取外网ip
判断外网ip是否与之前一致
外网ip不一致时,新增或者更新域名解析记录
实现方案
这里,我直接给出完整的Python代码,小伙伴们自行替换AccessKey和AccessSecret。
!/usr/bin/env pythoncoding=utf-8加载核心SDKfromaliyunsdkcore.clientimportAcsClientfromaliyunsdkcore.acs_exception.exceptionsimportClientExceptionfromaliyunsdkcore.acs_exception.exceptionsimportServerException加载获取 、 新增、 更新、 删除接口fromaliyunsdkalidns.request.v20150109importDescribeSubDomainRecordsRequest, AddDomainRecordRequest, UpdateDomainRecordRequest, DeleteDomainRecordRequest加载内置模块importjson,urllibAccessKey 和 Secret 建议使用 RAM 子账户的 KEY 和 SECRET 增加安全性ID =xxxxxxxSECRET =xxxxxx地区节点 可选地区取决于你的阿里云帐号等级,普通用户只有四个,分别是杭州、上海、深圳、河北,具体参考官网APIregionId =cn-hangzhou配置认证信息client = AcsClient(ID, SECRET, regionId)设置主域名DomainName =binghe.com子域名列表 列表参数可根据实际需求增加或减少值SubDomainList = [a,b,c]获取外网IP 三个地址返回的ip地址格式各不相同,3322 的是最纯净的格式, 备选1为 json格式 备选2 为curl方式获取 两个备选地址都需要对获取值作进一步处理才能使用defgetIp():备选地址:1, http://pv.sohu.com/cityjson?ie=utf-8 2,curl -L tool.lu/ipwithurllib.request.urlopen(http://www.3322.org/dyndns/getip)asresponse:html = response.read()ip = str(html, encoding=utf-8).replace("\n","")returnip查询记录defgetDomainInfo(SubDomain):request = DescribeSubDomainRecordsRequest.DescribeSubDomainRecordsRequest()request.set_accept_format(json)设置要查询的记录类型为 A记录 官网支持A / CNAME / MX / AAAA / TXT / NS / SRV / CAA / URL隐性(显性)转发 如果有需要可将该值配置为参数传入request.set_Type("A")指定查记的域名 格式为 test.binghe.comrequest.set_SubDomain(SubDomain)response = client.do_action_with_exception(request)response = str(response, encoding=utf-8)将获取到的记录转换成json对象并返回returnjson.loads(response)新增记录 (默认都设置为A记录,通过配置set_Type可设置为其他记录)defaddDomainRecord(client,value,rr,domainname):request = AddDomainRecordRequest.AddDomainRecordRequest()request.set_accept_format(json)request.set_Priority(1) MX 记录时的必选参数request.set_TTL(600)可选值的范围取决于你的阿里云账户等级,免费版为 600 - 86400 单位为秒request.set_Value(value)新增的 ip 地址request.set_Type(A)记录类型request.set_RR(rr)子域名名称request.set_DomainName(domainname)主域名获取记录信息,返回信息中包含 TotalCount 字段,表示获取到的记录条数 0 表示没有记录, 其他数字为多少表示有多少条相同记录,正常有记录的值应该为1,如果值大于1则应该检查是不是重复添加了相同的记录response = client.do_action_with_exception(request)response = str(response, encoding=utf-8)relsult = json.loads(response)returnrelsult更新记录defupdateDomainRecord(client,value,rr,record_id):request = UpdateDomainRecordRequest.UpdateDomainRecordRequest()request.set_accept_format(json)request.set_Priority(1)request.set_TTL(600)request.set_Value(value)新的ip地址request.set_Type(A)request.set_RR(rr)request.set_RecordId(record_id)更新记录需要指定 record_id ,该字段为记录的唯一标识,可以在获取方法的返回信息中得到该字段的值response = client.do_action_with_exception(request)response = str(response, encoding=utf-8)returnresponse删除记录defdelDomainRecord(client,subdomain):info = getDomainInfo(subdomain)ifinfo[TotalCount] ==0:print(没有相关的记录信息,删除失败!)elifinfo["TotalCount"] ==1:print(准备删除记录)request = DeleteDomainRecordRequest.DeleteDomainRecordRequest()request.set_accept_format(json)record_id = info["DomainRecords"]["Record"][0]["RecordId"]request.set_RecordId(record_id)删除记录需要指定 record_id ,该字段为记录的唯一标识,可以在获取方法的返回信息中得到该字段的值result = client.do_action_with_exception(request)print(删除成功,返回信息:)print(result)else:正常不应该有多条相同的记录,如果存在这种情况,应该手动去网站检查核实是否有操作失误print("存在多个相同子域名解析记录值,请核查后再操作!")有记录则更新,没有记录则新增defsetDomainRecord(client,value,rr,domainname):info = getDomainInfo(rr +.+ domainname)ifinfo[TotalCount] ==0:print(准备添加新记录)add_result = addDomainRecord(client,value,rr,domainname)print(add_result)elifinfo["TotalCount"] ==1:print(准备更新已有记录)record_id = info["DomainRecords"]["Record"][0]["RecordId"]cur_ip = getIp()old_ip = info["DomainRecords"]["Record"][0]["Value"]ifcur_ip == old_ip:print("新ip与原ip相同,不更新!")else:update_result = updateDomainRecord(client,value,rr,record_id)print(更新成功,返回信息:)print(update_result)else:正常不应该有多条相同的记录,如果存在这种情况,应该手动去网站检查核实是否有操作失误print("存在多个相同子域名解析记录值,请核查删除后再操作!")IP = getIp()循环子域名列表进行批量操作forxinSubDomainList:setDomainRecord(client,IP,x,DomainName)删除记录测试delDomainRecord(client,b.jsoner.com)新增或更新记录测试setDomainRecord(client,192.168.3.222,a,DomainName)获取记录测试print (getDomainInfo(DomainName, y))批量获取记录测试for x in SubDomainList:print (getDomainInfo(DomainName, x))获取外网ip地址测试print (( + getIp() + ))
Python脚本的功能如下:
获取外网ip地址。
获取域名解析记录。
新增域名解析记录。
更新域名解析记录。
删除域名解析记录 (并不建议将该功能添加在实际脚本中)。
批量操作,如果记录不存在则添加记录,存在则更新记录。
另外,有几点需要特别说明:
建议不要将删除记录添加进实际使用的脚本当中。
相同记录是同一个子域名的多条记录,比如 test.binghe.com。
脚本并没有验证记录类型,所以同一子域名下的不同类型的记录也会认为是相同记录,比如:有两条记录分别是 test.binghe.com 的 A 记录 和 test.binghe.com 的 AAAA 记录,会被认为是两条相同的 test.binghe.com 记录.如果需要判定为不同的记录,小伙伴们可以根据上述Python脚本自行实现。
可以通过判断获取记录返回的 record_id 来实现精确匹配记录。
最后,可以将以上脚本保存为文件之后,通过定时任务,来实现定期自动更新ip地址。