使用python3监测公网IP,实现DDNS

由于家庭宽带基本都是动态IP,每当你重启一次光猫,IP地址就会变化一次。当光猫因为停电、故障、维护等原因重启过后,网站就无法访问了。网上基本的解决方法是使用花生壳做DDNS(动态域名解析),但那个需要绑定自己的域名要付费不说,且linux无法使用。于是学了一波python和socket,成功实现了DDNS。

这个程序是一个典型的C/S程序,有客户端和服务端。画张草图来解释下这个python程序的运行原理:

image.png

说明:这个程序分为客户端和服务端。客户端运行在内网主机上,服务端运行在云服务器上。客户端会每隔30秒主动检测一次自己的公网IP是否变化,如果地址改变,则会通过socket连接服务端程序,把改变后的地址发送给服务端。服务端接收到地址后,先校验一遍地址是否合法,如果地址OK,那么将地址写入Nginx的反代理配置文件,反代理生效并指向新的内网主机公网地址,从而实现DDNS。

一、程序的使用环境

服务器系统需是centos7,内网主机可以是windows或者centos7。

注意:程序是使用python3开发的,不兼容python2。因此如果使用centos7默认自带的python2运行会报错。

1. 服务端环境:python3、retry模块。

python3环境搭建请看:《centos下安装python3并与自带的python2共存

搭建完成后,使用pip安装retry模块:

pip install retry

2. 客户端环境:python3、beautifulsoup4模块、retry模块

这里演示一下windows怎么安装python3和模块。

windows版python3下载地址:https://www.python.org/ftp/python/3.6.5/python-3.6.5-amd64.exe

直接打开安装,记得把下图箭头的地方勾上:

image.png

装完了win+R运行cmd,打开命令提示符窗口,输入python -V。有如下回显表示成功:

image.png

然后我们更新一下pip版本到10.0.1,默认的是9.0.3:

python -m pip install --upgrade pip

安装retry和beautifulsoup4模块

pip install retry
pip install beautifulsoup4

image.png

image.png

二、程序运行

1.服务端

服务端运行在云服务器地址上,使用以下命令即可下载运行。

wget https://www.xiaoweigod.com/shell/ddns_server.py 
python3 ddns_server.py

如图,程序运行后输入上一篇文章你绑定的域名和反代理的端口号,程序会实时监听。

image.png

2. 客户端

客户端以windows为例:

程序下载地址:https://www.xiaoweigod.com/shell/ddns_client.py

将下载好的程序放到D盘根目录,然后打开cmd执行以下命令:

d:
python ddns_client.py

如图,程序运行后输入服务端的地址,即可开始工作:

image.png

这样我们就实现了ddns了。当运行在内网主机上的客户端检测到自己的公网IP变更,则会主动把变更后的IP发送给云服务器上的服务端,服务端检测无误后,将地址写入nginx的配置文件。无论家庭的公网IP怎么变化,都可以正确指向。

运行截图(中间重启了光猫):

image.png

我们通过socket随便发个不是IP地址的文本给服务端:

image.png

服务端只会接受正确的IP地址格式,并写入配置文件,防止出错。

三、源码

1. server端

#coding=utf-8
#author:xiaowei
#blog:https://www.xiaoweigod.com/

import time
import socket
from retry import retry
import os

#os.system('clear')
server_name=str(input('输入你的网址:'))
port=int(input('输入反代理端口号(1-65535):'))
if port > 65535:
    print('端口不合法!')
    exit()
elif port <1:
    print('端口不合法!')
    exit()

@retry()
def recv_ip():
    sk=socket.socket(socket.AF_INET,socket.SOCK_STREAM,socket.IPPROTO_TCP)
    addr=('0.0.0.0',8888)
    sk.bind(addr)
    sk.listen(1)
    conn,addr=sk.accept()
    global new_ip
    new_ip=conn.recv(30).decode()
    #sk.sendto(send_data,addr)
    sk.close()
    conn.close()
#    if len(new_ip)>15:
#        print('地址不合法!')
#    elif len(new_ip)<8;
#        print('玩蛇皮')
#    else:
#        print("客户机公网地址变更:",new_ip)

def write_to():
    proxy='server{\nlisten 80;\nserver_name aaa.xiaoweigod.com;\nlocation / {\n    proxy_pass http://'+new_ip+':'+str(port)+';\n} \n access_log off;\n   }'
    os.remove('/usr/local/nginx/conf/vhost/'+server_name+'.conf')
    f=open('/usr/local/nginx/conf/vhost/'+server_name+'.conf','w',encoding='utf-8')
    f.write(proxy)
    f.close()
    print("----写入配置文件成功----")
    os.system('service nginx restart')
    print("已生效,当前反代理规则为:"+new_ip+':'+str(port))
    print('\n------继续监听ing..------')

print("开始监听客户机")
recv_ip()
write_to()

i=0
while i==0:
    time.sleep(5)
    recv_ip()
    if new_ip == None:
        print('地址不能为空!')
        continue
    elif new_ip==[]:
        print('禁止元素!')
        continue
    elif len(new_ip) >15:
        print('地址过长!')
        continue
    elif len(new_ip)<8:
        print('非法地址!')
        continue
    else:
        write_to()

2. client端

#coding=utf-8
#author:xiaowei
#blog:https://www.xiaoweigod.com/

import urllib.request
from bs4 import BeautifulSoup
import socket
import time
from urllib.error import URLError
from retry import retry

ip_addr=str(input('输入服务端IP:'))
ip_port=8888
#ip_port=int(input('输入服务端端口(1-65535):'))
#if ip_port > 65535:
#    print('端口不合法!')
#    exit()
#elif ip_port <1:
#    print('端口不合法!')
#    exit()

@retry()
def get_ip():
    url = 'http://www.net.cn/static/customercare/yourip.asp'
    req = urllib.request.Request(url)
    rsp=urllib.request.urlopen(req)
    html=rsp.read().decode('utf-8',"ignore")
    html=BeautifulSoup(html,'html.parser')
    iph2=html.h2
    global ip
    ip=iph2.get_text()
    #print("你的公网IP是:",ip)

@retry()
def send_ip():
#    ip_addr='alish02.ssrcn.me'
    addr=(ip_addr,ip_port)
    sk=socket.socket(socket.AF_INET,socket.SOCK_STREAM,socket.IPPROTO_TCP)
    sk.connect(addr)
    sk.send(ip.encode())
    sk.close()

print("\n----开始侦测本机公网IP地址----")
get_ip()
send_ip()

i=0
while i == 0:
    get_ip()
    print ("当前公网IP:",ip)
    tmp1_ip=ip
#    print("tmp1_ip:",tmp1_ip)
    print("------休息30秒------")
    time.sleep(30)
    get_ip()
    tmp2_ip=ip
#   print("tmp2_ip:",tmp2_ip)
    if tmp1_ip == tmp2_ip:
        print("########OJBK,地址没变!########")
    else:
        ip=tmp2_ip
        print("公网地址改变:",ip)
        send_ip()
        print("同步到远程服务器成功!")
        print("\n########继续检查########")
点赞

发表回复