type
status
date
slug
summary
tags
category
icon
password
Tweet Link
Staus
类型
平台
💡
大家好,我是可夫小子,《小白玩转ChatGPT》专栏作者,关注AIGC、互联网和自媒体。
前面文章介绍了家庭服务器接入外网的三种方式的第一种,今天介绍第二种,即通过获得电脑公网IP,然后再设置动态DNS保持更新。这种方案的成功率与你所使用网络运营商相关,我了解有些地方运营商并不同意给你分配公网IP。
notion image

框架

先看一下大概的框架,主要的思路就是自己在家庭路由器上拨号获得外网的IP,然后与服务器的端口绑定,就能通过外网访问家庭服务器了。
notion image

获得运营商的公网IP

我家里是使用电脑的网络,我就以电信为例
  1. 与10000客服沟通:电信光猫改成桥接模式,在路由器端重新拨号;
  1. 与10000客服沟通:把外网IP设置为公网IP(他们一般不会同意,说没有了。强烈要求,不给投诉,就可以免费一年)
  1. 重新在路由器上拨号就看得到公网IP(如果判断是不是公网IP,在https://www.ip138.com/查一下本机IP和外网IP是同一个,此时的IP就是公网IP)。同时也可以通过curl ifconfig.me获得外网IP。
  1. 在路由器设置端口转发
    1. notion image
到此,已经获得一个公网IP,并与ubuntu服务器做了端口映射。
 

域名绑定

  1. 购买域名,在阿里云购买一个top域名,大概22元/年。
  1. 阿里云域名映射:添加一条A记录
    1. 此时获得虽然是公网IP,但这个IP是动态,我测试下来大概一周变更一次,就需要写一个脚本,定时更新这个阿里云域名DNS服务,这个下再再讲。
  1. 测试:ssh -p 26919 koffu@xxxxx.top
    1. notion image
 

动态域名设置

之前提到,运营商给发布的IP虽然是公网的,但是动态的,我测试下来大概一周变更一次。所以,接下来我们要设置一个自动化任务,每天固定时间点去获取公网IP,然后通过软件接口去更新阿里云的域名解析的A记录。那么无论你什么时候访问这个域名,都是解析到更新的正常的IP。
 
先在阿里云的RAM访问控制,创建一个用户。
notion image
notion image
notion image
接下来完成账号验证,就创建成功了,AccessKey ID和Secret生成了,后面需要用到。
notion image
点击用户名,进行权限管理,添加DNS解析权限
notion image
 
 
自动更新的python代码如下,自行替换自己的ID和SECRET
#!/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 datetime import datetime # 加载获取 、 新增、 更新、 删除接口 from aliyunsdkalidns.request.v20150109 import DescribeSubDomainRecordsRequest, AddDomainRecordRequest, UpdateDomainRecordRequest, DeleteDomainRecordRequest # 加载内置模块 import json,urllib # AccessKey 和 Secret 建议使用 RAM 子账户的 KEY 和 SECRET 增加安全性 ID = '' SECRET = '' # 地区节点 可选地区取决于你的阿里云帐号等级,普通用户只有四个,分别是杭州、上海、深圳、河北,具体参考官网API regionId = 'cn-hangzhou' # 配置认证信息 client = AcsClient(ID, SECRET, regionId) # 设置主域名 DomainName = '' # 子域名列表 列表参数可根据实际需求增加或减少值 SubDomainList = ['tl'] # 获取外网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('https://ifconfig.me/') 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('tl.agihub.top') 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,subname,domainname): info = getDomainInfo(subname + '.' + domainname) if info['TotalCount'] == 0: print('准备添加新记录') add_result = addDomainRecord(client,value,subname,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"] print ('当前IP:' + cur_ip + '; 原IP:' + old_ip ) if cur_ip == old_ip: print ("新ip与原ip相同,不更新!") else: update_result = updateDomainRecord(client,value,subname,record_id) print('更新成功,返回信息:') print(update_result) else: # 正常不应该有多条相同的记录,如果存在这种情况,应该手动去网站检查核实是否有操作失误 print("存在多个相同子域名解析记录值,请核查删除后再操作!") # 获取当前IP IP = getIp() # 获取当前时间 current_time = datetime.now().strftime("%Y-%m-%d %H:%M:%S") print('===ready update DNS:' + current_time + '===') # 循环子域名列表进行批量操作 for x in SubDomainList: setDomainRecord(client,IP,x,DomainName)
 
 

小结

到此,本文就详细介绍了如何从运营商获取公网IP,并绑定一个域名,同时在服务上跑一个定时任务,运行脚本更新IP和域名的映射关系,实现DDNS。如果有遇到什么问题,欢迎留言讨论
 
 
参考:
Ubuntu远程访问3——通过SSH反向代理 Ubuntu远程访问1——Teamviewer安装与设置

可夫小子
可夫小子
关注AIGC、读书和自媒体
公告
type
status
date
slug
summary
tags
category
icon
password
Tweet Link
Staus
类型
平台
ChatGPT、Midjourney、AIGC、人工智能、程序员、
Notion、读书、写作、个人成长、
独立开发者、自媒体、个人副业、技术掘金。
请关注我的知识星球:
notion image