- 删除 ValidationQueue 双轨持久化队列,替换为纯内存 AsyncWorkerPool - 引入统一后台任务框架 JobExecutor(Job/CrawlJob/ValidateAllJob) - 新增 PluginRunner 统一插件执行(超时、重试、健康检查、统计) - 重构 SchedulerService 职责收敛为仅定时触发 ValidateAllJob - 使用 AsyncExitStack 重构 lifespan,安全管理长生命周期资源 - 路由层瘦身 50%+,业务异常上抛由全局中间件统一处理 - 实现设置全热更新(WorkerPool 并发、Validator 超时即时生效) - 前端 Store 强制写后重新拉取,消除乐观更新数据不同步 - 删除 queue.py / task_repo.py / task_service.py - 新增 execution 单元测试,全部 85 个测试通过
83 lines
3.2 KiB
Python
83 lines
3.2 KiB
Python
"""插件注册中心 - 显式注册,类型安全,测试友好"""
|
||
import importlib
|
||
import inspect
|
||
import os
|
||
from typing import Dict, List, Type, Optional
|
||
from app.core.plugin_system.base import BaseCrawlerPlugin
|
||
from app.core.log import logger
|
||
|
||
|
||
class PluginRegistry:
|
||
"""插件注册中心"""
|
||
|
||
def __init__(self):
|
||
self._plugins: Dict[str, Type[BaseCrawlerPlugin]] = {}
|
||
self._instances: Dict[str, BaseCrawlerPlugin] = {}
|
||
|
||
def register(self, plugin_cls: Type[BaseCrawlerPlugin]) -> Type[BaseCrawlerPlugin]:
|
||
"""注册一个插件类。支持装饰器语法。"""
|
||
if not inspect.isclass(plugin_cls) or not issubclass(plugin_cls, BaseCrawlerPlugin):
|
||
raise ValueError("Plugin must be a subclass of BaseCrawlerPlugin")
|
||
if not plugin_cls.name:
|
||
raise ValueError(f"Plugin {plugin_cls.__name__} must have a 'name' attribute")
|
||
|
||
self._plugins[plugin_cls.name] = plugin_cls
|
||
logger.info(f"Plugin registered: {plugin_cls.name} ({plugin_cls.__name__})")
|
||
return plugin_cls
|
||
|
||
def get(self, name: str) -> Optional[BaseCrawlerPlugin]:
|
||
"""获取插件实例(懒加载)"""
|
||
if name not in self._instances:
|
||
cls = self._plugins.get(name)
|
||
if cls:
|
||
self._instances[name] = cls()
|
||
return self._instances.get(name)
|
||
|
||
def list_plugins(self) -> List[BaseCrawlerPlugin]:
|
||
"""获取所有已注册插件的实例列表"""
|
||
result = []
|
||
for name in self._plugins:
|
||
instance = self.get(name)
|
||
if instance:
|
||
result.append(instance)
|
||
return result
|
||
|
||
def get_plugin_names(self) -> List[str]:
|
||
return list(self._plugins.keys())
|
||
|
||
def clear(self) -> None:
|
||
"""清空所有已注册插件(主要用于测试)"""
|
||
self._plugins.clear()
|
||
self._instances.clear()
|
||
|
||
def auto_discover(self, package_name: str):
|
||
"""自动扫描指定包下的所有模块并注册其中的插件类。
|
||
注意:为了类型安全和可控性,推荐显式注册。auto_discover 仅作为兼容。"""
|
||
try:
|
||
package = importlib.import_module(package_name)
|
||
package_dir = os.path.dirname(package.__file__)
|
||
except Exception as e:
|
||
logger.error(f"Auto discover failed for package {package_name}: {e}")
|
||
return
|
||
|
||
for filename in os.listdir(package_dir):
|
||
if filename.endswith(".py") and not filename.startswith("__"):
|
||
module_name = f"{package_name}.{filename[:-3]}"
|
||
try:
|
||
module = importlib.import_module(module_name)
|
||
for attr_name in dir(module):
|
||
obj = getattr(module, attr_name)
|
||
if (
|
||
inspect.isclass(obj)
|
||
and issubclass(obj, BaseCrawlerPlugin)
|
||
and obj is not BaseCrawlerPlugin
|
||
and obj not in self._plugins.values()
|
||
):
|
||
self.register(obj)
|
||
except Exception as e:
|
||
logger.error(f"Failed to load module {module_name}: {e}")
|
||
|
||
|
||
# 全局注册中心实例
|
||
registry = PluginRegistry()
|