Python实现阿里云DDNS动态域名解析

本实验的前期条件:

1、windwos下的python3.7.6版本

2、拥有一个已备案的阿里云域名

3、有公网ip(我这边用的是联通的宽带,路由拨号后分配的是公网ip,如果只有内网保留ip地址,则只能使用frp内网穿透,无法使用DDNS)

一、查看所在网络环境的支持情况

1、查看是否拥有公网ip,如果没有直接打电话给宽带运营商要求分配公网ip,否则无法实现ddns。在路由器管理页面查看获取的ip地址与 百度“ip”后显示的ip地址如果一致则说明拥有公网ip,如下:

2、如果你在该路由器下的局域网已经搭建了一个http服务器,那么还需要在路由器中进行一个端口映射,我这里是把80端口映射到http服务器ip的80端口了。协议的话只映射TCP也行。

二、建立阿里云的RAM账户,授权dns api权限

现在阿里云的RAM访问控制里面建立一个用户(创建时会验证阿里云管理员手机号验证码)

创建用户后会显示用户的AccessKey ID和 AccessKey Secret,务必复制保存下来,否则后面就查看不到了AccessKey Secret!

接下来给该用户添加DNS权限

直接输入dns进行搜索,添加AliyunDNSFullAccess

到此阶段,阿里云方面的准备工作已经完毕。

三、利用python实现阿里云DDNS的记录插入、更新、删除

阿里云 云解析DNS的API概览:https://help.aliyun.com/document_detail/29740.html?spm=a2c4g.11186623.6.624.74162009YpODyn

需要安装三个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

以下测试demo代码,是更新"ranjuan.cn"域名下的“testdns”子域名的ddns解析,也就是要动态解析“testdns.ranjuan.cn”到我们的路由器外网ip上。如果阿里云上不存在此解析,则会自动先创建。

#!/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 = 'LTAI4GCdRjssiGa'
SECRET = 'XDZIO4hvbfthoIQhunHtXSg2d'

# 地区节点 可选地区取决于你的阿里云帐号等级,普通用户只有四个,分别是杭州、上海、深圳、河北,具体参考官网API
regionId = 'cn-hangzhou'

# 配置认证信息
client = AcsClient(ID, SECRET, regionId)

# 设置主域名
DomainName = 'ranjuan.cn'

# 子域名列表  列表参数可根据实际需求增加或减少值
#SubDomainList = ['a', 'b', 'c']
SubDomainList = ['testdns']

# 获取外网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
    '''
    newip = json.load(urllib.request.urlopen('https://api.ipify.org/?format=json'))['ip']
    return newip

# 查询记录
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() + ')')

四、实现定期更新

上面代码每运行一次,则会更新一次,所以建议将其加入定日任务执行,或者优化下代码,循环检测外网ip情况,如果发生变化则立即进行更新(linux下使用 crontab 命令创建定时,windows下使用“任务计划程序”在开始菜单直接搜索该名字即可打开)!

下面是网上找到的另一个版本的利用python实现阿里云ddns的源码,有需要的也可以参考下:

import datetime,json
from urllib.request import urlopen
from aliyunsdkcore.client import AcsClient
from aliyunsdkalidns.request.v20150109.DescribeDomainRecordsRequest import DescribeDomainRecordsRequest
from aliyunsdkalidns.request.v20150109.UpdateDomainRecordRequest import UpdateDomainRecordRequest
"""
# 安装依赖
pip3 install pyyaml
pip3 install aliyun-python-sdk-core-v3
pip3 install aliyun-python-sdk-alidns==2.0.6
# 添加记录 RR_list
RR_list =  ['...','...']
"""
# 域名信息
RR_list = ['www','web']
DomainType = 'A'
region_id = "cn-shenzhen"
DomainName = 'baidu.com'
UpdateDomain = 'Auto_Lines'
# 当前时间
data_new = str(datetime.datetime.now())
# API 获取 最新IP
newip = json.load(urlopen('https://api.ipify.org/?format=json'))['ip']
# 阿里云 权限 密钥
AccessKey_ID = ''
Access_Key_Secret = '' 

for RR in RR_list:
    # 生成 阿里云 API 客户端
    def AliAccessKey(id,Secret,region):
            client = AcsClient(id, Secret, region)
            return client
    # 验证 RR 子域名 存在与否
    def GetDNSRecordId(client,DomainName):
            request = DescribeDomainRecordsRequest()
            request.set_accept_format('json')
            request.set_DomainName(DomainName)
            response = client.do_action_with_exception(request)
            json_data = json.loads(str(response, encoding='utf-8'))
            for RecordId in json_data['DomainRecords']['Record']:
                if RR == RecordId['RR']:
                    return RecordId['RecordId']
    # 更新 阿里云DNS 解析
    def UpdateDomainRecord(client,RecordId):
        try:
            request = UpdateDomainRecordRequest()
            request.set_accept_format('json')
            request.set_Value(newip)
            request.set_Type(DomainType)
            request.set_RR(RR)
            request.set_RecordId(RecordId)
            client.do_action_with_exception(request)
            print("域名:" + DomainName + " 主机:" + RR + " 记录类型:" +  DomainType + " 记录值:" +  newip)
        except Exception as e:
            print('结果:' + RR + '.infunvip.cn 解析一致 / 时间 : ' + data_new)
    # 运行 主程序
    def main():
        client = AliAccessKey(AccessKey_ID,Access_Key_Secret,region_id)
        RecordId = GetDNSRecordId(client,DomainName)
        UpdateDomainRecord(client,RecordId)
    if __name__ == "__main__":
        main()

基于互联网精神,在注明出处的前提下本站文章可自由转载!

本文链接:https://ranjuan.cn/python-aliyun-ddns/

赞赏

微信赞赏支付宝赞赏

发表评论