一、需要一个对外的公网IP
先查看路由器的对外IP 是否是公网IP,如果不是,可以致电宽带运营商,要求分配公网IP。对于普通用户,宽带运营商分配的公网IP是会变化的,每次启路由器,或者间隔一定时间,IP 都会变化一次。不过,这个问题可以解决。
二、 将树莓派的IP 设置为静态IP
这里假设家里的所有上网设备都是通过路由器连接上网。路由器自身的IP是公网的IP,连接路由器的各个设备,分配的都是私有IP。如果树莓派的IP 不是静态的,那么每次重启路由器,路由器的IP 都是会变的,这样不利于实现接下来要说的路由器端口转发功能。
可以通过修改树莓派的配置文件,实现静态IP 的分配。/ect/dhcpcd.conf 文件有静态IP设置的example。
也可以通过修改路由器的配置选项,实现静态IP的分配。登录路由器管理页面,在左侧找到DHCP服务器--静态地址分配,点击添加新条目输入要信息。
我的树莓派是通过自带WIFI连接路由器的,所以MAC地址填的是无线网卡的地址,IP地址填的是为树莓派分配的静态IP.
三、路由器端口映射
1. 拥有公网IP的是路由器,要实现外网访问路由器局域网内的树莓派,需要路由器做转发的处理。
2. 登录路由器管理界面,在左侧找到转发规则--虚拟服务器,按添加新条目添加转发规则。
如上图,添加的是SSH 的转发规则。在远程用putty工具登录树莓派时,Host name填的是路由器的IP(最好是域名),port填的是上图自己填入的服务端口号(7**7)。这样就可以在外网远程登录树莓派了。
四、动态域名解析
1. 为路由器的IP绑定一个域名。如果还没域名,先购买一个。我的域名是在腾讯云上购买的,域名解析也是利用dnspod上提供的接口。
2. 在树莓派上定期执行动态域名解析的任务。基本算法如下:
i. 获取当前主机对外的公网IP(路由器的IP)hostIP;
ii. 获取当前路由器域名的IP,domainIP;
iii. 如果domainIP 不等于hostIP,说明运营商更新了分配的IP。通过dnspod提供的API,提交最新的IP.
ddns脚本根据开源项目(https://gist.github.com/chuangbo/833369)修改而来
#!/usr/bin/env python
#-*- coding:utf-8 -*-
import httplib, urllib
import socket
import time
import datetime
params = dict(
login_email="your email addr", # replace with your email
login_password="password", # replace with your password
format="json",
domain_id=, # replace with your domain_od, can get it by API Domain.List
record_id=, # replace with your record_id, can get it by API Record.List
sub_domain="www", # replace with your sub_domain
record_line="默认",
)
domain="" #your domain
def ddns(ip):
params.update(dict(value=ip))
headers = {"Content-type": "application/x-www-form-urlencoded", "Accept": "text/json"}
conn = httplib.HTTPSConnection("dnsapi.cn")
conn.request("POST", "/Record.Ddns", urllib.urlencode(params), headers)
response = conn.getresponse()
#print response.status, response.reason
data = response.read()
#print data
conn.close()
return response.status == 200
def get_host_ip():
sock = socket.create_connection(('ns1.dnspod.net', 6666))
sock.settimeout(10)
ip = sock.recv(16)
sock.close()
return ip
def local_log(file_name,text):
fo = open(file_name,"a")
strTime = time.asctime(time.localtime(time.time()))
#print strTime
fo.write(strTime+" ip:"+text)
fo.close()
if __name__ == '__main__':
host_ip = get_host_ip()
domain_ip = socket.gethostbyname(domain)
if host_ip != domain_ip :
ddns(host_ip)
#print "current host ip:",host_ip
#print "domain ip:",domain_ip
#print "domain:",domain
来源:oschina
链接:https://my.oschina.net/u/4403736/blog/4200896