2024-05-20 23:07:29 +08:00
|
|
|
import subprocess
|
|
|
|
import re
|
|
|
|
import os
|
2024-05-25 00:24:51 +08:00
|
|
|
import psutil
|
|
|
|
import signal
|
2024-05-20 23:07:29 +08:00
|
|
|
import time
|
2024-05-21 01:42:24 +08:00
|
|
|
import datetime
|
2024-05-27 21:20:54 +08:00
|
|
|
import json
|
2024-05-20 23:07:29 +08:00
|
|
|
from flask import Flask, render_template, request, redirect, url_for, make_response
|
|
|
|
|
2024-05-24 23:16:37 +08:00
|
|
|
try:
|
|
|
|
from takway.board import OrangePi
|
|
|
|
led_enabled = True
|
|
|
|
orangepi = OrangePi()
|
|
|
|
except ImportError:
|
|
|
|
led_enabled = False
|
|
|
|
print("Error importing OrangePi")
|
|
|
|
|
2024-05-25 23:24:02 +08:00
|
|
|
# blue: APP
|
|
|
|
# red: hotspot
|
2024-05-25 00:24:51 +08:00
|
|
|
|
2024-05-25 21:59:11 +08:00
|
|
|
app = Flask(__name__)
|
2024-05-24 23:16:37 +08:00
|
|
|
|
2024-05-25 00:02:37 +08:00
|
|
|
def close_app():
|
2024-05-25 00:33:37 +08:00
|
|
|
if led_enabled:
|
2024-05-25 23:24:02 +08:00
|
|
|
orangepi.set_led_off('blue')
|
2024-05-27 21:20:54 +08:00
|
|
|
|
2024-05-25 00:02:37 +08:00
|
|
|
# 获取当前Flask应用程序的进程ID
|
|
|
|
flask_pid = os.getpid()
|
2024-05-27 21:20:54 +08:00
|
|
|
|
2024-05-25 00:02:37 +08:00
|
|
|
# 获取所有名为'python3'的进程ID
|
|
|
|
python_pids = [p.info['pid'] for p in psutil.process_iter(attrs=['pid', 'name']) if p.info['name'] == 'python3']
|
2024-05-27 21:20:54 +08:00
|
|
|
|
2024-05-25 00:02:37 +08:00
|
|
|
# 关闭Flask应用程序进程
|
|
|
|
os.kill(flask_pid, signal.SIGTERM)
|
2024-05-27 21:20:54 +08:00
|
|
|
|
2024-05-25 00:02:37 +08:00
|
|
|
# 关闭Python进程
|
|
|
|
for pid in python_pids:
|
|
|
|
os.kill(pid, signal.SIGTERM)
|
2024-05-24 23:16:37 +08:00
|
|
|
|
2024-05-25 21:59:11 +08:00
|
|
|
def start_hotspot():
|
|
|
|
try:
|
|
|
|
subprocess.Popen('sudo systemctl start hotspot.service', shell=True)
|
|
|
|
if led_enabled:
|
2024-05-25 23:24:02 +08:00
|
|
|
orangepi.set_led_on('red')
|
2024-05-25 21:59:11 +08:00
|
|
|
except subprocess.CalledProcessError as e:
|
|
|
|
print(f"{datetime.datetime.now()}: Error starting create_ap service: {e}")
|
2024-05-24 23:16:37 +08:00
|
|
|
|
2024-05-25 21:59:11 +08:00
|
|
|
def close_hotspot():
|
|
|
|
# 关闭热点
|
|
|
|
try:
|
|
|
|
subprocess.Popen('sudo systemctl stop hotspot.service', shell=True)
|
|
|
|
print(f"{datetime.datetime.now()}: Stopping create_ap service")
|
2024-05-25 23:24:02 +08:00
|
|
|
if led_enabled:
|
|
|
|
orangepi.set_led_off('red')
|
2024-05-25 21:59:11 +08:00
|
|
|
except subprocess.CalledProcessError as e:
|
|
|
|
print(f"{datetime.datetime.now()}: Error stopping hotspot: {e}")
|
2024-05-27 21:20:54 +08:00
|
|
|
|
|
|
|
|
2024-05-20 23:07:29 +08:00
|
|
|
# 检测 Wi-Fi 连接状态
|
|
|
|
def check_wifi_connection():
|
2024-05-21 01:42:24 +08:00
|
|
|
cmd = "nmcli dev status"
|
|
|
|
result = subprocess.run(cmd, shell=True, capture_output=True, text=True)
|
2024-05-24 23:16:37 +08:00
|
|
|
time.sleep(1)
|
2024-05-21 01:42:24 +08:00
|
|
|
output = result.stdout.strip()
|
|
|
|
lines = output.split("\n")[1:] # Skip the header line
|
|
|
|
for line in lines:
|
|
|
|
columns = line.split()
|
|
|
|
print(columns)
|
|
|
|
if len(columns) >= 4 and columns[2] == "connected":
|
2024-05-25 21:59:11 +08:00
|
|
|
wifi_ssid = columns[3]
|
2024-05-25 00:33:37 +08:00
|
|
|
if led_enabled:
|
|
|
|
orangepi.set_led_off('blue')
|
2024-05-25 21:59:11 +08:00
|
|
|
return True, wifi_ssid
|
2024-05-27 21:20:54 +08:00
|
|
|
return False, None
|
2024-05-20 23:07:29 +08:00
|
|
|
|
2024-05-25 21:59:11 +08:00
|
|
|
def scan_wifi():
|
2024-05-27 21:50:20 +08:00
|
|
|
try:
|
|
|
|
subprocess.run(['nmcli', 'dev', 'wifi', 'rescan'], check=True)
|
|
|
|
cmd = "nmcli dev wifi"
|
|
|
|
result = subprocess.run(cmd, shell=True, capture_output=True, text=True)
|
|
|
|
except subprocess.CalledProcessError as e:
|
|
|
|
print(f"{datetime.datetime.now()}: Error scanning Wi-Fi: {e}")
|
|
|
|
error_time = datetime.datetime.now()
|
|
|
|
while led_enabled:
|
|
|
|
orangepi.set_led_off('blue')
|
|
|
|
orangepi.set_led_on('red')
|
|
|
|
time.sleep(0.5)
|
|
|
|
orangepi.set_led_on('blue')
|
|
|
|
orangepi.set_led_off('red')
|
|
|
|
time.sleep(0.5)
|
|
|
|
if error_time + datetime.timedelta(seconds=5) < datetime.datetime.now():
|
|
|
|
break
|
|
|
|
close_app()
|
2024-05-25 22:44:42 +08:00
|
|
|
ssid_list = []
|
2024-05-25 21:59:11 +08:00
|
|
|
wifi_list = []
|
|
|
|
output = result.stdout.strip()
|
|
|
|
lines = output.split("\n")[1:] # Skip the header line
|
|
|
|
for line in lines:
|
|
|
|
columns = line.split()
|
2024-05-25 22:35:35 +08:00
|
|
|
'''
|
2024-05-25 21:59:11 +08:00
|
|
|
print(columns)
|
|
|
|
for i, column in enumerate(columns):
|
|
|
|
print(f"{i}: {column}")
|
2024-05-25 22:35:35 +08:00
|
|
|
'''
|
|
|
|
# ['94:14:57:15:13:50', 'Meeting', 'MG', 'Infra', '1', '130', 'Mbit/s', '100', '****', 'WPA1', 'WPA2']
|
2024-05-25 22:32:53 +08:00
|
|
|
|
|
|
|
# 提取MAC地址
|
|
|
|
mac_address = columns[0]
|
|
|
|
|
|
|
|
# 提取Wi-Fi名称
|
2024-05-25 22:44:42 +08:00
|
|
|
wifi_name = []
|
2024-05-25 22:32:53 +08:00
|
|
|
for i in range(1, len(columns)):
|
|
|
|
if columns[i] == 'Infra':
|
2024-05-25 22:44:42 +08:00
|
|
|
_wifi_name = ' '.join(columns[1:i])
|
2024-05-25 22:45:21 +08:00
|
|
|
wifi_name.append(_wifi_name)
|
2024-05-25 22:32:53 +08:00
|
|
|
columns = columns[i+1:]
|
|
|
|
break
|
2024-05-25 22:44:42 +08:00
|
|
|
ssid = wifi_name[0]
|
2024-05-25 23:41:36 +08:00
|
|
|
if ':' in ssid:
|
|
|
|
# 94:14:57:15:13:50 Meeting
|
|
|
|
# 去除mac地址
|
|
|
|
ssid = ssid.split(' ')[1:]
|
|
|
|
ssid = ''.join(ssid)
|
2024-05-27 21:20:54 +08:00
|
|
|
|
2024-05-25 22:44:42 +08:00
|
|
|
if ssid in ssid_list:
|
|
|
|
continue
|
|
|
|
ssid_list.append(ssid)
|
2024-05-27 21:20:54 +08:00
|
|
|
|
2024-05-25 22:32:53 +08:00
|
|
|
# 提取强度
|
|
|
|
strength = None
|
|
|
|
for i in range(len(columns)):
|
|
|
|
if columns[i] == 'Mbit/s':
|
|
|
|
strength = columns[i+1]
|
|
|
|
break
|
2024-05-25 22:46:52 +08:00
|
|
|
# print("MAC地址:", mac_address)
|
|
|
|
# print("Wi-Fi名称:", ssid)
|
|
|
|
# print("强度:", strength)
|
2024-05-27 21:20:54 +08:00
|
|
|
|
2024-05-25 22:59:48 +08:00
|
|
|
wifi_list.append({'ssid': ssid, 'signal': strength, 'mac': mac_address})
|
2024-05-25 21:59:11 +08:00
|
|
|
if len(wifi_list) == 15:
|
|
|
|
break
|
2024-05-27 21:20:54 +08:00
|
|
|
|
2024-05-25 21:59:11 +08:00
|
|
|
return wifi_list
|
2024-05-20 23:07:29 +08:00
|
|
|
|
|
|
|
# 连接 Wi-Fi
|
|
|
|
def connect_wifi(ssid, password):
|
|
|
|
# 连接到用户选择的 Wi-Fi 网络
|
|
|
|
try:
|
2024-05-25 21:59:11 +08:00
|
|
|
output = subprocess.check_output(['nmcli', 'dev', 'wifi', 'connect', ssid, 'password', password])
|
|
|
|
output_str = output.decode('utf-8') # 将输出转换为字符串
|
|
|
|
if "successfully" in output_str:
|
|
|
|
print(f"{datetime.datetime.now()}: Successfully connected to Wi-Fi: {ssid}")
|
2024-05-27 21:20:54 +08:00
|
|
|
save_wifi(ssid, password) # 保存连接成功的Wi-Fi信息
|
2024-05-25 21:59:11 +08:00
|
|
|
return True
|
|
|
|
else:
|
|
|
|
print(f"{datetime.datetime.now()}: Error connecting to Wi-Fi: {output_str}")
|
|
|
|
return False
|
2024-05-20 23:07:29 +08:00
|
|
|
except subprocess.CalledProcessError as e:
|
2024-05-25 00:00:33 +08:00
|
|
|
print(f"{datetime.datetime.now()}: Error connecting to Wi-Fi: {e}")
|
2024-05-20 23:07:29 +08:00
|
|
|
return False
|
2024-05-25 21:59:11 +08:00
|
|
|
|
|
|
|
# 关闭 Wi-Fi
|
|
|
|
def disconnect_wifi():
|
|
|
|
try:
|
|
|
|
output = subprocess.check_output(['nmcli', 'dev', 'disconnect', 'iface', 'wlan0'])
|
2024-05-25 22:19:52 +08:00
|
|
|
time.sleep(1)
|
2024-05-25 21:59:11 +08:00
|
|
|
output_str = output.decode('utf-8') # 将输出转换为字符串
|
|
|
|
if "successfully disconnected" in output_str:
|
2024-05-25 22:13:04 +08:00
|
|
|
print(f"{datetime.datetime.now()}: Wi-Fi disconnected successfully")
|
|
|
|
else:
|
|
|
|
print(f"{datetime.datetime.now()}: Error disconnecting from Wi-Fi: {output_str}")
|
2024-05-25 22:21:14 +08:00
|
|
|
except Exception as e:
|
|
|
|
print(f"{datetime.datetime.now()}: Error disconnecting from Wi-Fi: {e}")
|
2024-05-25 21:59:11 +08:00
|
|
|
|
2024-05-27 21:50:20 +08:00
|
|
|
def connect_saved_wifi(scaned_wifi_list):
|
|
|
|
wifi_list = load_saved_wifi()
|
|
|
|
if not wifi_list:
|
|
|
|
return False, None
|
|
|
|
for wifi in wifi_list:
|
|
|
|
if wifi['ssid'] in [item['ssid'] for item in scaned_wifi_list]:
|
|
|
|
connect_wifi(wifi['ssid'], wifi['password'])
|
|
|
|
return True, wifi['ssid']
|
|
|
|
return False, None
|
|
|
|
|
|
|
|
def load_saved_wifi():
|
|
|
|
try:
|
|
|
|
with open('wifi_list.json', 'r') as f:
|
|
|
|
wifi_list = json.load(f)
|
|
|
|
return wifi_list
|
|
|
|
except FileNotFoundError:
|
|
|
|
return []
|
|
|
|
|
|
|
|
def save_wifi(ssid, password):
|
|
|
|
wifi_list = load_saved_wifi()
|
|
|
|
wifi_list.append({'ssid': ssid, 'password': password})
|
|
|
|
with open('wifi_list.json', 'w') as f:
|
|
|
|
json.dump(wifi_list, f)
|
2024-05-25 21:59:11 +08:00
|
|
|
|
|
|
|
|
2024-05-20 23:07:29 +08:00
|
|
|
# 主页
|
|
|
|
@app.route('/')
|
|
|
|
def index():
|
|
|
|
global wifi_list
|
2024-05-27 21:20:54 +08:00
|
|
|
|
2024-05-20 23:07:29 +08:00
|
|
|
response = make_response(render_template('index.html', wifi_list=wifi_list))
|
|
|
|
response.headers.set('Content-Type', 'text/html')
|
|
|
|
response.headers.set('Apple-Web-App-Capable', 'yes')
|
|
|
|
response.headers.set('Apple-Mobile-Web-App-Status-Bar-Style', 'black-translucent')
|
|
|
|
return response
|
2024-05-27 21:20:54 +08:00
|
|
|
|
2024-05-20 23:07:29 +08:00
|
|
|
# 提交 Wi-Fi 信息
|
|
|
|
@app.route('/submit', methods=['POST'])
|
|
|
|
def submit():
|
2024-05-25 00:33:37 +08:00
|
|
|
if led_enabled:
|
|
|
|
orangepi.set_led_on('red')
|
2024-05-20 23:07:29 +08:00
|
|
|
ssid = request.form['ssid']
|
|
|
|
password = request.form['password']
|
2024-05-25 00:00:33 +08:00
|
|
|
print(f"{datetime.datetime.now()}: Connecting to Wi-Fi: {ssid} with password {password}")
|
2024-05-27 21:20:54 +08:00
|
|
|
|
2024-05-25 22:57:24 +08:00
|
|
|
# 关闭热点
|
|
|
|
close_hotspot()
|
2024-05-27 21:20:54 +08:00
|
|
|
|
2024-05-27 21:22:42 +08:00
|
|
|
time.sleep(3)
|
2024-05-25 22:57:24 +08:00
|
|
|
# 连接到用户选择的 Wi-Fi 网络
|
|
|
|
if connect_wifi(ssid, password):
|
2024-05-25 00:02:37 +08:00
|
|
|
close_app()
|
2024-05-27 21:20:54 +08:00
|
|
|
|
|
|
|
connected, wifi_ssid = check_wifi_connection()
|
|
|
|
if not connected:
|
2024-05-25 00:00:33 +08:00
|
|
|
print(f"{datetime.datetime.now()}: Wi-Fi连接失败。")
|
2024-05-25 22:57:24 +08:00
|
|
|
wifi_list = scan_wifi()
|
|
|
|
start_hotspot()
|
2024-05-27 21:20:54 +08:00
|
|
|
else:
|
|
|
|
save_wifi(wifi_ssid, password) # 保存连接成功的Wi-Fi信息
|
|
|
|
|
2024-05-20 23:07:29 +08:00
|
|
|
return redirect(url_for('index'))
|
|
|
|
|
|
|
|
|
|
|
|
if __name__ == '__main__':
|
2024-05-25 23:04:33 +08:00
|
|
|
debug_mode = False # 设置为 True 以跳过 Wi-Fi 连接状态检测
|
2024-05-25 21:59:11 +08:00
|
|
|
if debug_mode:
|
|
|
|
disconnect_wifi()
|
2024-05-25 22:53:36 +08:00
|
|
|
wifi_list = scan_wifi()
|
|
|
|
print(wifi_list)
|
2024-05-27 21:20:54 +08:00
|
|
|
|
2024-05-25 22:52:14 +08:00
|
|
|
start_hotspot()
|
2024-05-27 21:20:54 +08:00
|
|
|
|
2024-05-25 23:04:33 +08:00
|
|
|
# app.run(host='0.0.0.0', port=80)
|
2024-05-27 21:20:54 +08:00
|
|
|
|
2024-05-25 22:48:27 +08:00
|
|
|
if connect_wifi("Innoxsz-Public", "innox2023"):
|
|
|
|
close_app()
|
2024-05-25 22:10:04 +08:00
|
|
|
|
2024-05-25 00:33:37 +08:00
|
|
|
if led_enabled:
|
|
|
|
orangepi.set_led_on('blue')
|
2024-05-27 21:20:54 +08:00
|
|
|
|
|
|
|
connected, wifi_ssid = check_wifi_connection()
|
|
|
|
if connected:
|
2024-05-25 21:59:11 +08:00
|
|
|
print(f"{datetime.datetime.now()}: 系统已自动连接到 Wi-Fi 网络,退出程序")
|
|
|
|
close_app()
|
2024-05-27 21:50:20 +08:00
|
|
|
|
|
|
|
wifi_list = scan_wifi()
|
|
|
|
|
|
|
|
connected, wifi_ssid = connect_saved_wifi(wifi_list)
|
|
|
|
if connected:
|
|
|
|
print(f"{datetime.datetime.now()}: 系统已自动连接到 Wi-Fi 网络 {wifi_ssid},退出程序")
|
|
|
|
close_app()
|
|
|
|
|
|
|
|
print(f"{datetime.datetime.now()}: 未连接到 Wi-Fi 网络, 开始热点模式")
|
|
|
|
start_hotspot()
|
|
|
|
app.run(host='0.0.0.0', port=80)
|