Ubuntu技巧-Ubuntu远程访问之电信公网IP
创作时间:
作者:
@小白创作中心
Ubuntu技巧-Ubuntu远程访问之电信公网IP
引用
CSDN
1.
https://blog.csdn.net/koffuxu/article/details/141399290
本文将详细介绍如何在Ubuntu系统上通过电信公网IP实现远程访问家庭服务器。主要内容包括获得公网IP、设置动态DNS以及编写Python脚本实现自动更新DNS记录等技术细节。
框架
先看一下大概的框架,主要的思路就是自己在家庭路由器上拨号获得外网的IP,然后与服务器的端口绑定,就能通过外网访问家庭服务器了。
获得运营商的公网IP
以电信为例:
- 与10000客服沟通:电信光猫改成桥接模式,在路由器端重新拨号;
- 与10000客服沟通:把外网IP设置为公网IP(他们一般不会同意,说没有了。强烈要求,不给投诉,就可以免费一年)
- 重新在路由器上拨号就看得到公网IP(如果判断是不是公网IP,在https://www.ip138.com/查一下本机IP和外网IP是同一个,此时的IP就是公网IP)。同时也可以通过
curl ifconfig.me
获得外网IP。 - 在路由器设置端口转发
到此,已经获得一个公网IP,并与ubuntu服务器做了端口映射。
域名绑定
- 购买域名,在阿里云购买一个top域名,大概22元/年。
- 阿里云域名映射:添加一条A记录
此时获得虽然是公网IP,但这个IP是动态,我测试下来大概一周变更一次,就需要写一个脚本,定时更新这个阿里云域名DNS服务,这个下再再讲。 - 测试:
ssh -p 26919 koffu@xxxxx.top
动态域名设置
之前提到,运营商给发布的IP虽然是公网的,但是动态的,我测试下来大概一周变更一次。所以,接下来我们要设置一个自动化任务,每天固定时间点去获取公网IP,然后通过软件接口去更新阿里云的域名解析的A记录。那么无论你什么时候访问这个域名,都是解析到更新的正常的IP。
先在阿里云的RAM访问控制,创建一个用户。
接下来完成账号验证,就创建成功了,AccessKey ID和Secret生成了,后面需要用到。点击用户名,进行权限管理,添加DNS解析权限
自动更新的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。如果有遇到什么问题,欢迎留言讨论
热门推荐
节食减肥的“尽头”可能是内分泌科? 这些误区你踩没踩
管理怀孕期间的腰背痛,中医(针灸和推拿)的应用
如何选择合适的电源滤波器以减少电源中的纹波和噪声?
民国军阀汤玉麟:外号汤二虎,土匪出身,曾威震一方
酸汤福鼎肉片
有脚气是不是缺乏维生素补充
揭秘周杰伦艺术跨界之旅,《艺起“杰”作》画册发布
冲调奶粉,新手爸妈必学的实验级别操作
WiFi-7:下一代无线网络的全面解析与未来展望
烫伤恢复奇痒难耐?医生教你如何缓解!
视频转换后手机无法播放?这样设置视频编码器就能解决
历经终南山一夜,小龙女为何对尹志平念念不忘?长大后才赫然醒悟
机器学习的数学基础:向量与矩阵详解
长假期间员工的合法解雇问题探析
百家姓之24—张姓,起源·迁徙·家训·名人故事
如何计算住院医保费用?住院医保的报销标准是怎样的?
重庆以艺术创新擦亮“全球设计之都”名片
HTML5 泡泡游戏开发实战:从零开始制作一个有趣的网页小游戏
智能医疗,构建高效便捷的在线预约挂号平台
中字头个股全线大跌,中报数据揭示盈利质量隐忧
国际商标检索:全球品牌保护的核心工具
蚕宝宝变化规律有哪些?蚕宝宝蜕变记:四个阶段教你了解蚕生长!
如何高效实现安卓手机备份,确保重要数据?
金铲铲「恭喜发财」开局五费怎么玩?最全的出装及过渡阵容推荐,爽追三星五费!
两种实用方法,轻松检验镜头质量
一口气搞懂古代爵位制度!公、侯、伯、子、男都是啥?
醉酒就诊流程规范化标准怎么制定?
合资B级车钢材强度对比,同样是中级车,A柱强度能差一倍?
购买私人NAS时,如何选择合适的文件系统
中国史阶段特征概述以及重点历史名词解释