mirror of
https://github.com/LiteyukiStudio/LiteyukiBot.git
synced 2024-11-14 13:47:24 +08:00
🐛 在结束进程时无法杀死进程的问题
This commit is contained in:
parent
0d5f9fee52
commit
a61357f4e2
@ -4,10 +4,7 @@ import platform
|
|||||||
import sys
|
import sys
|
||||||
import threading
|
import threading
|
||||||
import time
|
import time
|
||||||
from typing import Any, Iterable, Optional
|
from typing import Any, Optional
|
||||||
|
|
||||||
from watchdog.events import FileSystemEventHandler
|
|
||||||
from watchdog.observers import Observer
|
|
||||||
|
|
||||||
from liteyuki.bot.lifespan import (LIFESPAN_FUNC, Lifespan)
|
from liteyuki.bot.lifespan import (LIFESPAN_FUNC, Lifespan)
|
||||||
from liteyuki.comm import get_channel
|
from liteyuki.comm import get_channel
|
||||||
@ -38,6 +35,7 @@ class LiteyukiBot:
|
|||||||
self.lifespan = Lifespan()
|
self.lifespan = Lifespan()
|
||||||
|
|
||||||
self.process_manager: ProcessManager = ProcessManager(bot=self)
|
self.process_manager: ProcessManager = ProcessManager(bot=self)
|
||||||
|
|
||||||
self.loop = asyncio.new_event_loop()
|
self.loop = asyncio.new_event_loop()
|
||||||
asyncio.set_event_loop(self.loop)
|
asyncio.set_event_loop(self.loop)
|
||||||
self.loop_thread = threading.Thread(target=self.loop.run_forever, daemon=True)
|
self.loop_thread = threading.Thread(target=self.loop.run_forever, daemon=True)
|
||||||
@ -50,15 +48,9 @@ class LiteyukiBot:
|
|||||||
"""
|
"""
|
||||||
启动逻辑
|
启动逻辑
|
||||||
"""
|
"""
|
||||||
self.loop_thread.start() # 启动事件循环
|
self.lifespan.before_start() # 启动前钩子
|
||||||
asyncio.run(self.lifespan.before_start()) # 启动前钩子
|
self.process_manager.start_all()
|
||||||
|
self.lifespan.after_start() # 启动后钩子
|
||||||
asyncio.run(self.lifespan.after_start()) # 启动后钩子
|
|
||||||
self.start_watcher() # 启动文件监视器,后续准备插件化
|
|
||||||
self.keep_running()
|
|
||||||
|
|
||||||
def start_watcher(self):
|
|
||||||
pass
|
|
||||||
|
|
||||||
def restart(self, delay: int = 0):
|
def restart(self, delay: int = 0):
|
||||||
"""
|
"""
|
||||||
@ -120,6 +112,15 @@ class LiteyukiBot:
|
|||||||
def init_config(self):
|
def init_config(self):
|
||||||
pass
|
pass
|
||||||
|
|
||||||
|
def stop(self):
|
||||||
|
"""
|
||||||
|
停止轻雪
|
||||||
|
Returns:
|
||||||
|
|
||||||
|
"""
|
||||||
|
self.stop_event.set()
|
||||||
|
self.loop.stop()
|
||||||
|
|
||||||
def on_before_start(self, func: LIFESPAN_FUNC):
|
def on_before_start(self, func: LIFESPAN_FUNC):
|
||||||
"""
|
"""
|
||||||
注册启动前的函数
|
注册启动前的函数
|
||||||
@ -198,9 +199,6 @@ class LiteyukiBot:
|
|||||||
"""
|
"""
|
||||||
return self.lifespan.on_after_nonebot_init(func)
|
return self.lifespan.on_after_nonebot_init(func)
|
||||||
|
|
||||||
def keep_running(self):
|
|
||||||
self.stop_event.wait()
|
|
||||||
|
|
||||||
|
|
||||||
_BOT_INSTANCE: Optional[LiteyukiBot] = None
|
_BOT_INSTANCE: Optional[LiteyukiBot] = None
|
||||||
|
|
||||||
|
@ -8,23 +8,27 @@ Copyright (C) 2020-2024 LiteyukiStudio. All Rights Reserved
|
|||||||
@File : lifespan.py
|
@File : lifespan.py
|
||||||
@Software: PyCharm
|
@Software: PyCharm
|
||||||
"""
|
"""
|
||||||
|
import asyncio
|
||||||
from typing import Any, Awaitable, Callable, TypeAlias
|
from typing import Any, Awaitable, Callable, TypeAlias
|
||||||
|
|
||||||
from liteyuki.log import logger
|
from liteyuki.log import logger
|
||||||
from liteyuki.utils import is_coroutine_callable
|
from liteyuki.utils import is_coroutine_callable, async_wrapper
|
||||||
|
|
||||||
SYNC_LIFESPAN_FUNC: TypeAlias = Callable[[], Any]
|
SYNC_LIFESPAN_FUNC: TypeAlias = Callable[[], Any]
|
||||||
ASYNC_LIFESPAN_FUNC: TypeAlias = Callable[[], Awaitable[Any]]
|
ASYNC_LIFESPAN_FUNC: TypeAlias = Callable[[], Awaitable[Any]]
|
||||||
LIFESPAN_FUNC: TypeAlias = SYNC_LIFESPAN_FUNC | ASYNC_LIFESPAN_FUNC
|
LIFESPAN_FUNC: TypeAlias = SYNC_LIFESPAN_FUNC | ASYNC_LIFESPAN_FUNC
|
||||||
|
|
||||||
|
SYNC_PROCESS_LIFESPAN_FUNC: TypeAlias = Callable[[str], Any]
|
||||||
|
ASYNC_PROCESS_LIFESPAN_FUNC: TypeAlias = Callable[[str], Awaitable[Any]]
|
||||||
|
PROCESS_LIFESPAN_FUNC: TypeAlias = SYNC_PROCESS_LIFESPAN_FUNC | ASYNC_PROCESS_LIFESPAN_FUNC
|
||||||
|
|
||||||
|
|
||||||
class Lifespan:
|
class Lifespan:
|
||||||
def __init__(self) -> None:
|
def __init__(self) -> None:
|
||||||
"""
|
"""
|
||||||
轻雪生命周期管理,启动、停止、重启
|
轻雪生命周期管理,启动、停止、重启
|
||||||
"""
|
"""
|
||||||
|
self.life_flag: int = 0
|
||||||
self.life_flag: int = 0 # 0: 启动前,1: 启动后,2: 停止前,3: 停止后
|
|
||||||
|
|
||||||
self._before_start_funcs: list[LIFESPAN_FUNC] = []
|
self._before_start_funcs: list[LIFESPAN_FUNC] = []
|
||||||
self._after_start_funcs: list[LIFESPAN_FUNC] = []
|
self._after_start_funcs: list[LIFESPAN_FUNC] = []
|
||||||
@ -38,18 +42,26 @@ class Lifespan:
|
|||||||
self._after_nonebot_init_funcs: list[LIFESPAN_FUNC] = []
|
self._after_nonebot_init_funcs: list[LIFESPAN_FUNC] = []
|
||||||
|
|
||||||
@staticmethod
|
@staticmethod
|
||||||
async def _run_funcs(funcs: list[LIFESPAN_FUNC]) -> None:
|
def _run_funcs(funcs: list[LIFESPAN_FUNC | PROCESS_LIFESPAN_FUNC], *args, **kwargs) -> None:
|
||||||
"""
|
"""
|
||||||
运行函数
|
运行函数
|
||||||
Args:
|
Args:
|
||||||
funcs:
|
funcs:
|
||||||
Returns:
|
Returns:
|
||||||
"""
|
"""
|
||||||
|
try:
|
||||||
|
loop = asyncio.get_event_loop()
|
||||||
|
except RuntimeError:
|
||||||
|
loop = asyncio.new_event_loop()
|
||||||
|
asyncio.set_event_loop(loop)
|
||||||
|
|
||||||
|
tasks = []
|
||||||
for func in funcs:
|
for func in funcs:
|
||||||
if is_coroutine_callable(func):
|
if is_coroutine_callable(func):
|
||||||
await func()
|
tasks.append(func(*args, **kwargs))
|
||||||
else:
|
else:
|
||||||
func()
|
tasks.append(async_wrapper(func)(*args, **kwargs))
|
||||||
|
loop.run_until_complete(asyncio.gather(*tasks))
|
||||||
|
|
||||||
def on_before_start(self, func: LIFESPAN_FUNC) -> LIFESPAN_FUNC:
|
def on_before_start(self, func: LIFESPAN_FUNC) -> LIFESPAN_FUNC:
|
||||||
"""
|
"""
|
||||||
@ -131,59 +143,51 @@ class Lifespan:
|
|||||||
self._after_nonebot_init_funcs.append(func)
|
self._after_nonebot_init_funcs.append(func)
|
||||||
return func
|
return func
|
||||||
|
|
||||||
async def before_start(self) -> None:
|
def before_start(self) -> None:
|
||||||
"""
|
"""
|
||||||
启动前
|
启动前
|
||||||
Returns:
|
Returns:
|
||||||
"""
|
"""
|
||||||
logger.debug("Running before_start functions")
|
logger.debug("Running before_start functions")
|
||||||
await self._run_funcs(self._before_start_funcs)
|
self._run_funcs(self._before_start_funcs)
|
||||||
|
|
||||||
async def after_start(self) -> None:
|
def after_start(self) -> None:
|
||||||
"""
|
"""
|
||||||
启动后
|
启动后
|
||||||
Returns:
|
Returns:
|
||||||
"""
|
"""
|
||||||
logger.debug("Running after_start functions")
|
logger.debug("Running after_start functions")
|
||||||
await self._run_funcs(self._after_start_funcs)
|
self._run_funcs(self._after_start_funcs)
|
||||||
|
|
||||||
async def before_process_shutdown(self) -> None:
|
def before_process_shutdown(self) -> None:
|
||||||
"""
|
"""
|
||||||
停止前
|
停止前
|
||||||
Returns:
|
Returns:
|
||||||
"""
|
"""
|
||||||
logger.debug("Running before_shutdown functions")
|
logger.debug("Running before_shutdown functions")
|
||||||
await self._run_funcs(self._before_process_shutdown_funcs)
|
self._run_funcs(self._before_process_shutdown_funcs)
|
||||||
|
|
||||||
async def after_shutdown(self) -> None:
|
def after_shutdown(self) -> None:
|
||||||
"""
|
"""
|
||||||
停止后
|
停止后
|
||||||
Returns:
|
Returns:
|
||||||
"""
|
"""
|
||||||
logger.debug("Running after_shutdown functions")
|
logger.debug("Running after_shutdown functions")
|
||||||
await self._run_funcs(self._after_shutdown_funcs)
|
self._run_funcs(self._after_shutdown_funcs)
|
||||||
|
|
||||||
async def before_process_restart(self) -> None:
|
def before_process_restart(self) -> None:
|
||||||
"""
|
"""
|
||||||
重启前
|
重启前
|
||||||
Returns:
|
Returns:
|
||||||
"""
|
"""
|
||||||
logger.debug("Running before_restart functions")
|
logger.debug("Running before_restart functions")
|
||||||
await self._run_funcs(self._before_process_restart_funcs)
|
self._run_funcs(self._before_process_restart_funcs)
|
||||||
|
|
||||||
async def after_restart(self) -> None:
|
def after_restart(self) -> None:
|
||||||
"""
|
"""
|
||||||
重启后
|
重启后
|
||||||
Returns:
|
Returns:
|
||||||
|
|
||||||
"""
|
"""
|
||||||
logger.debug("Running after_restart functions")
|
logger.debug("Running after_restart functions")
|
||||||
await self._run_funcs(self._after_restart_funcs)
|
self._run_funcs(self._after_restart_funcs)
|
||||||
|
|
||||||
async def after_nonebot_init(self) -> None:
|
|
||||||
"""
|
|
||||||
NoneBot 初始化后
|
|
||||||
Returns:
|
|
||||||
"""
|
|
||||||
logger.debug("Running after_nonebot_init functions")
|
|
||||||
await self._run_funcs(self._after_nonebot_init_funcs)
|
|
||||||
|
@ -36,53 +36,63 @@ class ProcessManager:
|
|||||||
self.targets: dict[str, tuple[callable, tuple, dict]] = {}
|
self.targets: dict[str, tuple[callable, tuple, dict]] = {}
|
||||||
self.processes: dict[str, Process] = {}
|
self.processes: dict[str, Process] = {}
|
||||||
|
|
||||||
def start(self, name: str, delay: int = 0):
|
def start(self, name: str):
|
||||||
"""
|
"""
|
||||||
开启后自动监控进程,并添加到进程字典中
|
开启后自动监控进程,并添加到进程字典中
|
||||||
Args:
|
Args:
|
||||||
name:
|
name:
|
||||||
delay:
|
|
||||||
|
|
||||||
Returns:
|
Returns:
|
||||||
|
|
||||||
"""
|
"""
|
||||||
if name not in self.targets:
|
if name not in self.targets:
|
||||||
raise KeyError(f"Process {name} not found.")
|
raise KeyError(f"Process {name} not found.")
|
||||||
|
|
||||||
def _start():
|
|
||||||
should_exit = False
|
|
||||||
while not should_exit:
|
|
||||||
chan_active = get_channel(f"{name}-active")
|
chan_active = get_channel(f"{name}-active")
|
||||||
|
|
||||||
|
def _start_process():
|
||||||
process = Process(target=self.targets[name][0], args=self.targets[name][1],
|
process = Process(target=self.targets[name][0], args=self.targets[name][1],
|
||||||
kwargs=self.targets[name][2])
|
kwargs=self.targets[name][2])
|
||||||
self.processes[name] = process
|
self.processes[name] = process
|
||||||
|
|
||||||
process.start()
|
process.start()
|
||||||
while not should_exit:
|
|
||||||
# 0退出 1重启
|
# 启动进程并监听信号
|
||||||
|
_start_process()
|
||||||
|
|
||||||
|
def _start_monitor():
|
||||||
|
while True:
|
||||||
|
try:
|
||||||
data = chan_active.receive()
|
data = chan_active.receive()
|
||||||
if data == 1:
|
if data == 0:
|
||||||
# 重启
|
# 停止
|
||||||
if self.is_process_alive(name):
|
logger.info(f"Stopping process {name}")
|
||||||
logger.info(f"Restarting process {name}")
|
self.bot.lifespan.before_process_shutdown()
|
||||||
asyncio.run(self.bot.lifespan.before_process_shutdown())
|
|
||||||
asyncio.run(self.bot.lifespan.before_process_restart())
|
|
||||||
self.terminate(name)
|
self.terminate(name)
|
||||||
break
|
break
|
||||||
else:
|
elif data == 1:
|
||||||
logger.warning(f"Process {name} is not restartable, cannot restart.")
|
# 重启
|
||||||
|
logger.info(f"Restarting process {name}")
|
||||||
elif data == 0:
|
self.bot.lifespan.before_process_shutdown()
|
||||||
logger.info(f"Stopping process {name}")
|
self.bot.lifespan.before_process_restart()
|
||||||
asyncio.run(self.bot.lifespan.before_process_shutdown())
|
|
||||||
should_exit = True
|
|
||||||
self.terminate(name)
|
self.terminate(name)
|
||||||
|
_start_process()
|
||||||
|
continue
|
||||||
else:
|
else:
|
||||||
logger.warning("Unknown data received, ignored.")
|
logger.warning("Unknown data received, ignored.")
|
||||||
|
except KeyboardInterrupt:
|
||||||
|
logger.info(f"Stopping process {name}")
|
||||||
|
self.bot.lifespan.before_process_shutdown()
|
||||||
|
self.terminate_all()
|
||||||
|
break
|
||||||
|
|
||||||
if delay:
|
threading.Thread(target=_start_monitor).start()
|
||||||
threading.Timer(delay, _start).start()
|
|
||||||
else:
|
def start_all(self):
|
||||||
threading.Thread(target=_start).start()
|
"""
|
||||||
|
启动所有进程
|
||||||
|
"""
|
||||||
|
for name in self.targets:
|
||||||
|
self.start(name)
|
||||||
|
|
||||||
def add_target(self, name: str, target, args: tuple = (), kwargs=None):
|
def add_target(self, name: str, target, args: tuple = (), kwargs=None):
|
||||||
"""
|
"""
|
||||||
@ -100,7 +110,6 @@ class ProcessManager:
|
|||||||
kwargs["chan_active"] = chan_active
|
kwargs["chan_active"] = chan_active
|
||||||
kwargs["chan_passive"] = chan_passive
|
kwargs["chan_passive"] = chan_passive
|
||||||
self.targets[name] = (target, args, kwargs)
|
self.targets[name] = (target, args, kwargs)
|
||||||
|
|
||||||
set_channels(
|
set_channels(
|
||||||
{
|
{
|
||||||
f"{name}-active" : chan_active,
|
f"{name}-active" : chan_active,
|
||||||
@ -128,6 +137,7 @@ class ProcessManager:
|
|||||||
process.join(TIMEOUT)
|
process.join(TIMEOUT)
|
||||||
if process.is_alive():
|
if process.is_alive():
|
||||||
process.kill()
|
process.kill()
|
||||||
|
logger.success(f"Process {name} terminated.")
|
||||||
|
|
||||||
def terminate_all(self):
|
def terminate_all(self):
|
||||||
for name in self.targets:
|
for name in self.targets:
|
||||||
@ -144,5 +154,4 @@ class ProcessManager:
|
|||||||
"""
|
"""
|
||||||
if name not in self.targets:
|
if name not in self.targets:
|
||||||
raise logger.warning(f"Process {name} not found.")
|
raise logger.warning(f"Process {name} not found.")
|
||||||
process = self.processes[name]
|
return self.processes[name].is_alive()
|
||||||
return process.is_alive()
|
|
||||||
|
@ -72,3 +72,18 @@ def path_to_module_name(path: Path) -> str:
|
|||||||
return ".".join(rel_path.parts[:-1])
|
return ".".join(rel_path.parts[:-1])
|
||||||
else:
|
else:
|
||||||
return ".".join(rel_path.parts[:-1] + (rel_path.stem,))
|
return ".".join(rel_path.parts[:-1] + (rel_path.stem,))
|
||||||
|
|
||||||
|
|
||||||
|
def async_wrapper(func: Callable[..., Any]) -> Callable[..., Coroutine]:
|
||||||
|
"""
|
||||||
|
异步包装器
|
||||||
|
Args:
|
||||||
|
func: Sync Callable
|
||||||
|
Returns:
|
||||||
|
Coroutine: Asynchronous Callable
|
||||||
|
"""
|
||||||
|
|
||||||
|
async def wrapper(*args, **kwargs):
|
||||||
|
return func(*args, **kwargs)
|
||||||
|
wrapper.__signature__ = inspect.signature(func)
|
||||||
|
return wrapper
|
||||||
|
@ -21,9 +21,6 @@ __plugin_meta__ = PluginMetadata(
|
|||||||
bot = get_bot()
|
bot = get_bot()
|
||||||
|
|
||||||
|
|
||||||
# nbp_chan = get_channel("nonebot-passive")
|
|
||||||
# mbp_chan = get_channel("melobot-passive")
|
|
||||||
|
|
||||||
@bot.on_before_start
|
@bot.on_before_start
|
||||||
def _():
|
def _():
|
||||||
logger.info("生命周期监控器:准备启动")
|
logger.info("生命周期监控器:准备启动")
|
||||||
@ -40,5 +37,6 @@ def _():
|
|||||||
|
|
||||||
|
|
||||||
@bot.on_after_start
|
@bot.on_after_start
|
||||||
def _():
|
async def _():
|
||||||
|
await asyncio.sleep(6)
|
||||||
logger.info("生命周期监控器:启动完成")
|
logger.info("生命周期监控器:启动完成")
|
||||||
|
@ -11,11 +11,9 @@ Copyright (C) 2020-2024 LiteyukiStudio. All Rights Reserved
|
|||||||
|
|
||||||
import nonebot
|
import nonebot
|
||||||
|
|
||||||
from liteyuki.plugin import PluginMetadata
|
|
||||||
from liteyuki import get_bot
|
|
||||||
from liteyuki.comm import Channel, set_channel
|
from liteyuki.comm import Channel, set_channel
|
||||||
from liteyuki.core import IS_MAIN_PROCESS
|
from liteyuki.core import IS_MAIN_PROCESS
|
||||||
|
from liteyuki.plugin import PluginMetadata
|
||||||
from .nb_utils import adapter_manager, driver_manager
|
from .nb_utils import adapter_manager, driver_manager
|
||||||
|
|
||||||
__plugin_meta__ = PluginMetadata(
|
__plugin_meta__ = PluginMetadata(
|
||||||
@ -54,7 +52,6 @@ if IS_MAIN_PROCESS:
|
|||||||
liteyuki = get_bot()
|
liteyuki = get_bot()
|
||||||
|
|
||||||
|
|
||||||
@liteyuki.on_after_start
|
@liteyuki.on_before_start
|
||||||
def start_run_nonebot():
|
async def start_run_nonebot():
|
||||||
liteyuki.process_manager.add_target(name="nonebot", target=nb_run, args=(), kwargs=liteyuki.config)
|
liteyuki.process_manager.add_target(name="nonebot", target=nb_run, args=(), kwargs=liteyuki.config)
|
||||||
liteyuki.process_manager.start("nonebot")
|
|
@ -6,6 +6,7 @@ from .rt_guide import *
|
|||||||
from .crt_matchers import *
|
from .crt_matchers import *
|
||||||
|
|
||||||
__plugin_meta__ = PluginMetadata(
|
__plugin_meta__ = PluginMetadata(
|
||||||
|
|
||||||
name="CRT生成工具",
|
name="CRT生成工具",
|
||||||
description="一些CRT牌子生成器",
|
description="一些CRT牌子生成器",
|
||||||
usage="我觉得你应该会用",
|
usage="我觉得你应该会用",
|
||||||
|
Loading…
Reference in New Issue
Block a user