248 lines
9.0 KiB
Python
Raw Normal View History

2025-09-20 16:50:45 +08:00
#!/usr/bin/env python
# -*- coding: utf-8 -*-
"""
MQTT客户端服务模块
提供MQTT连接发布订阅等功能
支持自动重连和消息处理
"""
import asyncio
import json
import time
import threading
from typing import Dict, Any, Optional, Callable, List
from utils.logger import get_logger
import paho.mqtt.client as mqtt
from config.tf_api_config import MQTT_CONFIG
logger = get_logger("services.mqtt_service")
class MQTTService:
"""内部MQTT客户端服务"""
def __init__(self, config: Dict[str, Any] = None):
self.config = config or MQTT_CONFIG
self.client = None
self.connected = False
self.reconnect_attempts = 0
self.message_handlers: Dict[str, List[Callable]] = {}
self._lock = threading.RLock()
self._running = False
def on_connect(self, client, userdata, flags, rc):
"""连接回调"""
if rc == 0:
self.connected = True
self.reconnect_attempts = 0
logger.info(f"MQTT连接成功: {self.config['host']}:{self.config['port']}")
else:
self.connected = False
logger.error(f"MQTT连接失败错误代码: {rc}")
def on_disconnect(self, client, userdata, rc):
"""断开连接回调"""
self.connected = False
logger.warning(f"MQTT连接断开代码: {rc}")
if self._running and rc != 0:
self._schedule_reconnect()
def on_message(self, client, userdata, msg):
"""消息接收回调"""
try:
topic = msg.topic
payload = msg.payload.decode('utf-8')
with self._lock:
handlers = self.message_handlers.get(topic, [])
for handler in handlers:
try:
if asyncio.iscoroutinefunction(handler):
asyncio.create_task(handler(topic, payload))
else:
handler(topic, payload)
except Exception as e:
logger.error(f"MQTT消息处理器执行失败: {e}", exc_info=True)
except Exception as e:
logger.error(f"处理MQTT消息失败: {e}", exc_info=True)
def _schedule_reconnect(self):
"""安排重连"""
if self.reconnect_attempts >= self.config.get('max_retries', 3):
logger.error("MQTT重连次数已达上限停止重连")
return
delay = self.config.get('reconnect_delay', 5) * (self.reconnect_attempts + 1)
self.reconnect_attempts += 1
logger.info(f"将在 {delay} 秒后重连MQTT{self.reconnect_attempts}次尝试)")
def reconnect():
time.sleep(delay)
if self._running:
self.connect()
threading.Thread(target=reconnect, daemon=True).start()
def connect(self):
"""连接MQTT服务器"""
try:
if self.client:
self.client.disconnect()
self.client = mqtt.Client()
self.client.on_connect = self.on_connect
self.client.on_message = self.on_message
self.client.on_disconnect = self.on_disconnect
# 设置认证信息
if self.config.get('username'):
self.client.username_pw_set(
self.config['username'],
self.config.get('password', '')
)
logger.info(f"正在连接MQTT服务器 {self.config['host']}:{self.config['port']}...")
self.client.connect(
self.config['host'],
self.config['port'],
self.config.get('keepalive', 60)
)
self.client.loop_start()
self._running = True
# 等待连接建立
timeout = 10
start_time = time.time()
while not self.connected and (time.time() - start_time) < timeout:
time.sleep(0.1)
if self.connected:
logger.info("MQTT连接建立成功")
return True
else:
logger.error("MQTT连接建立超时")
return False
except Exception as e:
logger.error(f"连接MQTT服务器失败: {e}")
return False
def disconnect(self):
"""断开连接"""
self._running = False
if self.client:
self.client.loop_stop()
self.client.disconnect()
self.connected = False
logger.info("MQTT连接已断开")
def subscribe(self, topic: str, handler: Callable = None):
"""订阅主题"""
try:
if not self.client or not self.connected:
logger.warning(f"MQTT未连接无法订阅主题: {topic}")
return False
result = self.client.subscribe(topic)
if result[0] == mqtt.MQTT_ERR_SUCCESS:
logger.info(f"成功订阅MQTT主题: {topic}")
# 添加消息处理器
if handler:
with self._lock:
if topic not in self.message_handlers:
self.message_handlers[topic] = []
if handler not in self.message_handlers[topic]:
self.message_handlers[topic].append(handler)
return True
else:
logger.error(f"订阅MQTT主题失败: {topic}, 错误码: {result[0]}")
return False
except Exception as e:
logger.error(f"订阅MQTT主题异常: {e}")
return False
def unsubscribe(self, topic: str):
"""取消订阅主题"""
try:
if self.client and self.connected:
result = self.client.unsubscribe(topic)
if result[0] == mqtt.MQTT_ERR_SUCCESS:
logger.info(f"取消订阅MQTT主题: {topic}")
# 清理消息处理器
with self._lock:
if topic in self.message_handlers:
del self.message_handlers[topic]
return True
else:
logger.error(f"取消订阅MQTT主题失败: {topic}, 错误码: {result[0]}")
return False
except Exception as e:
logger.error(f"取消订阅MQTT主题异常: {e}")
return False
def publish(self, topic: str, payload: Any, qos: int = 0, retain: bool = False):
"""发布消息"""
try:
if not self.client or not self.connected:
logger.warning(f"MQTT未连接无法发布消息到: {topic}")
return False
if isinstance(payload, dict):
payload_str = json.dumps(payload, ensure_ascii=False)
else:
payload_str = str(payload)
result = self.client.publish(topic, payload_str, qos, retain)
if result.rc == mqtt.MQTT_ERR_SUCCESS:
logger.debug(f"MQTT消息发布成功: {topic}")
return True
else:
logger.error(f"MQTT消息发布失败: {topic}, 错误码: {result.rc}")
return False
except Exception as e:
logger.error(f"发布MQTT消息异常: {e}")
return False
def add_message_handler(self, topic: str, handler: Callable):
"""为指定主题添加消息处理器"""
with self._lock:
if topic not in self.message_handlers:
self.message_handlers[topic] = []
if handler not in self.message_handlers[topic]:
self.message_handlers[topic].append(handler)
def remove_message_handler(self, topic: str, handler: Callable):
"""移除指定主题的消息处理器"""
with self._lock:
if topic in self.message_handlers:
if handler in self.message_handlers[topic]:
self.message_handlers[topic].remove(handler)
if not self.message_handlers[topic]:
del self.message_handlers[topic]
def is_connected(self) -> bool:
"""检查连接状态"""
return self.connected
def get_connection_info(self) -> Dict[str, Any]:
"""获取连接信息"""
return {
"host": self.config['host'],
"port": self.config['port'],
"connected": self.connected,
"reconnect_attempts": self.reconnect_attempts,
"running": self._running,
"subscribed_topics": list(self.message_handlers.keys())
}
def create_mqtt_service(config: Dict[str, Any] = None) -> MQTTService:
"""创建MQTT服务实例"""
return MQTTService(config)