mirror of
https://github.com/fawney19/Aether.git
synced 2026-01-12 04:28:28 +08:00
203 lines
5.2 KiB
Python
203 lines
5.2 KiB
Python
|
|
"""
|
|||
|
|
插件系统通用定义
|
|||
|
|
包含所有插件类型共享的类和接口
|
|||
|
|
"""
|
|||
|
|
|
|||
|
|
from abc import ABC, abstractmethod
|
|||
|
|
from dataclasses import dataclass
|
|||
|
|
from enum import Enum
|
|||
|
|
from typing import Any, Dict, List, Optional
|
|||
|
|
|
|||
|
|
from src.core.logger import logger
|
|||
|
|
|
|||
|
|
|
|||
|
|
class HealthStatus(Enum):
|
|||
|
|
"""插件健康状态"""
|
|||
|
|
|
|||
|
|
HEALTHY = "healthy"
|
|||
|
|
DEGRADED = "degraded"
|
|||
|
|
UNHEALTHY = "unhealthy"
|
|||
|
|
|
|||
|
|
|
|||
|
|
@dataclass
|
|||
|
|
class PluginMetadata:
|
|||
|
|
"""插件元数据"""
|
|||
|
|
|
|||
|
|
name: str
|
|||
|
|
version: str = "1.0.0"
|
|||
|
|
author: str = "Unknown"
|
|||
|
|
description: str = ""
|
|||
|
|
api_version: str = "1.0"
|
|||
|
|
dependencies: List[str] = None
|
|||
|
|
provides: List[str] = None
|
|||
|
|
|
|||
|
|
def __post_init__(self):
|
|||
|
|
if self.dependencies is None:
|
|||
|
|
self.dependencies = []
|
|||
|
|
if self.provides is None:
|
|||
|
|
self.provides = []
|
|||
|
|
|
|||
|
|
|
|||
|
|
class BasePlugin(ABC):
|
|||
|
|
"""
|
|||
|
|
所有插件的基类
|
|||
|
|
定义插件的基本生命周期和元数据管理
|
|||
|
|
"""
|
|||
|
|
|
|||
|
|
def __init__(
|
|||
|
|
self,
|
|||
|
|
name: str,
|
|||
|
|
priority: int = 0,
|
|||
|
|
version: str = "1.0.0",
|
|||
|
|
author: str = "Unknown",
|
|||
|
|
description: str = "",
|
|||
|
|
api_version: str = "1.0",
|
|||
|
|
dependencies: List[str] = None,
|
|||
|
|
provides: List[str] = None,
|
|||
|
|
config: Dict[str, Any] = None,
|
|||
|
|
):
|
|||
|
|
"""
|
|||
|
|
初始化插件
|
|||
|
|
|
|||
|
|
Args:
|
|||
|
|
name: 插件名称
|
|||
|
|
priority: 优先级(数字越大优先级越高)
|
|||
|
|
version: 插件版本
|
|||
|
|
author: 插件作者
|
|||
|
|
description: 插件描述
|
|||
|
|
api_version: API版本
|
|||
|
|
dependencies: 依赖的其他插件
|
|||
|
|
provides: 提供的服务
|
|||
|
|
config: 配置字典
|
|||
|
|
"""
|
|||
|
|
self.name = name
|
|||
|
|
self.priority = priority
|
|||
|
|
self.enabled = True
|
|||
|
|
self.config = config or {}
|
|||
|
|
self.metadata = PluginMetadata(
|
|||
|
|
name=name,
|
|||
|
|
version=version,
|
|||
|
|
author=author,
|
|||
|
|
description=description,
|
|||
|
|
api_version=api_version,
|
|||
|
|
dependencies=dependencies or [],
|
|||
|
|
provides=provides or [],
|
|||
|
|
)
|
|||
|
|
self._initialized = False
|
|||
|
|
|
|||
|
|
async def initialize(self) -> bool:
|
|||
|
|
"""
|
|||
|
|
初始化插件
|
|||
|
|
|
|||
|
|
Returns:
|
|||
|
|
初始化成功返回True,失败返回False
|
|||
|
|
"""
|
|||
|
|
if self._initialized:
|
|||
|
|
return True
|
|||
|
|
|
|||
|
|
try:
|
|||
|
|
await self._do_initialize()
|
|||
|
|
self._initialized = True
|
|||
|
|
return True
|
|||
|
|
except Exception as e:
|
|||
|
|
logger.warning(f"Failed to initialize plugin {self.name}: {e}")
|
|||
|
|
return False
|
|||
|
|
|
|||
|
|
async def _do_initialize(self):
|
|||
|
|
"""
|
|||
|
|
子类可以重写此方法来实现特定的初始化逻辑
|
|||
|
|
"""
|
|||
|
|
pass
|
|||
|
|
|
|||
|
|
async def shutdown(self):
|
|||
|
|
"""
|
|||
|
|
关闭插件,清理资源
|
|||
|
|
"""
|
|||
|
|
if not self._initialized:
|
|||
|
|
return
|
|||
|
|
|
|||
|
|
try:
|
|||
|
|
await self._do_shutdown()
|
|||
|
|
except Exception as e:
|
|||
|
|
logger.warning(f"Error during plugin {self.name} shutdown: {e}")
|
|||
|
|
finally:
|
|||
|
|
self._initialized = False
|
|||
|
|
|
|||
|
|
async def _do_shutdown(self):
|
|||
|
|
"""
|
|||
|
|
子类可以重写此方法来实现特定的清理逻辑
|
|||
|
|
"""
|
|||
|
|
pass
|
|||
|
|
|
|||
|
|
async def health_check(self) -> HealthStatus:
|
|||
|
|
"""
|
|||
|
|
检查插件健康状态
|
|||
|
|
|
|||
|
|
Returns:
|
|||
|
|
插件健康状态
|
|||
|
|
"""
|
|||
|
|
if not self._initialized or not self.enabled:
|
|||
|
|
return HealthStatus.UNHEALTHY
|
|||
|
|
|
|||
|
|
try:
|
|||
|
|
return await self._do_health_check()
|
|||
|
|
except Exception:
|
|||
|
|
return HealthStatus.UNHEALTHY
|
|||
|
|
|
|||
|
|
async def _do_health_check(self) -> HealthStatus:
|
|||
|
|
"""
|
|||
|
|
子类可以重写此方法来实现特定的健康检查逻辑
|
|||
|
|
默认实现:如果插件已初始化且启用,则认为健康
|
|||
|
|
"""
|
|||
|
|
return (
|
|||
|
|
HealthStatus.HEALTHY if (self._initialized and self.enabled) else HealthStatus.UNHEALTHY
|
|||
|
|
)
|
|||
|
|
|
|||
|
|
def configure(self, config: Dict[str, Any]):
|
|||
|
|
"""
|
|||
|
|
配置插件
|
|||
|
|
|
|||
|
|
Args:
|
|||
|
|
config: 配置字典
|
|||
|
|
"""
|
|||
|
|
self.config.update(config)
|
|||
|
|
self.enabled = config.get("enabled", True)
|
|||
|
|
|
|||
|
|
def get_metadata(self) -> PluginMetadata:
|
|||
|
|
"""
|
|||
|
|
获取插件元数据
|
|||
|
|
|
|||
|
|
Returns:
|
|||
|
|
插件元数据
|
|||
|
|
"""
|
|||
|
|
return self.metadata
|
|||
|
|
|
|||
|
|
@property
|
|||
|
|
def is_initialized(self) -> bool:
|
|||
|
|
"""检查插件是否已初始化"""
|
|||
|
|
return self._initialized
|
|||
|
|
|
|||
|
|
def validate_dependencies(self, available_plugins: Dict[str, List[str]]) -> List[str]:
|
|||
|
|
"""
|
|||
|
|
验证插件依赖是否满足
|
|||
|
|
|
|||
|
|
Args:
|
|||
|
|
available_plugins: 可用插件字典 {plugin_type: [plugin_names]}
|
|||
|
|
|
|||
|
|
Returns:
|
|||
|
|
缺失的依赖列表
|
|||
|
|
"""
|
|||
|
|
missing_deps = []
|
|||
|
|
for dep in self.metadata.dependencies:
|
|||
|
|
found = False
|
|||
|
|
for plugin_type, plugin_names in available_plugins.items():
|
|||
|
|
if dep in plugin_names:
|
|||
|
|
found = True
|
|||
|
|
break
|
|||
|
|
if not found:
|
|||
|
|
missing_deps.append(dep)
|
|||
|
|
return missing_deps
|
|||
|
|
|
|||
|
|
def __repr__(self):
|
|||
|
|
return f"<{self.__class__.__name__}(name={self.name}, priority={self.priority}, enabled={self.enabled}, version={self.metadata.version})>"
|