用树莓派搭建外网可以访问的服务器
一、需要一个对外的公网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