用python写一个脚本,自动连wifi,自动登录校园网

原来在本科期间买的老华硕电脑,最近开始无缘无故的黑屏、死机,让我开始有了换电脑的念头,早都想试一试苹果的系统了,所以趁着这次618活动来临,也是狠下手笔,入手了人生第一台MacBook-Air。在适应了一天之后,基本上使用起来没什么障碍了,肯定还有很多功能是我没发现的,以后在慢慢探索了。期间我也遇到了一个令人烦恼的事情,就是每次连学校的校园网,都要弹窗、登录,很繁琐,就想着试试看,自己能不能也写一个脚本来实现自动连接校园网的功能。第一次搞这玩意儿,也是遇到各种问题,写帖记录一下。

1.实现原理

刚好最近学了http协议,就当是复习了。简单概括就是,通过网址找到登录界面,然后发送post请求,把登录信息提交给服务器,从而完成登录。图片来源于朋友博客的,他的更详细介绍了http协议,感兴趣可以看看。

1.1认识 URL

我们所说的网址,其实就是统一资源定位符(uniform resource locator简称URL),通过这个唯一的地址,可以找到对应的服务。它的标准格式如下:

协议://用户名:密码@子域名.域名.顶级域名:端口号/目录/文件名.文件后缀?参数=值#标志

在这里插入图片描述

这个只是标准的格式,有些信息是可以省略的,比如登录信息等,还有服务器地址可以用域名地址,也可以用ip地址。带层次的文件路径其实就是你要访问的服务器资源,问号?后面是get请求的参数。http协议有多种请求方法,post和get只是其中的两种。

1.get方法主要是获取服务器的资源信息,请求的参数一般放在url?后面。
2.post方法主要是把数据提交给服务器,在报文的正文部分进行提交。

http协议本质是获得某种“资源”(视频、音频、网页、图片……),而传输则是其功能。实际上,上网的大部分行为,都在进行着进程间通信,既然是通信,就需要获取信息和发送信息,所以对应到我们生活中,大部分的上网行为无非两种:
1.把服务器上面的资源拿到本地(下载短视频、网络小说……)
2.把本地的服务器推送到服务器(搜索、登录、下单……)

1.2 http请求报文格式

在这里插入图片描述


首行: [方法] + [url] + [版本]
Header: 请求的属性,冒号分割的键值对;每组属性之间使用\n分隔;遇到空行表示Header部分结束
Body: 空行后面的内容都是Body. Body允许为空字符串. 如果Body存在,则在Header中会有一个 Content-Length属性来标识Body的长度;

1.3 http响应报头格式

在这里插入图片描述


首行: [版本号] + [状态码] + [状态码解释]
Header: 请求的属性,则在Header中会有一个 Content-Length属性来标识Body的长度; 如果服务器返回了一个html页面,那么html页面内容就是在 body中。

HTTP常见Header:
Content-Type: 数据类型(text/html等) Content-Length: Body的长度
Host: 客户端告知服务器,所请求的资源是在哪个主机的哪个端口上; User-Agent: 声明用户的操作系统和浏览器版本信息;
referer: 当前页面是从哪个页面跳转过来的;
location: 搭配3xx状态码使用,告诉客户端接下来要去哪里访问;
Cookie: 用于在客户端存储少量信息. 通常用于实现会话(session)的功能;

2.具体实现

import requests
import socket

# 获取ip地址
def get_host_ip():
    """
    查询本机ip地址
    :return: ip
    """
    try:
        s = socket.socket(socket.AF_INET, socket.SOCK_DGRAM)
        s.connect(('10.255.255.255', 1))
        ip = s.getsockname()[0]
    finally:
        s.close()
 
    return ip
user_ip = get_host_ip()

# 校园网地址,最好不要用浏览器里的url,还是建议抓包获取
post_addr = "http://10.10.244.11:801/eportal/"

#下面两个大括号里面都是复制自己学校校园网登录网站中的,冒号两边都要加上双引号
post_header = {
#报头信息,通过抓包,获取
}
 
post_data = {
 #正文数据,通过抓包获取
}
 #提交http请求报文
z = requests.post(post_addr, data=post_data, headers=post_header)

print("登录校园网成功,局域网ip如下:")
print(user_ip)
#input("")

上面是代码的主要逻辑,细节信息还需要抓包填充。一开始电脑上是没有安装requests包的,需要自己先安装一下,后面python需要导入的包都是用pip3命令安装。如果没安装pip3命令的,请自行安装。

pip3 install requests

2.1 获取url

在谷歌浏览器先打开上网登录窗口,然后按F12键进入开发者模式,勾选保留日志,输入账号密码,进行登录,在网络那里获取登录时的http请求报文。

在这里插入图片描述

然后查看抓到的包,查看第一个即可,一般是第一个,如果不放心可以点进区查看,看到标头里的请求方法,确保是post。然后里面还有一个请求网址,就是url了。只需要复制?问号前面的内容即可,后面的是一些get方法的请求参数,不明白什么意思的看长文url的解释。往下拉,还有响应标头,请求标头等信息,⚠️注意,因为我们要向服务器请求登录,所以我们需要的是请求标头,而不是响应,别搞错了。

在这里插入图片描述

# 校园网地址,最好不要用浏览器里的url,还是建议抓包获取
post_addr = "http://10.10.244.11:801/eportal/"

这样就完成了第一步,获取到了校园网地址。为什么说不建议直接从浏览器里面复制呢,比如我们学校这种情况返回的响应是3xx,说明网址被重定向过了,所以抓包到的地址比较准确一些。

2.2 获取请求报文的报头

在这里插入图片描述

把请求标头里的内容填充到代码块里,部分header的含义上文已经解释过,还想了解更多请自行搜索。填充的格式是键值 key:values模式,key和values都是字符串需要加引号,上下键值用逗号隔开,下面是我自己的报文,只是个例子。

#下面两个大括号里面都是复制自己学校校园网登录网站中的,冒号两边都要加上双引号
post_header = {
    'Accept': '*/*',
    'Accept-Encoding': 'gzip,deflate',
    'Accept-Language': 'zh-CN,zh;q=0.9',
    'Cache-Control': 'max-age=0',
    'Connection': 'keep-alive',
    'Host': '10.10.244.11',
    'Referer': 'http://10.10.244.11/',
    'Upgrade-Insecure-Requests': '1',
    'User-Agent': 'Mozilla/5.0 (Macintosh; Intel Mac OS X 10_15_7) AppleWebKit/537.36 (KHTML,like Gecko) Chrome/101.0.4951.64 Safari/537.36',
}

2.3 获取请求报文的数据

在这里插入图片描述


在这里插入图片描述

把载荷里的查询字符串、表单数据都填充到程序块中,这里主要上传的就是你的登录信息,不要填错了。

post_data = {
    'c': 'ACSetting',
    'a': 'Login',
    'DDDDD': 'xxxx',
    'upass': 'xxxxx',
    'protocol': 'http:',
    'hostname': '10.10.244.11',
    'iTermType': '1',
    'wlanuserip': user_ip,
    'wlanacip': 'xxxxxx',
    'wlanacname': 'SPL-BRAS-SR8806-X',
    'mac': '00-00-00-00-00-00',
    'ip': user_ip,
    'enAdvert': '0',
    'queryACIP': '0',
    'loginMethod': '1'
}

2.4 获取本机的局域网ip

为什么要单独写一个函数获取主机IP呢,因为IP地址分为固定IP地址和动态IP地址,我们需要获取的是动态的IP地址,它是一直变化的,不能直接在请求数据里填抓包拿到的地址,不然你换个地方,可能那个地址就失效了。

固定IP:固定IP地址是长期固定分配给一台计算机使用的IP地址,一般是特殊的服务器才拥有固定IP地址。
动态IP:因为IP地址资源非常短缺,通过电话拨号上网或普通宽带上网用户一般不具备固定IP地址,而是由ISP动态分配暂时的一个IP地址,这些都是计算机系统自动完成的。

# 获取ip地址
#需要导入socket包,系统应该自带
def get_host_ip():
    """
    查询本机ip地址
    :return: ip
    """
    try:
        s = socket.socket(socket.AF_INET, 1))
        ip = s.getsockname()[0]
    finally:
        s.close()
 
    return ip
user_ip = get_host_ip()

写到这里其实,已经可以自动连接校园网了,但是前提是你先打开Wi-Fi,连到学校的校园网Wi-Fi。说到底,现在的功能只能帮助你登录校园网,连接校园网Wi-Fi的事情还是得你来做,如果你之前连的是其他网络,那么你还有进行网络的切换。所以还要再增加一个自动连接Wi-Fi的功能,刚好python里有一个pywifi包可以支持这个功能。但是!!直接pip3 安装的pywifi包里面不支持mac os的Wi-Fi控制,因为开发这个pywifi包的作者不用mac os系统,所以pywifi包只支持windows和linux。好在,后来有人提出这个问题,作者后来又写了一个适合mac os的包,不过需要自己下载。我也是经历很多波折,才解决了这个问题。

3.自动连接Wi-Fi

还是先在终端安装pywifi包,然后找到pywifi包对应位置,把内容全部替换成支持moc os 的pywifi包。

pip3 install pywifi

如果找不到pywifi路径可以先执行卸载命令,然后就会弹出所以安装过的包路径了,然后复制所需的路径,最好选择n命令,停止卸载就行。

在这里插入图片描述


得到安装路径以后,可以在终端里查看,也可以在mac可视化文件模式里查看,我更喜欢可视化,打开的时候有的文件夹就翻译成中文了,我相信只要用心肯定能找到。

在这里插入图片描述

在这里插入图片描述

在这里插入图片描述


在这里插入图片描述


找到pywifi路径之后,就要下载支持mac os的pywifi包了,下载完进行替换就行。那这个支持mac os的pywifi在哪呢?这里给出作者github的地址,作者awkman在Issue24里面也回答了,他写了一个兼容Macos的demo程序。

在这里插入图片描述


moc版pywifi
作者回复

可以在终端用git命令下载,也可以,直接到作者仓库取自己下载,大家随意。git命令下载指令如下:-b 后面带的是分支,作者放在macos_dev里了。

git clone -b macos_dev https://github.com/awkman/pywifi.git

在这里插入图片描述


下载完检查一下是不是包含了mac的.py文件,包含了就没问题。然后把包含了mac的这个pywifi文件和之前的pywifi进行替换就行。先cd到当前文件夹,然后cp拷贝到原来路径(怎么找路径前文已经说了),文件名相同会自动替换里面内容。

cd pywifi
cp -r pywifi /Users/wenanqin/Library/Python/3.8/lib/python/site-packages

之前我在这样做完,运行还是报错,因为发现_wifiutil_macos.py里有一个包没安装,装完就好了。

pip3 install pyobjc

在这里插入图片描述


下面开始完成连接wifi功能的代码,在统一路径下,新建一个wifi.py文件。

import pywifi
import time
#保存包中写义的常量
from pywifi import const

def wifi_connect_status():
    """
    判断本机是否有无线网卡,以及连接状态
    :return: 已连接或存在无线网卡返回1,否则返回0
    """
    #创建一个元线对象
    wifi = pywifi.PyWiFi()

    #取当前机器,第一个元线网卡
    iface = wifi.interfaces()[0] #有可能有多个无线网卡,所以要指定
   

    #判断是否连接成功
    if iface.status() in [const.IFACE_CONNECTED,const.IFACE_INACTIVE]:
        #print('wifi已经连接了网络')
        return 1
    else:
        print("兄弟,我没设置自动打开Wi-Fi功能,你先打开wifi再试?")
    return 0

def scan_wifi():
    """
    扫描附件wifi
    :return: 扫描结果对象
    """
    #扫描附件wifi
    wifi = pywifi.PyWiFi()
    iface = wifi.interfaces()[0]

    iface.scan() #扫描附件wifi
    time.sleep(1)
    basewifi = iface.scan_results()
    # for i in basewifi:
    #     print('wifi扫描结果:{}'.format(i.ssid)) # ssid 为wifi名称
    #     print('wifi设备MAC地址:{}'.format(i.bssid))
    return basewifi

def connect_wifi():
    wifi = pywifi.PyWiFi()  # 创建一个wifi对象
    ifaces = wifi.interfaces()[0]  # 取第一个无限网卡
    #print("本机无线网卡名称:")
    #print(ifaces.name())  # 输出无线网卡名称
    ifaces.disconnect()  # 断开网卡连接
    time.sleep(3)  # 缓冲3秒


    profile = pywifi.Profile()  # 配置文件
    profile.ssid = "NJUPT-CMCC"  # wifi名称
    #连校园网不需要密码登录,另有登录模块
    # profile.auth = const.AUTH_ALG_OPEN  # 需要密码
    # profile.akm.append(const.AKM_TYPE_WPA2PSK)  # 加密类型
    # profile.cipher = const.CIPHER_TYPE_CCMP  # 加密单元
    # profile.key = '4000103000' #wifi密码

    ifaces.remove_all_network_profiles()  # 删除其他配置文件
    tmp_profile = ifaces.add_network_profile(profile)  # 加载配置文件

    ifaces.connect(tmp_profile)  # 连接
    time.sleep(1)  # 尝试10秒能否成功连接
    isok = True
    if ifaces.status() == const.IFACE_CONNECTED:
        print("连接校园网成功")
    else:
        print("连接校园网失败")
    #ifaces.disconnect()  # 断开连接
    time.sleep(1)
    return isok

这里有三个功能,前两个测试用的,实际可以只调用第三个。link.py登录校园网之前先调用连接wifi模块。

import requests
import socket
#导入刚才写的wifi模块,一定放在同一文件夹内
import wifi

#查看wifi状态
wifi.wifi_connect_status()
#连接wifi
wifi.connect_wifi()

# 获取ip地址
def get_host_ip():
    """
    查询本机ip地址
    :return: ip
    """
    try:
        s = socket.socket(socket.AF_INET, 1))
        ip = s.getsockname()[0]
    finally:
        s.close()
 
    return ip
user_ip = get_host_ip()


# 校园网地址
post_addr = "http://10.10.244.11:801/eportal/"

#下面两个大括号里面都是复制自己学校校园网登录网站中的,冒号两边都要加上双引号
post_header = {
    'Accept': '*/*',
}
 
post_data = {
    'c': 'ACSetting',
    'DDDDD': ',xxxxxxx@cmcc',
    'wlanacip': 'xxxxxxx',
    'loginMethod': '1'
}
 
z = requests.post(post_addr, headers=post_header)
#如果不想每次都手动关闭窗口可以删除下面的input,然后将print里的内容改成自己想要的
print("登录校园网成功,局域网ip如下:")
print(user_ip)
#input("")

4.打包成exe文件

1.先安装pyinstaller包

pip3 install pyinstaller

2.找到pyinstaller命令路径(带bin,老方法卸载看路径),我直接执行不了pyinstaller指令,因为python系统就有,环境变量还没配置。

3.执行指令打包

先cd到需要打包文件的路径下,然后执行指令,我安装了一个超级右键程序,很方便操作

在这里插入图片描述

#将 xx.py 打包为 xx.exe

/Users/wenanqin/Library/Python/3.8/bin/pyinstaller -F xx.py

在这里插入图片描述


执行完操作,会生成三个文件,exe文件在dist文件内,至此,全部工作完成。

在这里插入图片描述


执行程序,效果如上。

如果想实现开盖自连,可看MacBook利用sleepwatcher实现开盖自动联网

原文地址:https://blog.csdn.net/weixin_42907822

版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 dio@foxmail.com 举报,一经查实,本站将立刻删除。

相关推荐


Python中的函数(二) 在上一篇文章中提到了Python中函数的定义和使用,在这篇文章里我们来讨论下关于函数的一些更深的话题。在学习C语言函数的时候,遇到的问题主要有形参实参的区别、参数的传递和改变、变量的作用域。同样在Python中,关于对函数的理解和使用也存在这些问题。下面来逐一讲解。一.函
Python中的字符串 可能大多数人在学习C语言的时候,最先接触的数据类型就是字符串,因为大多教程都是以"Hello world"这个程序作为入门程序,这个程序中要打印的"Hello world"就是字符串。如果你做过自然语言处理方面的研究,并且用Python
Python 面向对象编程(一) 虽然Python是解释性语言,但是它是面向对象的,能够进行对象编程。下面就来了解一下如何在Python中进行对象编程。一.如何定义一个类 在进行python面向对象编程之前,先来了解几个术语:类,类对象,实例对象,属性,函数和方法。 类是对现实世界中一些事物的封装,
Python面向对象编程(二) 在前面一篇文章中谈到了类的基本定义和使用方法,这只体现了面向对象编程的三大特点之一:封装。下面就来了解一下另外两大特征:继承和多态。 在Python中,如果需要的话,可以让一个类去继承一个类,被继承的类称为父类或者超类、也可以称作基类,继承的类称为子类。并且Pytho
Python中的函数(一) 接触过C语言的朋友对函数这个词肯定非常熟悉,无论在哪门编程语言当中,函数(当然在某些语言里称作方法,意义是相同的)都扮演着至关重要的角色。今天就来了解一下Python中的函数用法。一.函数的定义 在某些编程语言当中,函数声明和函数定义是区分开的(在这些编程语言当中函数声明
在windows下如何快速搭建web.py开发框架 用Python进行web开发的话有很多框架供选择,比如最出名的Django,tornado等,除了这些框架之外,有一个轻量级的框架使用起来也是非常方便和顺手,就是web.py。它由一名黑客所创建,但是不幸的是这位创建者于2013年自杀了。据说现在由
将Sublime Text 2搭建成一个好用的IDE 说起编辑器,可能大部分人要推荐的是Vim和Emacs,本人用过Vim,功能确实强大,但是不是很习惯,之前一直有朋友推荐SUblime Text 2这款编辑器,然后这段时间就试了一下,就深深地喜欢上这款编辑器了...
Python中的模块 有过C语言编程经验的朋友都知道在C语言中如果要引用sqrt这个函数,必须用语句"#include<math.h>"引入math.h这个头文件,否则是无法正常进行调用的。那么在Python中,如果要引用一些内置的函数,该怎么处理呢?在Python中
Python的基础语法 在对Python有了基础的认识之后,下面来了解一下Python的基础语法,看看它和C语言、java之间的基础语法差异。一.变量、表达式和语句 Python中的语句也称作命令,比如print "hello python"这就是一条语句。 表达式,顾名思义,是
Eclipse+PyDevʽjango+Mysql搭建Python web开发环境 Python的web框架有很多,目前主流的有Django、Tornado、Web.py等,最流行的要属Django了,也是被大家最看好的框架之一。下面就来讲讲如何搭建Django的开发环境。一.准备工作 需要下载的
在windows下安装配置Ulipad 今天推荐一款轻便的文本编辑器Ulipad,用来写一些小的Python脚本非常方便。 Ulipad下载地址: https://github.com/limodou/ulipad http://files.cnblogs.com/dolphin0520/u...
Python中的函数(三) 在前面两篇文章中已经探讨了函数的一些相关用法,下面一起来了解一下函数参数类型的问题。在C语言中,调用函数时必须依照函数定义时的参数个数以及类型来传递参数,否则将会发生错误,这个是严格进行规定的。然而在Python中函数参数定义和传递的方式相比而言就灵活多了。一.函数参数的
在Notepad++中搭配Python开发环境 Python在最近几年一度成为最流行的语言之一,不仅仅是因为它简洁明了,更在于它的功能之强大。它不仅能够完成一般脚本语言所能做的事情,还能很方便快捷地进行大规模的项目开发。在学习Python之前我们来看一下Python的历史由来,"Pytho
Python中的条件选择和循环语句 同C语言、Java一样,Python中也存在条件选择和循环语句,其风格和C语言、java的很类似,但是在写法和用法上还是有一些区别。今天就让我们一起来了解一下。一.条件选择语句 Python中条件选择语句的关键字为:if 、elif 、else这三个。其基本形式如
关于raw_input( )和sys.stdin.readline( )的区别 之前一直认为用raw_input( )和sys.stdin.readline( )来获取输入的效果完全相同,但是最近在写程序时有类似这样一段代码:import sysline = sys.stdin.readline()
初识Python 跟学习所有的编程语言一样,首先得了解这门语言的编程风格和最基础的语法。下面就让我们一起来了解一下Python的编程风格。1.逻辑行与物理行 在Python中有逻辑行和物理行这个概念,物理行是指在编辑器中实际看到的一行,逻辑行是指一条Python语句。在Python中提倡一个物理行只
当我们的代码是有访问网络相关的操作时,比如http请求或者访问远程数据库,经常可能会发生一些错误,有些错误可能重新去发送请求就会成功,本文分析常见可能需要重试的场景,并最后给出python代码实现。
1.经典迭代器 2.将Sentence中的__iter__改成生成器函数 改成生成器后用法不变,但更加简洁。 3.惰性实现 当列表比较大,占内存较大时,我们可以采用惰性实现,每次只读取一个元素到内存。 或者使用更简洁的生成器表达式 4.yield from itertools模块含有大量生成器函数可
本文介绍简单介绍socket的常用函数,并以python-kafka中的源码socketpair为例,来讲解python socket的运用
python实践中经常出现编码相关的异常,大多网上找资料而没有理解原理,导致一次次重复错误。本文对常用Unicode、UTF-8、GB2312编码的原理进行介绍,接着介绍了python字符类型unicode和str以及常见编解码错误UnicodeEncodeError和UnicodeDEcodeEr