-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathmain.py
144 lines (126 loc) · 5.02 KB
/
main.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
import websockets
import asyncio, json
from core import on_message
from utils.Decorators import startup_handler
from utils.Logger import logger, logging, on_stop, log_queue
from utils.Manager import load_plugins
from utils.Config import config
import os
import threading
import queue
from SimpfunAnnouncement import main as announcement_main
from SpigotMCResource import main as spigotmc_resource_main
dir = os.path.dirname(__file__)
status = 0
log_queue = queue.Queue()
def log_receiver():
while True:
try:
log_message = log_queue.get(timeout=3) # 设置超时避免永久阻塞
if log_message is None:
# 如果收到None,退出循环
break
# 这里可以处理日志消息,例如打印到控制台或发送到其他地方
print(log_message)
except queue.Empty:
# 如果队列为空,什么也不做,继续循环
pass
async def run_announcement():
from SimpfunAnnouncement import __init__ as announcement_init # 导入SimpfunAnnouncement的初始化函数
from SpigotMCResource import __init__ as spigotmc_resource_init
announcement_init() # 运行SimpfunAnnouncement的初始化函数
spigotmc_resource_init()
async def stop(ws=None):
logger.info("框架 >>> 正在关闭监听......")
if ws:
await ws.close()
logger.info("框架 >>> 正在关闭日志记录器......")
logging.shutdown()
await on_stop()
if status == 1:
os.remove('session.lock')
else:
pass
os._exit(0)
async def handle_websocket(ws):
if os.path.exists('session.lock'):
logger.error(f'框架 >>> 警告,框架已启动或未正常关机,请先关闭存在的框架或删除 session.lock 后再启动!')
await stop()
else:
pass
try:
async with websockets.connect(config.url) as websocket:
info = json.loads(await websocket.recv())
if info['sub_type'] == 'connect':
get_login_info = {"action": "get_login_info", "params": {}}
await websocket.send(json.dumps(get_login_info))
info = json.loads(await websocket.recv())
uid = info['data']['user_id']
nickname = info['data']['nickname']
logger.info(f'框架 >>> QQ {uid} 已连接')
logger.info(f'框架 >>> 欢迎您,{nickname}!')
with open('session.lock', 'w') as f:
f.close()
global status
status = 1
log_thread = threading.Thread(target=log_receiver, daemon=True)
log_thread.start()
asyncio.create_task(run_announcement())
async for message in ws:
await on_message(message,nickname)
except websockets.exceptions.ConnectionClosed:
await handle_reconnection()
except asyncio.CancelledError:
await stop(ws)
async def handle_reconnection():
logger.warning("框架 >>> WebSocket 连接被关闭,正在尝试重连")
reconnect_attempts = config.retry
if reconnect_attempts != 0:
for attempt in range(1, reconnect_attempts + 1):
if await attempt_reconnect(attempt, reconnect_attempts):
break
else:
await stop()
else:
await attempt_reconnect()
async def attempt_reconnect(attempt=0, max_attempts=0):
try:
async with websockets.connect(config.url) as ws:
await handle_websocket(ws)
return True
except (ConnectionRefusedError, websockets.exceptions.WebSocketException) as e:
logger.error(f'框架 >>> 连接失败:{e}')
if attempt < max_attempts:
logger.warning(f'框架 >>> 正在尝试重连 {attempt}/{max_attempts}...')
await asyncio.sleep(5)
else:
logger.error('框架 >>> Websocket连接失败,即将退出...')
return False
async def connect_and_run():
version = config.version
logger.info(f'框架 >>> 框架已启动 当前框架版本:SPR-Robot:{version}')
if config.debug:
logger.warning(f'框架 >>> 开发者模式已启动,如果您不是开发人员,请将config.yml中的debug设为False')
logger.info('插件管理器 >>> 开始加载插件......')
await load_plugins(dir)
logger.info('插件管理器 >>> 加载完毕')
reconnect_attempts = config.retry
if reconnect_attempts != 0:
for attempt in range(1, reconnect_attempts + 1):
if await attempt_reconnect(attempt, reconnect_attempts):
break
else:
await stop()
else:
await attempt_reconnect()
async def main():
log_thread = threading.Thread(target=log_receiver)
log_thread.daemon = True # 设置为守护线程,以便主程序退出时可以立即退出
log_thread.start()
await connect_and_run()
await startup_handler()
if __name__ == "__main__":
try:
asyncio.run(main())
except KeyboardInterrupt:
asyncio.run(stop())