域名繫結動態IP實戰案例

大雄45發表於2022-03-02
導讀 一般家庭網路的公網IP都是不固定的,而我又想通過域名來訪問自己伺服器上的應用,也就是說:需要通過將域名繫結到動態IP上來實現這個需求。

域名繫結動態IP實戰案例域名繫結動態IP實戰案例

一般家庭網路的公網IP都是不固定的,而我又想通過域名來訪問自己伺服器上的應用,也就是說:需要通過將域名繫結到動態IP上來實現這個需求。於是乎,我開始探索實現的技術方案。

通過在網上查閱一系列的資料後,發現阿里雲可以做到實現動態域名解析DDNS。於是乎,一頓操作下來,我實現了域名繫結動態IP。這裡,我們以Python為例實現。

好了,說幹就幹,我們開始吧,走起~~

阿里雲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, ,curl -L tool.lu/ip
        with urllib.request.urlopen(') 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地址。

    本文原創地址:

    來自 “ ITPUB部落格 ” ,連結:http://blog.itpub.net/69955379/viewspace-2861784/,如需轉載,請註明出處,否則將追究法律責任。

    相關文章