域名还能绑定动态IP?看完又涨知识了!!

作者阿里云代理 文章分类 分类:linux图文教程 阅读次数 已被围观 753

阿里云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 python #coding=utf-8 # 加载核心SDK from aliyunsdkcore.client import AcsClient from aliyunsdkcore.acs_exception.exceptions import ClientException from aliyunsdkcore.acs_exception.exceptions import ServerException # 加载获取 、 新增、 更新、 删除接口 from aliyunsdkalidns.request.v20150109 import DescribeSubDomainRecordsRequest, AddDomainRecordRequest, UpdateDomainRecordRequest, DeleteDomainRecordRequest # 加载内置模块 import json,urllib # AccessKey 和 Secret  建议使用 RAM 子账户的 KEY 和 SECRET 增加安全性 ID = 'xxxxxxx' SECRET = 'xxxxxx' # 地区节点 可选地区取决于你的阿里云帐号等级,普通用户只有四个,分别是杭州、上海、深圳、河北,具体参考官网API regionId = 'cn-hangzhou' # 配置认证信息 client = AcsClient(ID, SECRET, regionId) # 设置主域名 DomainName = 'binghe.com' # 子域名列表  列表参数可根据实际需求增加或减少值 SubDomainList = ['a', 'b', 'c'] # 获取外网IP   三个地址返回的ip地址格式各不相同,3322 的是最纯净的格式, 备选1为 json格式  备选2 为curl方式获取  两个备选地址都需要对获取值作进一步处理才能使用 def getIp():  # 备选地址:1, http://pv.sohu.com/cityjson?ie=utf-8    2,curl -L tool.lu/ip  with urllib.request.urlopen('http://www.3322.org/dyndns/getip') as response:  html = response.read()  ip = str(html, encoding='utf-8').replace("\n", "")  return ip # 查询记录 def getDomainInfo(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.com'  request.set_SubDomain(SubDomain)  response = client.do_action_with_exception(request)  response = str(response, encoding='utf-8')  # 将获取到的记录转换成json对象并返回  return json.loads(response) # 新增记录 (默认都设置为A记录,通过配置set_Type可设置为其他记录) def addDomainRecord(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)  return relsult # 更新记录 def updateDomainRecord(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')  return response # 删除记录 def delDomainRecord(client,subdomain):  info = getDomainInfo(subdomain)  if info['TotalCount'] == 0:  print('没有相关的记录信息,删除失败!')  elif info["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("存在多个相同子域名解析记录值,请核查后再操作!") # 有记录则更新,没有记录则新增 def setDomainRecord(client,value,rr,domainname):  info = getDomainInfo(rr + '.' + domainname)  if info['TotalCount'] == 0:  print('准备添加新记录')  add_result = addDomainRecord(client,value,rr,domainname)  print(add_result)  elif info["TotalCount"] == 1:  print('准备更新已有记录')  record_id = info["DomainRecords"]["Record"][0]["RecordId"]  cur_ip = getIp()  old_ip = info["DomainRecords"]["Record"][0]["Value"]  if cur_ip == old_ip:  print ("新ip与原ip相同,不更新!")  else:  update_result = updateDomainRecord(client,value,rr,record_id)  print('更新成功,返回信息:')  print(update_result)  else:  # 正常不应该有多条相同的记录,如果存在这种情况,应该手动去网站检查核实是否有操作失误  print("存在多个相同子域名解析记录值,请核查删除后再操作!") IP = getIp() # 循环子域名列表进行批量操作 for x in SubDomainList:  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地址。

本公司销售:阿里云、腾讯云、百度云、天翼云、金山大米云、金山企业云盘!可签订合同,开具发票。

我有话说: