mirror of
https://github.com/kxgx/2.13-Ink-screen-clock.git
synced 2026-03-15 07:03:16 +08:00
193 lines
6.2 KiB
Python
193 lines
6.2 KiB
Python
#!/usr/bin/env python
|
||
# -*- coding:utf-8 -*-
|
||
import os
|
||
import json
|
||
import time
|
||
import logging
|
||
import requests
|
||
import random
|
||
import socket
|
||
from functools import wraps
|
||
from threading import Timer
|
||
|
||
logging.basicConfig(
|
||
level=logging.INFO,
|
||
format='%(asctime)s - %(message)s',
|
||
datefmt='%Y-%m-%d %H:%M:%S'
|
||
)
|
||
|
||
def check_network_connection():
|
||
"""检查网络连接状态"""
|
||
try:
|
||
# 尝试连接一个可靠的公共DNS服务器
|
||
socket.create_connection(("223.5.5.5", 53), timeout=5)
|
||
return True
|
||
except OSError:
|
||
return False
|
||
|
||
def get_ip():
|
||
"""改进的IP获取函数"""
|
||
if not check_network_connection():
|
||
logging.warning("网络连接不可用")
|
||
return None
|
||
|
||
services = [
|
||
{"url": "https://api.ipify.org?format=json", "field": "ip"},
|
||
{"url": "https://ipinfo.io/json", "field": "ip"},
|
||
{"url": "https://ifconfig.me/all.json", "field": "ip_addr"}
|
||
]
|
||
|
||
for service in services:
|
||
try:
|
||
resp = requests.get(service["url"], timeout=10)
|
||
data = resp.json()
|
||
return data.get(service["field"])
|
||
except Exception:
|
||
continue
|
||
|
||
logging.error("所有IP服务尝试失败")
|
||
return None
|
||
def get_ip():
|
||
"""从ip.cn获取当前IP地址"""
|
||
url = "https://ip.cn/api/index?ip=&type=0"
|
||
headers = {
|
||
'User-Agent': 'Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/120.0.0.0 Safari/537.36'
|
||
}
|
||
try:
|
||
resp = requests.get(url, headers=headers, timeout=10)
|
||
resp.raise_for_status()
|
||
data = resp.json()
|
||
if data.get('code') == 'Success':
|
||
return data.get('ip', '')
|
||
logging.error("获取IP失败: %s", data.get('msg', '未知错误'))
|
||
except Exception as e:
|
||
logging.error("获取IP异常: %s", str(e))
|
||
return None
|
||
|
||
def get_current_city():
|
||
"""通过IP地址获取当前定位城市并去除'市'后缀"""
|
||
ip = get_ip()
|
||
if not ip:
|
||
return None
|
||
|
||
url = f"http://ip-api.com/json/{ip}?fields=city&lang=zh-CN"
|
||
headers = {
|
||
'User-Agent': 'Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/120.0.0.0 Safari/537.36'
|
||
}
|
||
while True:
|
||
try:
|
||
resp = requests.get(url, headers=headers, timeout=10)
|
||
data = resp.json()
|
||
if data.get('status') == 'success':
|
||
return data.get('city', '').replace('市', '')
|
||
logging.error("定位失败: %s", data.get('message', '未知错误'))
|
||
except Exception as e:
|
||
logging.error("定位异常: %s", str(e))
|
||
time.sleep(180)
|
||
|
||
def schedule_getWeath():
|
||
"""改进的定时任务调度"""
|
||
retry_count = 0
|
||
max_retries = 3
|
||
|
||
while retry_count < max_retries:
|
||
try:
|
||
getWeath()
|
||
break
|
||
except Exception as e:
|
||
retry_count += 1
|
||
wait_time = min(300, retry_count * 60) # 指数退避
|
||
logging.error(f"天气更新失败({retry_count}/{max_retries}): {str(e)}")
|
||
time.sleep(wait_time)
|
||
|
||
# 无论成功与否,都安排下一次执行
|
||
Timer(1800, schedule_getWeath).start() # 30分钟间隔
|
||
|
||
def getWeath(default_city='101060101'):
|
||
"""改进的天气获取函数"""
|
||
# 1. 检查网络连接
|
||
if not check_network_connection():
|
||
logging.error("网络不可用,跳过天气更新")
|
||
return
|
||
|
||
# 2. 获取城市信息
|
||
city_name = None
|
||
try:
|
||
city_name = get_current_city()
|
||
except Exception as e:
|
||
logging.error(f"获取城市失败: {str(e)}")
|
||
|
||
# 3. 确定区域ID
|
||
area_id = default_city
|
||
if city_name:
|
||
try:
|
||
found_id = get_area_id(city_name)
|
||
if found_id:
|
||
area_id = found_id
|
||
logging.info(f"使用城市区域ID: {area_id} ({city_name})")
|
||
except Exception as e:
|
||
logging.error(f"获取区域ID失败: {str(e)}")
|
||
|
||
# 4. 获取天气数据
|
||
weather_apis = [
|
||
f'https://d1.weather.com.cn/sk_2d/{area_id}.html',
|
||
f'https://www.weather.com.cn/weather1d/{area_id}.shtml'
|
||
]
|
||
|
||
for api_url in weather_apis:
|
||
try:
|
||
resp = requests.get(
|
||
api_url,
|
||
headers={'User-Agent': 'Mozilla/5.0'},
|
||
timeout=15
|
||
)
|
||
resp.raise_for_status()
|
||
|
||
# 处理不同API的响应格式
|
||
if 'sk_2d' in api_url:
|
||
weather_data = resp.content[11:].decode('utf-8')
|
||
else:
|
||
weather_data = parse_html_weather(resp.text)
|
||
|
||
# 验证数据有效性
|
||
if validate_weather_data(weather_data):
|
||
with open('/root/2.13-Ink-screen-clock/bin/weather.json', 'w') as f:
|
||
json.dump(weather_data, f)
|
||
logging.info("天气数据更新成功")
|
||
return
|
||
|
||
except Exception as e:
|
||
logging.warning(f"天气API {api_url} 失败: {str(e)}")
|
||
continue
|
||
|
||
logging.error("所有天气API尝试失败")
|
||
|
||
def get_area_id(city_name):
|
||
"""从city.js中检索AREAID,无限重试直到成功"""
|
||
url = "https://j.i8tq.com/weather2020/search/city.js"
|
||
headers = {
|
||
'User-Agent': 'Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/120.0.0.0 Safari/537.36'
|
||
}
|
||
while True:
|
||
try:
|
||
resp = requests.get(url, headers=headers, timeout=10)
|
||
resp.encoding = 'utf-8'
|
||
city_data = json.loads(resp.text.split('=', 1)[1].rstrip(';'))
|
||
for province in city_data.values():
|
||
for city in province.values():
|
||
for district, info in city.items():
|
||
if info['NAMECN'] == city_name:
|
||
return info['AREAID']
|
||
logging.error("未找到城市: %s", city_name)
|
||
except Exception as e:
|
||
logging.error("获取城市ID失败: %s", str(e))
|
||
time.sleep(180)
|
||
|
||
if __name__ == "__main__":
|
||
try:
|
||
schedule_getWeath()
|
||
while True:
|
||
time.sleep(1)
|
||
except KeyboardInterrupt:
|
||
logging.info("程序已终止")
|