一、需要一個對外的公網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
