域名是如何绑定动态IP的?

网络 通信技术
一般家庭网络的公网IP都是不固定的,而我又想通过域名来访问自己服务器上的应用,也就是说:需要通过将域名绑定到动态IP上来实现这个需求。于是乎,我开始探索实现的技术方案。

[[439935]]

一般家庭网络的公网IP都是不固定的,而我又想通过域名来访问自己服务器上的应用,也就是说:需要通过将域名绑定到动态IP上来实现这个需求。于是乎,我开始探索实现的技术方案。

通过在网上查阅一系列的资料后,发现阿里云可以做到实现动态域名解析DDNS。于是乎,一顿操作下来,我实现了域名绑定动态IP。这里,我们以Python为例实现。

小伙伴们注意啦:Java版源码已提交到:https://github.com/sunshinelyz/mykit-ddns

好了,说干就干,我们开始吧,走起~~

阿里云DDNS前置条件

  • 域名是在阿里云购买的
  • 地址必须是公网地址,不然加了解析也没有用

通过阿里云提供的SDK,然后自己编写程序新增或者修改域名的解析,达到动态解析域名的目的;主要应用于pppoe拨号的环境,比如家里设置了服务器,但是外网地址经常变化的场景;再比如公司的pppoe网关,需要建立虚拟的场景。

安装阿里云SDK

需要安装两个SDK库,一个是阿里云核心SDK库,一个是阿里云域名SDK库;

阿里云核心SDK库

  1. pip install aliyun-python-sdk-core 

阿里云域名SDK库

  1. pip install aliyun-python-sdk-domain 

阿里云DNSSDK库

  1. pip install aliyun-python-sdk-alidns 

设计思路

  • 获取阿里云的accessKeyId和accessSecret
  • 获取外网ip
  • 判断外网ip是否与之前一致
  • 外网ip不一致时,新增或者更新域名解析记录

实现方案

这里,我直接给出完整的Python代码,小伙伴们自行替换AccessKey和AccessSecret。

  1. #!/usr/bin/env python 
  2. #coding=utf-8 
  3.  
  4. # 加载核心SDK 
  5. from aliyunsdkcore.client import AcsClient 
  6. from aliyunsdkcore.acs_exception.exceptions import ClientException 
  7. from aliyunsdkcore.acs_exception.exceptions import ServerException 
  8.  
  9. # 加载获取 、 新增、 更新、 删除接口 
  10. from aliyunsdkalidns.request.v20150109 import DescribeSubDomainRecordsRequest, AddDomainRecordRequest, UpdateDomainRecordRequest, DeleteDomainRecordRequest 
  11.  
  12. # 加载内置模块 
  13. import json,urllib 
  14.  
  15. # AccessKey 和 Secret  建议使用 RAM 子账户的 KEY 和 SECRET 增加安全性 
  16. ID = 'xxxxxxx' 
  17. SECRET = 'xxxxxx' 
  18.  
  19. # 地区节点 可选地区取决于你的阿里云帐号等级,普通用户只有四个,分别是杭州、上海、深圳、河北,具体参考官网API 
  20. regionId = 'cn-hangzhou' 
  21.  
  22. # 配置认证信息 
  23. client = AcsClient(ID, SECRET, regionId) 
  24.  
  25. # 设置主域名 
  26. DomainName = 'binghe.com' 
  27.  
  28. # 子域名列表  列表参数可根据实际需求增加或减少值 
  29. SubDomainList = ['a''b''c'
  30.  
  31. # 获取外网IP   三个地址返回的ip地址格式各不相同,3322 的是最纯净的格式, 备选1为 json格式  备选2 为curl方式获取  两个备选地址都需要对获取值作进一步处理才能使用 
  32. def getIp(): 
  33.     # 备选地址:1, http://pv.sohu.com/cityjson?ie=utf-8    2,curl -L tool.lu/ip 
  34.     with urllib.request.urlopen('http://www.3322.org/dyndns/getip'as response: 
  35.         html = response.read() 
  36.         ip = str(html, encoding='utf-8').replace("\n"""
  37.     return ip 
  38.  
  39. # 查询记录 
  40. def getDomainInfo(SubDomain): 
  41.     request = DescribeSubDomainRecordsRequest.DescribeSubDomainRecordsRequest() 
  42.     request.set_accept_format('json'
  43.  
  44.     # 设置要查询的记录类型为 A记录   官网支持A / CNAME / MX / AAAA / TXT / NS / SRV / CAA / URL隐性(显性)转发  如果有需要可将该值配置为参数传入 
  45.     request.set_Type("A"
  46.  
  47.     # 指定查记的域名 格式为 'test.binghe.com' 
  48.     request.set_SubDomain(SubDomain) 
  49.  
  50.     response = client.do_action_with_exception(request) 
  51.     response = str(response, encoding='utf-8'
  52.  
  53.     # 将获取到的记录转换成json对象并返回 
  54.     return json.loads(response) 
  55.  
  56. # 新增记录 (默认都设置为A记录,通过配置set_Type可设置为其他记录) 
  57. def addDomainRecord(client,value,rr,domainname): 
  58.     request = AddDomainRecordRequest.AddDomainRecordRequest() 
  59.     request.set_accept_format('json'
  60.  
  61.     # request.set_Priority('1')  # MX 记录时的必选参数 
  62.     request.set_TTL('600')       # 可选值的范围取决于你的阿里云账户等级,免费版为 600 - 86400 单位为秒  
  63.     request.set_Value(value)     # 新增的 ip 地址 
  64.     request.set_Type('A')        # 记录类型 
  65.     request.set_RR(rr)           # 子域名名称   
  66.     request.set_DomainName(domainname) #主域名 
  67.  
  68.     # 获取记录信息,返回信息中包含 TotalCount 字段,表示获取到的记录条数 0 表示没有记录, 其他数字为多少表示有多少条相同记录,正常有记录的值应该为1,如果值大于1则应该检查是不是重复添加了相同的记录 
  69.     response = client.do_action_with_exception(request) 
  70.     response = str(response, encoding='utf-8'
  71.     relsult = json.loads(response) 
  72.     return relsult 
  73.  
  74. # 更新记录 
  75. def updateDomainRecord(client,value,rr,record_id): 
  76.     request = UpdateDomainRecordRequest.UpdateDomainRecordRequest() 
  77.     request.set_accept_format('json'
  78.  
  79.     # request.set_Priority('1'
  80.     request.set_TTL('600'
  81.     request.set_Value(value) # 新的ip地址 
  82.     request.set_Type('A'
  83.     request.set_RR(rr) 
  84.     request.set_RecordId(record_id)  # 更新记录需要指定 record_id ,该字段为记录的唯一标识,可以在获取方法的返回信息中得到该字段的值 
  85.  
  86.     response = client.do_action_with_exception(request) 
  87.     response = str(response, encoding='utf-8'
  88.     return response 
  89.  
  90. # 删除记录 
  91. def delDomainRecord(client,subdomain): 
  92.     info = getDomainInfo(subdomain) 
  93.     if info['TotalCount'] == 0: 
  94.         print('没有相关的记录信息,删除失败!'
  95.     elif info["TotalCount"] == 1: 
  96.         print('准备删除记录'
  97.         request = DeleteDomainRecordRequest.DeleteDomainRecordRequest() 
  98.         request.set_accept_format('json'
  99.  
  100.         record_id = info["DomainRecords"]["Record"][0]["RecordId"
  101.         request.set_RecordId(record_id) # 删除记录需要指定 record_id ,该字段为记录的唯一标识,可以在获取方法的返回信息中得到该字段的值 
  102.         result = client.do_action_with_exception(request) 
  103.         print('删除成功,返回信息:'
  104.         print(result) 
  105.     else
  106.         # 正常不应该有多条相同的记录,如果存在这种情况,应该手动去网站检查核实是否有操作失误 
  107.         print("存在多个相同子域名解析记录值,请核查后再操作!"
  108.  
  109. # 有记录则更新,没有记录则新增 
  110. def setDomainRecord(client,value,rr,domainname): 
  111.     info = getDomainInfo(rr + '.' + domainname) 
  112.     if info['TotalCount'] == 0: 
  113.         print('准备添加新记录'
  114.         add_result = addDomainRecord(client,value,rr,domainname) 
  115.         print(add_result) 
  116.     elif info["TotalCount"] == 1: 
  117.         print('准备更新已有记录'
  118.         record_id = info["DomainRecords"]["Record"][0]["RecordId"
  119.         cur_ip = getIp() 
  120.         old_ip = info["DomainRecords"]["Record"][0]["Value"
  121.         if cur_ip == old_ip: 
  122.             print ("新ip与原ip相同,不更新!"
  123.         else
  124.             update_result = updateDomainRecord(client,value,rr,record_id) 
  125.             print('更新成功,返回信息:'
  126.             print(update_result) 
  127.     else
  128.         # 正常不应该有多条相同的记录,如果存在这种情况,应该手动去网站检查核实是否有操作失误 
  129.         print("存在多个相同子域名解析记录值,请核查删除后再操作!"
  130.  
  131.  
  132. IP = getIp() 
  133.  
  134. # 循环子域名列表进行批量操作 
  135. for x in SubDomainList: 
  136.     setDomainRecord(client,IP,x,DomainName) 
  137.  
  138. # 删除记录测试 
  139. # delDomainRecord(client,'b.jsoner.com'
  140.  
  141. # 新增或更新记录测试 
  142. # setDomainRecord(client,'192.168.3.222','a',DomainName) 
  143.  
  144. # 获取记录测试 
  145. # print (getDomainInfo(DomainName, 'y')) 
  146.  
  147. # 批量获取记录测试 
  148. for x in SubDomainList: 
  149. #     print (getDomainInfo(DomainName, x)) 
  150.  
  151. # 获取外网ip地址测试 
  152. # print ('(' + getIp() + ')'

Python脚本的功能如下:

  • 获取外网ip地址。
  • 获取域名解析记录。
  • 新增域名解析记录。
  • 更新域名解析记录。
  • 删除域名解析记录 (并不建议将该功能添加在实际脚本中)。
  • 批量操作,如果记录不存在则添加记录,存在则更新记录。

另外,有几点需要特别说明:

  • 建议不要将删除记录添加进实际使用的脚本当中。
  • 相同记录是同一个子域名的多条记录,比如 test.binghe.com。
  • 脚本并没有验证记录类型,所以同一子域名下的不同类型的记录也会认为是相同记录,比如:有两条记录分别是 test.binghe.com 的 A 记录 和 test.binghe.com 的 AAAA 记录,会被认为是两条相同的 test.binghe.com 记录.如果需要判定为不同的记录,小伙伴们可以根据上述Python脚本自行实现。
  • 可以通过判断获取记录返回的 record_id 来实现精确匹配记录。

 

最后,可以将以上脚本保存为文件之后,通过定时任务,来实现定期自动更新ip地址。

 

责任编辑:武晓燕 来源: 冰河技术
相关推荐

2022-02-18 08:28:49

域名公网IP

2020-10-12 06:28:05

动态IP框架

2021-09-01 09:54:10

IP地址域名

2021-11-03 10:52:39

数据IP域名

2012-01-09 11:26:15

Java

2017-12-05 14:24:31

应用绑定域名

2017-08-07 16:39:03

JSX动态数据

2016-12-14 14:29:30

Java动态绑定机制

2009-04-09 10:11:00

TCPIP通讯

2022-02-11 09:31:23

IPV4IP地址IANA

2022-05-07 09:49:59

IPSSLLinux

2009-07-22 08:52:05

Scala动态绑定

2020-02-10 19:05:46

DNS域名

2011-03-04 15:20:09

vsFTPDIP

2018-02-23 09:32:13

LinuxUnixntpd

2009-04-08 15:56:31

Linux路由MAC绑定

2009-09-02 20:18:17

域名劫持域名安全

2010-09-25 16:26:32

DHCP保留特定IP地

2015-02-28 15:45:44

IP端口域名

2010-05-20 17:52:02

点赞
收藏

51CTO技术栈公众号