阿里云域名动态IP解析Shell小脚本

当你手里面有一个闲置的域名,有一个漂浮不定的外网ip,你又想在外网ip变更后自动解析到域名上,此情此景此脚本可能会帮上你。

  • 2020-09-29 支持IPV6解析(支持设置解析记录类型:A、NS、MX、TXT、CNAME、SRV、AAAA、CAA、REDIRECT_URL、FORWARD_URL)

使用步骤:

  • 在阿里云域名管理中解析自己的域名
    • 记录类型:A
    • 主机记录:按你需要输入
    • 解析路线:默认(随你)
    • 记录值:随便输入一个Ip地址(运行脚本后会自动更新到正确的)
    • TTL值:10分钟
  • 需要通过阿里云域名解析Api操作,所以需要申请阿里云的Acesskeys
  • 运行自动解析域名脚本
    • git clone https://github.com/risfeng/aliyun-ddns-shell.git
    • 给aliyun-ddns.sh脚本赋可执行权:chmod +x aliyun-ddns-shell/src/aliyun/aliyun-ddns.sh
    • 运行脚本,根据提示进行配置(第一次运行需要配置)
    • 如需要使用钉钉自定义机器人推送成功失败消息,请新建自定义钉钉机器人,配置access_token和消息加签密钥
    • 脚本支持2种运行方式:
      • 直接运行脚本根据提示选择需要使用的功能
      > aliyun-ddns.sh (mac: > bash aliyun-ddns.sh) 1.配置并执行脚本 2.仅配置 3.仅执行脚本 4.恢复出厂设置 5.清理日志文件 0.退出 
      • 带参数运行:
      aliyun-ddns.sh -config -run 配置并执行脚 aliyun-ddns.sh -run 执行脚本(前提需要有配置文件) aliyun-ddns.sh -config 仅配置信息 aliyun-ddns.sh -restore 恢复出厂设置(会清除配置文件等) aliyun-ddns.sh -clearn 清理日志文件 aliyun-ddns.sh -version 显示脚本说明及版本信息 
  • 实时监听外网IP变更后自动解析到域名
    • 利用定时任务服务:crond
    • 检测是否安装:crond -V 有输出版本号即已安装。
    • 如未安装:yum install vixie-cron crontabs -y 如有疑问请自行查找资料解决
    • crond服务常用命令
    service crond status # 查看服务运行状态 service crond start # 启动服务 service crond stop # 停止服务 service crond restart # 重启服务 service crond reload # 不中断服务,重新载入配置 crontab -e # 编辑配置文件 crontab -l # 列出某个用户的任务计划 
    • 配置任务定时任务
      • 建议不要把任务执行频率设置小于等10,因为域名解析记录生效时间最短理论上是10分钟,往往都会超过10分,建议15-20分钟。
      • 开始配置
      crontab -e # 按i进入标记模式 # 输入: */20 * * * * /root/aliyun-ddns-shell/src/aliyun/aliyun-ddns.sh -run >> /root/aliyun-ddns-shell/src/aliyun/crontab-log.log # 说明: # */20 * * * * :每20分钟执行一次 需要执行都脚本全路径 >> 执行日志输出位置全路径 # 按 esc 后 输入:wq 回车 保存并退出 
      • 重新加载配置:service cron restart
      • 注意观察任务是否成功执行,如有疑问请自行百度

效果图

前置条件

1、域名是在阿里云购买的 (我的域名本身就是阿里云买的,其他的域名我没有测试过)
2、地址必须是公网地址,不然加了解析也没有用 (这个不用多加解释了)

安装阿里云SDK

需要安装三个SDK库,一个是阿里云核心SDK库,一个是阿里云域名SDK库,一个是DNS库
阿里云核心SDK库:pip install aliyun-python-sdk-core
阿里云域名SDK库:pip install aliyun-python-sdk-domain
阿里云DNSSDK库:pip install aliyun-python-sdk-alidns
说明:
1, 目前官方文档上说的是需要安装前两个SDK库,但我实测确定还需要第三个库!
2, 如果你使用的是ubuntu系统,并且同时有python2/python3和pip2/pip3,安装时前面的命令请写清楚版本,特别是pip。 我的系统pip默认是2.x,使用pip install xxx 命令安装的库无效,使用 pip3 install xxx 才成功(如果知道如何随意切换python和pip版本,请留言告知,感谢)。

脚本的具体功能

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

脚本代码

#!/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 = 'example.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.example.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() + ')') 

相同记录是同一个子域名的多条记录,比如 test.example.com。
脚本并没有验证记录类型,所以同一子域名下的不同类型的记录也会认为是相同记录,比如:
有两条记录分别是 test.examlpe.com 的 A 记录 和 test.examlpe.com 的 CNAME 记录,会被认为是两条相同的 test.example.com 记录

作者:O无为学长O
链接:https://www.jianshu.com/p/cd3fe4fa8ea8
来源:简书
著作权归作者所有。商业转载请联系作者获得授权,非商业转载请注明出处。

原文链接:https://blog.csdn.net/mier9042/article/details/118338717?ops_request_misc=%257B%2522request%255Fid%2522%253A%2522168449620216782425143177%2522%252C%2522scm%2522%253A%252220140713.130102334.pc%255Fblog.%2522%257D&request_id=168449620216782425143177&biz_id=0&utm_medium=distribute.pc_search_result.none-task-blog-2~blog~first_rank_ecpm_v1~times_rank-26-118338717-null-null.blog_rank_default&utm_term=NAS%E3%80%81%E7%BE%A4%E6%99%96%E3%80%81%E9%98%BF%E9%87%8C%E4%BA%91%E3%80%81%E5%9F%9F%E5%90%8D%E8%A7%A3%E6%9E%90%E3%80%81%E5%86%85%E7%BD%91%E7%A9%BF%E9%80%8F%E3%80%81ipv6%E3%80%81ddns%E3%80%81%E8%BD%BB%E9%87%8F%E7%BA%A7%E4%BA%91%E6%9C%8D%E5%8A%A1%E5%99%A8%E3%80%81%E9%93%81%E5%A8%81%E9%A9%AC%E3%80%81%E5%A8%81%E8%81%94%E9%80%9A%E3%80%81DSM%E3%80%81DSM6.0%E3%80%81%E7%BE%A4%E6%99%96nas%E3%80%81%E4%BA%91%E6%9C%8D%E5%8A%A1%E5%99%A8%E3%80%81%E8%9C%97%E7%89%9B%E6%98%9F%E9%99%85%E3%80%81%E9%BB%91%E7%BE%A4%E6%99%96%E3%80%81docker%E3%80%81%E5%AE%B9%E5%99%A8%E9%95%9C%E5%83%8F%E3%80%81%E5%9F%9F%E5%90%8D%E6%B3%A8%E5%86%8C%E3%80%81%E5%AE%9D%E5%A1%94%E3%80%81%E5%8F%8D%E5%90%91%E4%BB%A3%E7%90%86%E3%80%81nginx%E3%80%81frp%E3%80%81%E5%8A%A8%E6%80%81%E5%9F%9F%E5%90%8D%E8%A7%A3%E6%9E%90

© 版权声明
THE END
喜欢就支持一下吧
点赞0 分享