用树莓派搭建外网可以访问的服务器

£可爱£侵袭症+ 提交于 2020-04-26 13:55:30

一、需要一个对外的公网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
易学教程内所有资源均来自网络或用户发布的内容,如有违反法律规定的内容欢迎反馈
该文章没有解决你所遇到的问题?点击提问,说说你的问题,让更多的人一起探讨吧!