app.py 96 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360136113621363136413651366136713681369137013711372137313741375137613771378137913801381138213831384138513861387138813891390139113921393139413951396139713981399140014011402140314041405140614071408140914101411141214131414141514161417141814191420142114221423142414251426142714281429143014311432143314341435143614371438143914401441144214431444144514461447144814491450145114521453145414551456145714581459146014611462146314641465146614671468146914701471147214731474147514761477147814791480148114821483148414851486148714881489149014911492149314941495149614971498149915001501150215031504150515061507150815091510151115121513151415151516151715181519152015211522152315241525152615271528152915301531153215331534153515361537153815391540154115421543154415451546154715481549155015511552155315541555155615571558155915601561156215631564156515661567156815691570157115721573157415751576157715781579158015811582158315841585158615871588158915901591159215931594159515961597159815991600160116021603160416051606160716081609161016111612161316141615161616171618161916201621162216231624162516261627162816291630163116321633163416351636163716381639164016411642164316441645164616471648164916501651165216531654165516561657165816591660166116621663166416651666166716681669167016711672167316741675167616771678167916801681168216831684168516861687168816891690169116921693169416951696169716981699170017011702170317041705170617071708170917101711171217131714171517161717171817191720172117221723172417251726172717281729173017311732173317341735173617371738173917401741174217431744174517461747174817491750175117521753175417551756175717581759176017611762176317641765176617671768176917701771177217731774177517761777177817791780178117821783178417851786178717881789179017911792179317941795179617971798179918001801180218031804180518061807180818091810181118121813181418151816181718181819182018211822182318241825182618271828182918301831183218331834183518361837183818391840184118421843184418451846184718481849185018511852185318541855185618571858185918601861186218631864186518661867186818691870187118721873187418751876187718781879188018811882188318841885188618871888188918901891189218931894189518961897189818991900190119021903190419051906190719081909191019111912191319141915191619171918191919201921192219231924192519261927192819291930193119321933193419351936193719381939194019411942194319441945194619471948194919501951195219531954195519561957195819591960196119621963196419651966196719681969197019711972197319741975197619771978197919801981198219831984198519861987198819891990199119921993199419951996199719981999200020012002200320042005200620072008200920102011201220132014201520162017201820192020202120222023202420252026202720282029203020312032203320342035203620372038203920402041204220432044204520462047204820492050205120522053205420552056205720582059206020612062206320642065206620672068206920702071207220732074207520762077207820792080208120822083208420852086208720882089209020912092209320942095209620972098209921002101210221032104210521062107210821092110211121122113211421152116211721182119212021212122212321242125212621272128212921302131213221332134213521362137213821392140214121422143214421452146214721482149215021512152215321542155215621572158215921602161216221632164216521662167216821692170217121722173217421752176217721782179218021812182218321842185218621872188218921902191219221932194219521962197219821992200220122022203220422052206220722082209221022112212221322142215221622172218221922202221222222232224222522262227222822292230223122322233223422352236223722382239224022412242224322442245224622472248224922502251225222532254225522562257225822592260226122622263226422652266226722682269227022712272227322742275227622772278227922802281228222832284228522862287228822892290229122922293229422952296229722982299230023012302230323042305230623072308230923102311231223132314231523162317231823192320232123222323232423252326232723282329233023312332233323342335233623372338233923402341234223432344234523462347234823492350235123522353235423552356235723582359236023612362236323642365236623672368236923702371237223732374237523762377237823792380238123822383238423852386238723882389239023912392239323942395239623972398239924002401240224032404240524062407240824092410241124122413241424152416241724182419242024212422242324242425242624272428242924302431243224332434243524362437243824392440244124422443244424452446244724482449245024512452245324542455245624572458245924602461246224632464246524662467246824692470247124722473247424752476247724782479248024812482248324842485248624872488248924902491249224932494249524962497249824992500250125022503250425052506250725082509251025112512251325142515251625172518251925202521252225232524252525262527252825292530253125322533253425352536253725382539254025412542254325442545254625472548254925502551255225532554255525562557255825592560256125622563256425652566256725682569257025712572257325742575257625772578257925802581258225832584258525862587258825892590259125922593259425952596259725982599260026012602260326042605260626072608260926102611261226132614261526162617261826192620262126222623262426252626262726282629263026312632263326342635263626372638263926402641264226432644264526462647264826492650265126522653265426552656265726582659266026612662266326642665266626672668266926702671267226732674267526762677267826792680268126822683268426852686268726882689269026912692269326942695
  1. from flask import Flask, jsonify, request, abort
  2. from flask_cors import CORS
  3. from flask_socketio import SocketIO, emit
  4. import threading
  5. import time
  6. import json
  7. import os
  8. import logging
  9. import uuid
  10. # 配置文件路径
  11. SERIAL_CONFIG_FILE = os.path.join(os.path.dirname(os.path.abspath(__file__)), 'serial_config.json')
  12. # 导入配置
  13. from config import (
  14. MAX_BUFFER_SIZE,
  15. FLASK_SECRET_KEY,
  16. FLASK_DEBUG,
  17. FLASK_HOST,
  18. FLASK_PORT,
  19. LOG_LEVEL,
  20. LOG_FORMAT,
  21. LOG_FILE,
  22. SOCKETIO_ASYNC_MODE,
  23. SOCKETIO_ALLOWED_ORIGINS,
  24. DEFAULT_FORWARD_SERIAL_TO_MQTT,
  25. DEFAULT_FORWARD_MQTT_TO_SERIAL,
  26. DEFAULT_MQTT_PUBLISH_TOPIC,
  27. SOCKETIO_NAMESPACE_DATA,
  28. SOCKETIO_NAMESPACE_STATUS,
  29. SOCKETIO_NAMESPACE_CONTROL,
  30. ERROR_CODES,
  31. ERROR_MESSAGES,
  32. DEFAULT_CUSTOMER_ID,
  33. DEFAULT_DTU_ID,
  34. DTU_HEARTBEAT_INTERVAL,
  35. DEFAULT_MQTT_TOPIC_PREFIX,
  36. DTU_REGISTRATION_TOPIC,
  37. DTU_STATUS_TOPIC,
  38. DTU_CONTROL_TOPIC,
  39. DTU_RESPONSE_TOPIC,
  40. DTU_EVENT_TOPIC,
  41. DTU_ALARM_TOPIC,
  42. DTU_BROADCAST_TOPIC,
  43. MAX_PANELS,
  44. MAX_PORTS_PER_PANEL
  45. )
  46. # 配置日志
  47. logging_config = {
  48. 'level': getattr(logging, LOG_LEVEL),
  49. 'format': LOG_FORMAT
  50. }
  51. if LOG_FILE:
  52. logging_config['filename'] = LOG_FILE
  53. logging.basicConfig(**logging_config)
  54. logger = logging.getLogger('serial_mqtt_gateway')
  55. from modules.serial_port import SerialPort
  56. from modules.mqtt_client import MQTTClient
  57. from modules.network_config import network_manager
  58. from modules.modbus_rtu import ModbusRTUClient, ANTENNA_ADDRESSES, AddressConfigProtocol, build_broadcast_query, build_confirm_address, build_assign_address
  59. app = Flask(__name__)
  60. app.config['SECRET_KEY'] = FLASK_SECRET_KEY
  61. # 配置CORS以允许nginx代理的前端访问
  62. CORS(app, resources={r"/api/*": {"origins": "*"}, r"/socket.io/*": {"origins": "*"}})
  63. # 初始化SocketIO
  64. socketio = SocketIO(
  65. app,
  66. cors_allowed_origins=SOCKETIO_ALLOWED_ORIGINS,
  67. async_mode=SOCKETIO_ASYNC_MODE,
  68. manage_session=False, # 禁用会话管理以提高性能
  69. ping_timeout=30, # 心跳超时时间
  70. ping_interval=25, # 心跳间隔
  71. logger=FLASK_DEBUG, # 根据Flask调试模式决定是否记录SocketIO日志
  72. engineio_logger=FLASK_DEBUG
  73. )
  74. # 初始化串口和MQTT客户端
  75. serial_client = SerialPort()
  76. mqtt_client = MQTTClient()
  77. modbus_client = ModbusRTUClient(serial_client)
  78. address_config = AddressConfigProtocol(serial_client)
  79. # 转发标志
  80. forward_serial_to_mqtt = DEFAULT_FORWARD_SERIAL_TO_MQTT
  81. forward_mqtt_to_serial = DEFAULT_FORWARD_MQTT_TO_SERIAL
  82. mqtt_publish_topic = DEFAULT_MQTT_PUBLISH_TOPIC
  83. # DTU MQTT协议配置(运行时可修改)
  84. dtu_config = {
  85. 'topic_prefix': DEFAULT_MQTT_TOPIC_PREFIX,
  86. 'customer_id': DEFAULT_CUSTOMER_ID,
  87. 'dtu_id': DEFAULT_DTU_ID,
  88. 'firmware_version': 'v1.0.0',
  89. 'hardware_version': 'v1.0',
  90. 'heartbeat_interval': DTU_HEARTBEAT_INTERVAL,
  91. 'enabled': True # 是否启用DTU MQTT协议
  92. }
  93. # 端口状态追踪(用于事件检测)
  94. port_state = {} # {panel_id: {port_id: {'last_uid': str, 'expected_uid': str, 'alarm_count': int}}}
  95. # 面板配置(从地址配置模块加载)
  96. panel_config = {} # {panel_id: {'address': int, 'position': int, 'panel_uid': str}}
  97. # 数据存储缓冲区
  98. serial_data_buffer = []
  99. mqtt_data_buffer = []
  100. # 状态标志
  101. serial_status = False
  102. mqtt_status = False
  103. # 串口配置保存和加载函数
  104. def save_serial_config(port, baudrate=9600, timeout=1.0, bytesize=8, parity='N', stopbits=1):
  105. """保存串口配置"""
  106. config = {
  107. 'port': port,
  108. 'baudrate': baudrate,
  109. 'timeout': timeout,
  110. 'bytesize': bytesize,
  111. 'parity': parity,
  112. 'stopbits': stopbits
  113. }
  114. try:
  115. with open(SERIAL_CONFIG_FILE, 'w') as f:
  116. json.dump(config, f)
  117. logger.info(f"串口配置已保存: {port} @ {baudrate}")
  118. except Exception as e:
  119. logger.error(f"保存串口配置失败: {e}")
  120. def load_serial_config():
  121. """加载串口配置"""
  122. if os.path.exists(SERIAL_CONFIG_FILE):
  123. try:
  124. with open(SERIAL_CONFIG_FILE, 'r') as f:
  125. config = json.load(f)
  126. logger.info(f"已加载串口配置: {config.get('port')} @ {config.get('baudrate')}")
  127. return config
  128. except Exception as e:
  129. logger.error(f"加载串口配置失败: {e}")
  130. return None
  131. def auto_connect_serial():
  132. """自动连接上次使用的串口"""
  133. config = load_serial_config()
  134. if config and config.get('port'):
  135. port = config.get('port')
  136. baudrate = config.get('baudrate', 9600)
  137. timeout = config.get('timeout', 1.0)
  138. bytesize = config.get('bytesize', 8)
  139. parity = config.get('parity', 'N')
  140. stopbits = config.get('stopbits', 1)
  141. logger.info(f"尝试自动连接串口: {port} @ {baudrate}")
  142. success, message = serial_client.connect(port, baudrate=baudrate, timeout=timeout, bytesize=bytesize, parity=parity, stopbits=stopbits)
  143. if success:
  144. logger.info(f"自动连接串口成功: {port}")
  145. return True
  146. else:
  147. logger.warning(f"自动连接串口失败: {message}")
  148. return False
  149. # 客户端连接管理
  150. connected_clients = {
  151. 'data': set(),
  152. 'status': set(),
  153. 'control': set()
  154. }
  155. # 设置回调函数
  156. def serial_data_handler(data):
  157. """处理串口接收的数据"""
  158. try:
  159. timestamp = time.strftime('%Y-%m-%d %H:%M:%S')
  160. serial_data_buffer.append({
  161. 'timestamp': timestamp,
  162. 'data': data,
  163. 'direction': 'in'
  164. })
  165. if len(serial_data_buffer) > MAX_BUFFER_SIZE:
  166. serial_data_buffer.pop(0)
  167. socketio.emit('serial_data', {
  168. 'timestamp': timestamp,
  169. 'data': data,
  170. 'direction': 'in'
  171. }, namespace=SOCKETIO_NAMESPACE_DATA)
  172. if forward_serial_to_mqtt and mqtt_client.get_status():
  173. success, msg = mqtt_client.publish(mqtt_publish_topic, data)
  174. if not success:
  175. logger.warning(f"串口数据转发到MQTT失败: {msg}")
  176. except Exception as e:
  177. logger.error(f"处理串口数据时出错: {str(e)}")
  178. def serial_send_handler(data):
  179. """处理串口发送的数据"""
  180. try:
  181. timestamp = time.strftime('%Y-%m-%d %H:%M:%S')
  182. serial_data_buffer.append({
  183. 'timestamp': timestamp,
  184. 'data': data,
  185. 'direction': 'out'
  186. })
  187. if len(serial_data_buffer) > MAX_BUFFER_SIZE:
  188. serial_data_buffer.pop(0)
  189. socketio.emit('serial_data', {
  190. 'timestamp': timestamp,
  191. 'data': data,
  192. 'direction': 'out'
  193. }, namespace=SOCKETIO_NAMESPACE_DATA)
  194. except Exception as e:
  195. logger.error(f"处理串口发送数据时出错: {str(e)}")
  196. def serial_status_handler(status):
  197. """处理串口状态变化"""
  198. try:
  199. global serial_status
  200. serial_status = status
  201. # 通过WebSocket广播状态变化
  202. socketio.emit('serial_status', {
  203. 'connected': status
  204. }, namespace=SOCKETIO_NAMESPACE_STATUS)
  205. logger.info(f"串口状态更新: {'已连接' if status else '已断开'}")
  206. except Exception as e:
  207. logger.error(f"处理串口状态时出错: {str(e)}")
  208. def mqtt_data_handler(data):
  209. """处理MQTT接收的数据"""
  210. try:
  211. # 添加到缓冲区
  212. timestamp = time.strftime('%Y-%m-%d %H:%M:%S')
  213. mqtt_data_buffer.append({
  214. 'timestamp': timestamp,
  215. 'topic': data['topic'],
  216. 'payload': data['payload']
  217. })
  218. # 保持缓冲区大小
  219. if len(mqtt_data_buffer) > MAX_BUFFER_SIZE:
  220. mqtt_data_buffer.pop(0)
  221. # 通过WebSocket广播数据
  222. socketio.emit('mqtt_data', {
  223. 'timestamp': timestamp,
  224. 'topic': data['topic'],
  225. 'payload': data['payload']
  226. }, namespace=SOCKETIO_NAMESPACE_DATA)
  227. # 如果启用了转发且串口已连接,转发数据到串口
  228. if forward_mqtt_to_serial and serial_client.get_status():
  229. success, msg = serial_client.send_data(data['payload'])
  230. if not success:
  231. logger.warning(f"MQTT数据转发到串口失败: {msg}")
  232. except Exception as e:
  233. logger.error(f"处理MQTT数据时出错: {str(e)}")
  234. def mqtt_status_handler(status):
  235. """处理MQTT状态变化"""
  236. try:
  237. global mqtt_status
  238. mqtt_status = status
  239. # 通过WebSocket广播状态变化
  240. socketio.emit('mqtt_status', {
  241. 'connected': status
  242. }, namespace=SOCKETIO_NAMESPACE_STATUS)
  243. logger.info(f"MQTT状态更新: {'已连接' if status else '已断开'}")
  244. # 如果MQTT连接成功且启用了DTU协议,发送注册消息和订阅控制主题
  245. if status and dtu_config.get('enabled'):
  246. socketio.sleep(1) # 等待连接稳定
  247. # 发送DTU注册消息
  248. dtu_register()
  249. # 订阅控制主题
  250. control_topic = build_dtu_topic(dtu_config['customer_id'], 'dtu', dtu_config['dtu_id'], 'control')
  251. mqtt_client.subscribe(control_topic)
  252. logger.info(f"已订阅控制主题: {control_topic}")
  253. # 订阅广播主题 (DTU发现 7.14 和批量配置 7.15)
  254. discover_topic = build_dtu_topic('broadcast', 'dtu', 'discover')
  255. config_topic = build_dtu_topic('broadcast', 'dtu', 'config')
  256. mqtt_client.subscribe(discover_topic)
  257. mqtt_client.subscribe(config_topic)
  258. logger.info(f"已订阅广播主题: {discover_topic}, {config_topic}")
  259. except Exception as e:
  260. logger.error(f"处理MQTT状态时出错: {str(e)}")
  261. # ========== DTU MQTT协议处理函数 ==========
  262. def build_dtu_topic(*parts):
  263. """构建DTU MQTT主题"""
  264. prefix = dtu_config.get('topic_prefix', '线架系统')
  265. return '/'.join([prefix] + list(parts))
  266. def dtu_register():
  267. """发送DTU注册消息"""
  268. if not mqtt_client.get_status():
  269. logger.warning("MQTT未连接,无法发送注册消息")
  270. return False
  271. try:
  272. # 加载面板配置
  273. devices = address_config.get_stored_devices()
  274. panels = []
  275. panel_id = 1
  276. for uid_hex, addr in devices.items():
  277. panels.append({
  278. 'panel_id': f"PANEL_{dtu_config['dtu_id']}_{addr}",
  279. 'address': addr,
  280. 'position': panel_id
  281. })
  282. panel_id += 1
  283. payload = {
  284. 'msg_id': f"reg_{int(time.time() * 1000)}",
  285. 'timestamp': int(time.time() * 1000),
  286. 'dtu_id': dtu_config['dtu_id'],
  287. 'type': 'REGISTER',
  288. 'payload': {
  289. 'firmware_version': dtu_config.get('firmware_version', 'v1.0.0'),
  290. 'hardware_version': dtu_config.get('hardware_version', 'v1.0'),
  291. 'panel_count': len(panels),
  292. 'network_type': 'ethernet',
  293. 'signal_strength': None,
  294. 'uptime': int(time.time() * 1000), # 简化处理
  295. 'panels': panels
  296. }
  297. }
  298. topic = build_dtu_topic(dtu_config['customer_id'], 'dtu', dtu_config['dtu_id'], 'register')
  299. success, msg = mqtt_client.publish(topic, json.dumps(payload))
  300. if success:
  301. logger.info(f"DTU注册消息已发送: {topic}")
  302. else:
  303. logger.error(f"DTU注册消息发送失败: {msg}")
  304. return success
  305. except Exception as e:
  306. logger.error(f"发送DTU注册消息失败: {str(e)}")
  307. return False
  308. def handle_broadcast_discover(payload):
  309. """处理广播发现消息 (7.14)
  310. 服务器通过广播主题发送DISCOVER消息,
  311. DTU收到后应立即回应REGISTER消息进行注册
  312. """
  313. try:
  314. logger.info(f"收到广播发现消息: {payload}")
  315. # 检查消息是否是有效的DISCOVER类型
  316. if isinstance(payload, dict):
  317. msg_type = payload.get('type', '')
  318. if msg_type == 'DISCOVER':
  319. # 立即响应注册消息
  320. logger.info("收到DISCOVER广播,响应注册消息")
  321. dtu_register()
  322. # 可选:也发送一次状态消息
  323. socketio.sleep(0.5)
  324. dtu_publish_status()
  325. except Exception as e:
  326. logger.error(f"处理广播发现消息失败: {str(e)}")
  327. def handle_broadcast_config(payload):
  328. """处理广播批量配置消息 (7.15)
  329. 服务器通过广播主题发送CONFIG消息,
  330. 用于批量配置所有在线DTU
  331. """
  332. try:
  333. logger.info(f"收到广播配置消息: {payload}")
  334. if isinstance(payload, dict):
  335. msg_type = payload.get('type', '')
  336. if msg_type == 'CONFIG':
  337. config_data = payload.get('payload', {})
  338. # 处理批量配置
  339. # 这里可以处理各种配置项,如:
  340. # - 上报间隔配置
  341. # - 告警阈值配置
  342. # - 网络配置等
  343. logger.info(f"应用批量配置: {config_data}")
  344. # 可以在这里添加配置应用逻辑
  345. # 例如:更新本地配置、发送确认等
  346. # 发送配置确认响应
  347. response_topic = build_dtu_topic(
  348. dtu_config['customer_id'], 'dtu', dtu_config['dtu_id'], 'status'
  349. )
  350. response_payload = {
  351. 'msg_id': f"cfg_ack_{int(time.time() * 1000)}",
  352. 'timestamp': int(time.time() * 1000),
  353. 'dtu_id': dtu_config['dtu_id'],
  354. 'type': 'CONFIG_ACK',
  355. 'payload': {
  356. 'config_received': True,
  357. 'applied_settings': config_data
  358. }
  359. }
  360. mqtt_client.publish(response_topic, json.dumps(response_payload))
  361. logger.info("配置确认响应已发送")
  362. except Exception as e:
  363. logger.error(f"处理广播配置消息失败: {str(e)}")
  364. def dtu_publish_status(force=False):
  365. """发送DTU状态/心跳消息"""
  366. if not mqtt_client.get_status() or not dtu_config.get('enabled'):
  367. return False
  368. try:
  369. # 统计面板在线状态
  370. panel_online = 0
  371. panel_offline = 0
  372. # 检查串口状态
  373. serial_st = serial_client.get_status()
  374. rs485_status = 'normal' if (isinstance(serial_st, dict) and serial_st.get('connected', False)) else 'error'
  375. payload = {
  376. 'msg_id': f"hbt_{int(time.time() * 1000)}",
  377. 'timestamp': int(time.time() * 1000),
  378. 'dtu_id': dtu_config['dtu_id'],
  379. 'type': 'STATUS',
  380. 'payload': {
  381. 'cpu_usage': None, # 可后续添加
  382. 'memory_usage': None,
  383. 'temperature': None,
  384. 'network_status': 'online',
  385. 'panel_online': panel_online,
  386. 'panel_offline': panel_offline,
  387. 'screen_connected': False,
  388. 'mqtt_connected': mqtt_client.get_status(),
  389. 'rs485_status': rs485_status
  390. }
  391. }
  392. topic = build_dtu_topic(dtu_config['customer_id'], 'dtu', dtu_config['dtu_id'], 'status')
  393. success, _ = mqtt_client.publish(topic, json.dumps(payload))
  394. return success
  395. except Exception as e:
  396. logger.error(f"发送DTU状态消息失败: {str(e)}")
  397. return False
  398. def dtu_publish_event(panel_id, port_id, event_type, jumper_uid, previous_jumper_uid=None):
  399. """发送端口事件消息"""
  400. if not mqtt_client.get_status() or not dtu_config.get('enabled'):
  401. return False
  402. try:
  403. payload = {
  404. 'msg_id': f"evt_{int(time.time() * 1000)}",
  405. 'timestamp': int(time.time() * 1000),
  406. 'dtu_id': dtu_config['dtu_id'],
  407. 'type': 'EVENT',
  408. 'payload': {
  409. 'panel_id': panel_id,
  410. 'port_id': port_id,
  411. 'event_type': event_type,
  412. 'event_id': f"evt_{uuid.uuid4().hex[:8]}",
  413. 'jumper_uid': jumper_uid,
  414. 'previous_jumper_uid': previous_jumper_uid
  415. }
  416. }
  417. topic = build_dtu_topic(dtu_config['customer_id'], 'patchpanel', dtu_config['dtu_id'], panel_id, 'event')
  418. success, _ = mqtt_client.publish(topic, json.dumps(payload))
  419. if success:
  420. logger.info(f"端口事件已发送: {panel_id}:{port_id} - {event_type}")
  421. # 记录到事件历史
  422. port_event_history.append({
  423. 'timestamp': int(time.time() * 1000),
  424. 'panel_id': panel_id,
  425. 'port_id': port_id,
  426. 'event_type': event_type,
  427. 'jumper_uid': jumper_uid,
  428. 'previous_jumper_uid': previous_jumper_uid
  429. })
  430. # 限制历史记录数量
  431. if len(port_event_history) > 1000:
  432. port_event_history[:] = port_event_history[-1000:]
  433. return success
  434. except Exception as e:
  435. logger.error(f"发送端口事件失败: {str(e)}")
  436. return False
  437. def dtu_publish_alarm(panel_id, port_id, alarm_type, expected_jumper_uid, actual_jumper_uid):
  438. """发送非法告警消息"""
  439. if not mqtt_client.get_status() or not dtu_config.get('enabled'):
  440. return False
  441. try:
  442. description = f"端口{port_id}期望跳线{expected_jumper_uid},实际{'未读到' if not actual_jumper_uid else actual_jumper_uid}"
  443. payload = {
  444. 'msg_id': f"alm_{int(time.time() * 1000)}",
  445. 'timestamp': int(time.time() * 1000),
  446. 'dtu_id': dtu_config['dtu_id'],
  447. 'type': 'ALARM',
  448. 'payload': {
  449. 'panel_id': panel_id,
  450. 'port_id': port_id,
  451. 'alarm_type': alarm_type,
  452. 'severity': 'WARNING',
  453. 'expected_jumper_uid': expected_jumper_uid,
  454. 'actual_jumper_uid': actual_jumper_uid,
  455. 'description': description
  456. }
  457. }
  458. topic = build_dtu_topic(dtu_config['customer_id'], 'patchpanel', dtu_config['dtu_id'], panel_id, 'alarm')
  459. success, _ = mqtt_client.publish(topic, json.dumps(payload))
  460. if success:
  461. logger.warning(f"非法告警已发送: {panel_id}:{port_id} - {alarm_type}")
  462. # 记录到事件历史
  463. port_event_history.append({
  464. 'timestamp': int(time.time() * 1000),
  465. 'panel_id': panel_id,
  466. 'port_id': port_id,
  467. 'event_type': 'ALARM',
  468. 'alarm_type': alarm_type,
  469. 'expected_jumper_uid': expected_jumper_uid,
  470. 'actual_jumper_uid': actual_jumper_uid
  471. })
  472. # 限制历史记录数量
  473. if len(port_event_history) > 1000:
  474. port_event_history[:] = port_event_history[-1000:]
  475. return success
  476. except Exception as e:
  477. logger.error(f"发送非法告警失败: {str(e)}")
  478. return False
  479. def dtu_handle_control(topic, payload):
  480. """处理下行控制指令"""
  481. try:
  482. if not dtu_config.get('enabled'):
  483. return
  484. command = payload.get('payload', {}).get('command')
  485. target = payload.get('payload', {}).get('target')
  486. params = payload.get('payload', {}).get('params', {})
  487. logger.info(f"收到控制指令: command={command}, target={target}")
  488. response_payload = {
  489. 'msg_id': payload.get('msg_id'),
  490. 'timestamp': int(time.time() * 1000),
  491. 'dtu_id': dtu_config['dtu_id'],
  492. 'type': 'RESPONSE',
  493. 'payload': {
  494. 'command': command,
  495. 'target': target,
  496. 'success': True,
  497. 'result': {}
  498. }
  499. }
  500. # 处理各命令
  501. if command == 'SET_PORT_LED':
  502. # 设置端口LED
  503. port_id = params.get('port_id')
  504. led_mode = params.get('led_mode', 'OFF')
  505. # LED模式映射
  506. led_mode_map = {'OFF': 0, 'BLINK_RED': 1, 'BLINK_GREEN': 2, 'BLINK_BLUE': 3}
  507. color = led_mode_map.get(led_mode, 0)
  508. # 查找设备地址
  509. device_address = 1 # 默认
  510. for panel_id, cfg in panel_config.items():
  511. if panel_id == target:
  512. device_address = cfg.get('address', 1)
  513. break
  514. result = modbus_client.set_rgb_led(device_address, port_id, color)
  515. response_payload['payload']['success'] = 'error' not in result
  516. response_payload['payload']['result'] = result
  517. elif command == 'QUERY_DTU_STATUS':
  518. # 查询DTU状态
  519. dtu_publish_status(force=True)
  520. elif command == 'SYNC_PORT_MAPPING':
  521. # 同步单端口期望映射
  522. port_id = params.get('port_id')
  523. jumper_uid = params.get('jumper_uid')
  524. if target not in port_state:
  525. port_state[target] = {}
  526. if port_id not in port_state[target]:
  527. port_state[target][port_id] = {'last_uid': None, 'expected_uid': None, 'alarm_count': 0}
  528. port_state[target][port_id]['expected_uid'] = jumper_uid
  529. elif command == 'SYNC_ALL_MAPPING':
  530. # 批量同步期望映射
  531. mappings = params.get('mappings', [])
  532. for mapping in mappings:
  533. panel_id = mapping.get('panel_id')
  534. port_id = mapping.get('port_id')
  535. jumper_uid = mapping.get('jumper_uid')
  536. if panel_id not in port_state:
  537. port_state[panel_id] = {}
  538. if port_id not in port_state[panel_id]:
  539. port_state[panel_id][port_id] = {'last_uid': None, 'expected_uid': None, 'alarm_count': 0}
  540. port_state[panel_id][port_id]['expected_uid'] = jumper_uid
  541. elif command == 'REBOOT':
  542. # 重启DTU(模拟)
  543. response_payload['payload']['result'] = {'message': 'Reboot command received'}
  544. elif command == 'READ_PANEL_STATUS':
  545. # 读取面板状态
  546. # 读取面板的多个寄存器获取状态信息
  547. # 寄存器地址定义: 0x0000=运行状态, 0x0001=LED控制, 0x0002-0x0009=天线卡状态
  548. device_address = 1
  549. for panel_id, cfg in panel_config.items():
  550. if panel_id == target:
  551. device_address = cfg.get('address', 1)
  552. break
  553. # 读取面板状态寄存器 (地址0x0000开始,读取10个寄存器)
  554. status_result = modbus_client.read_holding_registers(device_address, 0x0000, 10)
  555. # 解析状态
  556. registers = status_result.get('registers', [])
  557. panel_status = {
  558. 'device_address': device_address,
  559. 'run_status': registers[0] if len(registers) > 0 else None,
  560. 'led_control': registers[1] if len(registers) > 1 else None,
  561. 'antenna_status': registers[2:10] if len(registers) >= 10 else [],
  562. 'raw_registers': registers
  563. }
  564. response_payload['payload']['success'] = 'error' not in status_result
  565. response_payload['payload']['result'] = panel_status
  566. elif command == 'QUERY_JUMPER_STATUS':
  567. # 查询跳线器状态
  568. # 跳线器状态寄存器: 0x0100=跳线连接状态, 0x0101=跳线ID高字节, 0x0102=跳线ID低字节
  569. device_address = 1
  570. for panel_id, cfg in panel_config.items():
  571. if panel_id == target:
  572. device_address = cfg.get('address', 1)
  573. break
  574. # 读取跳线器状态寄存器 (地址0x0100开始,读取3个寄存器)
  575. jumper_result = modbus_client.read_holding_registers(device_address, 0x0100, 3)
  576. registers = jumper_result.get('registers', [])
  577. jumper_status = {
  578. 'device_address': device_address,
  579. 'connected': bool(registers[0]) if len(registers) > 0 else False,
  580. 'jumper_id_high': registers[1] if len(registers) > 1 else 0,
  581. 'jumper_id_low': registers[2] if len(registers) > 2 else 0,
  582. 'jumper_id': (registers[1] << 16 | registers[2]) if len(registers) > 2 else 0,
  583. 'raw_registers': registers
  584. }
  585. response_payload['payload']['success'] = 'error' not in jumper_result
  586. response_payload['payload']['result'] = jumper_status
  587. elif command == 'QUERY_ENV_SENSOR':
  588. # 查询环境传感器
  589. # 环境传感器寄存器: 0x0200=温度, 0x0201=湿度, 0x0202=光照, 0x0203=烟雾
  590. device_address = 1
  591. for panel_id, cfg in panel_config.items():
  592. if panel_id == target:
  593. device_address = cfg.get('address', 1)
  594. break
  595. # 读取环境传感器寄存器 (地址0x0200开始,读取4个寄存器)
  596. sensor_result = modbus_client.read_holding_registers(device_address, 0x0200, 4)
  597. registers = sensor_result.get('registers', [])
  598. # 温度和湿度为有符号16位整数,需要除以100转换为实际值
  599. env_sensor = {
  600. 'device_address': device_address,
  601. 'temperature': registers[0] / 100.0 if len(registers) > 0 else None,
  602. 'humidity': registers[1] / 100.0 if len(registers) > 1 else None,
  603. 'light_level': registers[2] if len(registers) > 2 else None,
  604. 'smoke_detected': bool(registers[3]) if len(registers) > 3 else False,
  605. 'raw_registers': registers
  606. }
  607. response_payload['payload']['success'] = 'error' not in sensor_result
  608. response_payload['payload']['result'] = env_sensor
  609. elif command == 'OTA_CANCEL':
  610. # 取消OTA升级
  611. # 向OTA控制寄存器写入取消命令 (0x0000=取消)
  612. device_address = 1
  613. for panel_id, cfg in panel_config.items():
  614. if panel_id == target:
  615. device_address = cfg.get('address', 1)
  616. break
  617. # OTA控制寄存器地址: 0x0300
  618. # 值: 0x0000=取消升级
  619. cancel_result = modbus_client.write_single_register(device_address, 0x0300, 0x0000)
  620. response_payload['payload']['success'] = 'error' not in cancel_result
  621. response_payload['payload']['result'] = {
  622. 'message': 'OTA upgrade cancelled' if 'error' not in cancel_result else 'Failed to cancel OTA',
  623. 'raw_result': cancel_result
  624. }
  625. # 发送响应
  626. response_topic = build_dtu_topic(dtu_config['customer_id'], 'dtu', dtu_config['dtu_id'], 'response')
  627. mqtt_client.publish(response_topic, json.dumps(response_payload))
  628. except Exception as e:
  629. logger.error(f"处理控制指令失败: {str(e)}")
  630. # 启动DTU心跳定时器
  631. def start_dtu_heartbeat():
  632. """启动DTU心跳定时任务"""
  633. def heartbeat_task():
  634. while True:
  635. socketio.sleep(dtu_config.get('heartbeat_interval', DTU_HEARTBEAT_INTERVAL))
  636. if mqtt_client.get_status() and dtu_config.get('enabled'):
  637. dtu_publish_status()
  638. socketio.start_background_task(target=heartbeat_task)
  639. # 修改mqtt_data_handler以处理控制指令
  640. def mqtt_data_handler_extended(data):
  641. """处理MQTT接收的数据(扩展版,含DTU协议)"""
  642. try:
  643. topic = data.get('topic', '')
  644. payload_str = data.get('payload', '')
  645. # 尝试解析JSON
  646. try:
  647. payload = json.loads(payload_str) if isinstance(payload_str, str) else payload_str
  648. except:
  649. payload = payload_str
  650. # 添加到缓冲区
  651. timestamp = time.strftime('%Y-%m-%d %H:%M:%S')
  652. mqtt_data_buffer.append({
  653. 'timestamp': timestamp,
  654. 'topic': topic,
  655. 'payload': payload_str
  656. })
  657. if len(mqtt_data_buffer) > MAX_BUFFER_SIZE:
  658. mqtt_data_buffer.pop(0)
  659. # 通过WebSocket广播
  660. socketio.emit('mqtt_data', {
  661. 'timestamp': timestamp,
  662. 'topic': topic,
  663. 'payload': payload_str
  664. }, namespace=SOCKETIO_NAMESPACE_DATA)
  665. # 检查是否是控制指令主题
  666. expected_control_topic = build_dtu_topic(dtu_config['customer_id'], 'dtu', dtu_config['dtu_id'], 'control')
  667. if topic == expected_control_topic:
  668. dtu_handle_control(topic, payload)
  669. # 检查是否是状态主题,用于处理OTA状态上报
  670. expected_status_topic = build_dtu_topic(dtu_config['customer_id'], 'dtu', dtu_config['dtu_id'], 'status')
  671. if topic == expected_status_topic and isinstance(payload, dict):
  672. if 'ota_status' in payload or 'firmware_version' in payload:
  673. handle_ota_status(dtu_config['dtu_id'], payload)
  674. # 检查是否是环境传感器数据主题
  675. sensor_topic = build_dtu_topic(dtu_config['customer_id'], 'sensor', dtu_config['dtu_id'], 'data')
  676. if topic == sensor_topic and isinstance(payload, dict):
  677. if 'temperature' in payload or 'humidity' in payload:
  678. update_env_sensor_data(
  679. payload.get('temperature'),
  680. payload.get('humidity'),
  681. payload.get('dtu_temperature'),
  682. payload.get('sensor_update_time')
  683. )
  684. # 检查是否是广播主题 - DTU发现 (7.14)
  685. discover_topic = build_dtu_topic('broadcast', 'dtu', 'discover')
  686. if topic == discover_topic:
  687. handle_broadcast_discover(payload)
  688. # 检查是否是广播主题 - 批量配置 (7.15)
  689. config_topic = build_dtu_topic('broadcast', 'dtu', 'config')
  690. if topic == config_topic:
  691. handle_broadcast_config(payload)
  692. # 转发到串口(如果启用)
  693. if forward_mqtt_to_serial and serial_client.get_status():
  694. success, msg = serial_client.send_data(payload_str)
  695. if not success:
  696. logger.warning(f"MQTT数据转发到串口失败: {msg}")
  697. except Exception as e:
  698. logger.error(f"处理MQTT数据时出错: {str(e)}")
  699. # WebSocket事件处理
  700. @socketio.on('connect', namespace=SOCKETIO_NAMESPACE_DATA)
  701. def handle_data_connect():
  702. """处理数据命名空间的连接"""
  703. try:
  704. client_id = str(uuid.uuid4())
  705. connected_clients['data'].add(client_id)
  706. logger.info(f'客户端已连接到数据命名空间,当前连接数: {len(connected_clients["data"])}')
  707. # 发送当前的缓冲区数据,限制发送的历史记录数量
  708. max_history = 100 # 限制发送的历史记录数量以提高性能
  709. emit('serial_data_history', {'data': serial_data_buffer[-max_history:]})
  710. emit('mqtt_data_history', {'data': mqtt_data_buffer[-max_history:]})
  711. # 存储客户端ID以便断开连接时使用
  712. socketio.start_background_task(target=lambda: None) # 确保上下文可用
  713. except Exception as e:
  714. logger.error(f"处理数据命名空间连接时出错: {str(e)}")
  715. @socketio.on('connect', namespace=SOCKETIO_NAMESPACE_STATUS)
  716. def handle_status_connect():
  717. """处理状态命名空间的连接"""
  718. try:
  719. client_id = str(uuid.uuid4())
  720. connected_clients['status'].add(client_id)
  721. logger.info(f'客户端已连接到状态命名空间,当前连接数: {len(connected_clients["status"])}')
  722. # 发送当前状态
  723. emit('serial_status', {'connected': serial_status})
  724. emit('mqtt_status', {'connected': mqtt_status})
  725. emit('forward_status', {
  726. 'serial_to_mqtt': forward_serial_to_mqtt,
  727. 'mqtt_to_serial': forward_mqtt_to_serial,
  728. 'publish_topic': mqtt_publish_topic
  729. })
  730. except Exception as e:
  731. logger.error(f"处理状态命名空间连接时出错: {str(e)}")
  732. @socketio.on('connect', namespace=SOCKETIO_NAMESPACE_CONTROL)
  733. def handle_control_connect():
  734. """处理控制命名空间的连接"""
  735. try:
  736. client_id = str(uuid.uuid4())
  737. connected_clients['control'].add(client_id)
  738. logger.info(f'客户端已连接到控制命名空间,当前连接数: {len(connected_clients["control"])}')
  739. except Exception as e:
  740. logger.error(f"处理控制命名空间连接时出错: {str(e)}")
  741. @socketio.on('disconnect', namespace=SOCKETIO_NAMESPACE_DATA)
  742. def handle_data_disconnect():
  743. """处理数据命名空间的断开连接"""
  744. try:
  745. # 清理客户端连接记录
  746. # 在实际应用中,可能需要更复杂的逻辑来追踪具体哪个客户端断开了连接
  747. if len(connected_clients['data']) > 0:
  748. # 这里简化处理,实际应该维护session到client_id的映射
  749. connected_clients['data'].pop() # 注意:这是一个简化的实现
  750. logger.info(f'客户端已断开数据命名空间的连接,当前连接数: {len(connected_clients["data"])}')
  751. except Exception as e:
  752. logger.error(f"处理数据命名空间断开连接时出错: {str(e)}")
  753. @socketio.on('disconnect', namespace=SOCKETIO_NAMESPACE_STATUS)
  754. def handle_status_disconnect():
  755. """处理状态命名空间的断开连接"""
  756. try:
  757. # 清理客户端连接记录
  758. if len(connected_clients['status']) > 0:
  759. connected_clients['status'].pop()
  760. logger.info(f'客户端已断开状态命名空间的连接,当前连接数: {len(connected_clients["status"])}')
  761. except Exception as e:
  762. logger.error(f"处理状态命名空间断开连接时出错: {str(e)}")
  763. @socketio.on('disconnect', namespace=SOCKETIO_NAMESPACE_CONTROL)
  764. def handle_control_disconnect():
  765. """处理控制命名空间的断开连接"""
  766. try:
  767. # 清理客户端连接记录
  768. if len(connected_clients['control']) > 0:
  769. connected_clients['control'].pop()
  770. logger.info(f'客户端已断开控制命名空间的连接,当前连接数: {len(connected_clients["control"])}')
  771. except Exception as e:
  772. logger.error(f"处理控制命名空间断开连接时出错: {str(e)}")
  773. @socketio.on('serial_send', namespace=SOCKETIO_NAMESPACE_CONTROL)
  774. def handle_serial_send(data):
  775. """通过WebSocket处理串口发送数据请求"""
  776. try:
  777. message = data.get('message', '')
  778. if not message:
  779. emit('serial_send_response', {
  780. 'success': False,
  781. 'message': '消息内容不能为空',
  782. 'error_code': ERROR_CODES['CONFIG_ERROR']
  783. })
  784. return
  785. if not (isinstance(serial_client.get_status(), dict) and serial_client.get_status().get("connected", False)):
  786. emit('serial_send_response', {
  787. 'success': False,
  788. 'message': '串口未连接',
  789. 'error_code': ERROR_CODES['SERIAL_CONNECTION_ERROR']
  790. })
  791. return
  792. success, msg = serial_client.send_data(message)
  793. emit('serial_send_response', {
  794. 'success': success,
  795. 'message': msg,
  796. 'error_code': ERROR_CODES['SUCCESS'] if success else ERROR_CODES['SERIAL_SEND_ERROR']
  797. })
  798. if success:
  799. logger.info(f"通过WebSocket发送串口数据成功: {message[:50]}..." if len(message) > 50 else message)
  800. except Exception as e:
  801. error_msg = f"处理串口发送请求时出错: {str(e)}"
  802. logger.error(error_msg)
  803. emit('serial_send_response', {
  804. 'success': False,
  805. 'message': error_msg,
  806. 'error_code': ERROR_CODES['UNKNOWN_ERROR']
  807. })
  808. @socketio.on('mqtt_publish', namespace=SOCKETIO_NAMESPACE_CONTROL)
  809. def handle_mqtt_publish(data):
  810. """通过WebSocket处理MQTT发布数据请求"""
  811. try:
  812. topic = data.get('topic', '')
  813. message = data.get('message', '')
  814. if not topic or not message:
  815. emit('mqtt_publish_response', {
  816. 'success': False,
  817. 'message': '主题或消息内容不能为空',
  818. 'error_code': ERROR_CODES['CONFIG_ERROR']
  819. })
  820. return
  821. if not mqtt_client.get_status():
  822. emit('mqtt_publish_response', {
  823. 'success': False,
  824. 'message': 'MQTT未连接',
  825. 'error_code': ERROR_CODES['MQTT_CONNECTION_ERROR']
  826. })
  827. return
  828. success, msg = mqtt_client.publish(topic, message)
  829. emit('mqtt_publish_response', {
  830. 'success': success,
  831. 'message': msg,
  832. 'error_code': ERROR_CODES['SUCCESS'] if success else ERROR_CODES['MQTT_PUBLISH_ERROR']
  833. })
  834. if success:
  835. logger.info(f"通过WebSocket发布MQTT消息成功: 主题={topic}, 消息={message[:50]}..." if len(message) > 50 else message)
  836. except Exception as e:
  837. error_msg = f"处理MQTT发布请求时出错: {str(e)}"
  838. logger.error(error_msg)
  839. emit('mqtt_publish_response', {
  840. 'success': False,
  841. 'message': error_msg,
  842. 'error_code': ERROR_CODES['UNKNOWN_ERROR']
  843. })
  844. @socketio.on('update_forward_config', namespace=SOCKETIO_NAMESPACE_CONTROL)
  845. def handle_update_forward_config(data):
  846. """通过WebSocket更新转发配置"""
  847. global forward_serial_to_mqtt, forward_mqtt_to_serial, mqtt_publish_topic
  848. try:
  849. # 更新转发标志
  850. if 'serial_to_mqtt' in data:
  851. forward_serial_to_mqtt = bool(data['serial_to_mqtt'])
  852. if 'mqtt_to_serial' in data:
  853. forward_mqtt_to_serial = bool(data['mqtt_to_serial'])
  854. if 'publish_topic' in data:
  855. new_topic = str(data['publish_topic'])
  856. if not new_topic.strip():
  857. raise ValueError("发布主题不能为空")
  858. mqtt_publish_topic = new_topic
  859. # 广播配置更新
  860. socketio.emit('forward_status', {
  861. 'serial_to_mqtt': forward_serial_to_mqtt,
  862. 'mqtt_to_serial': forward_mqtt_to_serial,
  863. 'publish_topic': mqtt_publish_topic
  864. }, namespace=SOCKETIO_NAMESPACE_STATUS)
  865. emit('update_forward_config_response', {
  866. 'success': True,
  867. 'message': '转发配置已更新',
  868. 'error_code': ERROR_CODES['SUCCESS']
  869. })
  870. logger.info(f"转发配置已更新 - 串口到MQTT: {forward_serial_to_mqtt}, MQTT到串口: {forward_mqtt_to_serial}, 发布主题: {mqtt_publish_topic}")
  871. except ValueError as e:
  872. error_msg = str(e)
  873. logger.warning(f"转发配置更新失败: {error_msg}")
  874. emit('update_forward_config_response', {
  875. 'success': False,
  876. 'message': error_msg,
  877. 'error_code': ERROR_CODES['CONFIG_ERROR']
  878. })
  879. except Exception as e:
  880. error_msg = f'更新转发配置失败: {str(e)}'
  881. logger.error(error_msg)
  882. emit('update_forward_config_response', {
  883. 'success': False,
  884. 'message': error_msg,
  885. 'error_code': ERROR_CODES['UNKNOWN_ERROR']
  886. })
  887. @socketio.on('clear_data_buffer', namespace=SOCKETIO_NAMESPACE_CONTROL)
  888. def handle_clear_data_buffer(data):
  889. """通过WebSocket清空数据缓冲区"""
  890. try:
  891. buffer_type = data.get('type', '')
  892. if buffer_type == 'serial':
  893. serial_data_buffer.clear()
  894. emit('clear_data_buffer_response', {
  895. 'success': True,
  896. 'message': '串口数据缓冲区已清空',
  897. 'error_code': ERROR_CODES['SUCCESS']
  898. })
  899. logger.info("串口数据缓冲区已清空")
  900. elif buffer_type == 'mqtt':
  901. mqtt_data_buffer.clear()
  902. emit('clear_data_buffer_response', {
  903. 'success': True,
  904. 'message': 'MQTT数据缓冲区已清空',
  905. 'error_code': ERROR_CODES['SUCCESS']
  906. })
  907. logger.info("MQTT数据缓冲区已清空")
  908. elif buffer_type == 'all':
  909. serial_data_buffer.clear()
  910. mqtt_data_buffer.clear()
  911. emit('clear_data_buffer_response', {
  912. 'success': True,
  913. 'message': '所有数据缓冲区已清空',
  914. 'error_code': ERROR_CODES['SUCCESS']
  915. })
  916. logger.info("所有数据缓冲区已清空")
  917. else:
  918. emit('clear_data_buffer_response', {
  919. 'success': False,
  920. 'message': '无效的缓冲区类型',
  921. 'error_code': ERROR_CODES['CONFIG_ERROR']
  922. })
  923. logger.warning(f"清空缓冲区失败: 无效的缓冲区类型 '{buffer_type}'")
  924. except Exception as e:
  925. error_msg = f'清空缓冲区失败: {str(e)}'
  926. logger.error(error_msg)
  927. emit('clear_data_buffer_response', {
  928. 'success': False,
  929. 'message': error_msg,
  930. 'error_code': ERROR_CODES['UNKNOWN_ERROR']
  931. })
  932. # 设置回调
  933. serial_client.set_data_callback(serial_data_handler)
  934. serial_client.set_send_callback(serial_send_handler)
  935. serial_client.set_status_callback(serial_status_handler)
  936. mqtt_client.set_data_callback(mqtt_data_handler_extended)
  937. mqtt_client.set_status_callback(mqtt_status_handler)
  938. # 设备配置文件路径
  939. DEVICE_CONFIG_FILE = '/root/dzxj_dtu/devices.json'
  940. confirm_loop_running = False
  941. def save_device_config():
  942. """保存设备配置到文件"""
  943. try:
  944. filepath = DEVICE_CONFIG_FILE
  945. success = address_config.save_config(filepath)
  946. if success:
  947. logger.info(f"设备配置已保存到: {filepath}")
  948. except Exception as e:
  949. logger.error(f"保存设备配置失败: {str(e)}")
  950. def load_device_config():
  951. """加载设备配置"""
  952. try:
  953. filepath = DEVICE_CONFIG_FILE
  954. if os.path.exists(filepath):
  955. success = address_config.load_config(filepath)
  956. if success:
  957. devices = address_config.get_stored_devices()
  958. logger.info(f"已加载 {len(devices)} 个设备配置")
  959. return devices
  960. except Exception as e:
  961. logger.error(f"加载设备配置失败: {str(e)}")
  962. return {}
  963. def confirm_loop():
  964. """后台确认线程:每10秒对所有已存储设备发送 confirm_address"""
  965. global confirm_loop_running
  966. confirm_loop_running = True
  967. logger.info("启动设备确认线程 (间隔10秒)")
  968. while confirm_loop_running:
  969. try:
  970. devices = address_config.get_stored_devices()
  971. if not devices:
  972. time.sleep(10)
  973. continue
  974. if not serial_client.get_status():
  975. time.sleep(10)
  976. continue
  977. _st = serial_client.get_status()
  978. if not (isinstance(_st, dict) and _st.get('connected', False)):
  979. time.sleep(10)
  980. continue
  981. for uid_hex, addr in devices.items():
  982. try:
  983. uid_bytes = bytes.fromhex(uid_hex)
  984. cmd = build_confirm_address(addr, uid_bytes)
  985. success, msg = serial_client.send_raw(cmd)
  986. if success:
  987. logger.debug(f"确认设备: 地址={addr}, UID={uid_hex[:16]}...")
  988. else:
  989. logger.warning(f"确认失败 地址={addr}: {msg}")
  990. except Exception as e:
  991. logger.error(f"确认异常 地址={addr}: {str(e)}")
  992. time.sleep(0.1)
  993. except Exception as e:
  994. logger.error(f"确认线程异常: {str(e)}")
  995. time.sleep(10)
  996. # API路由
  997. # 移除静态文件服务,前端由nginx提供服务
  998. # 根路径路由
  999. @app.route('/')
  1000. def index():
  1001. return send_from_directory(STATIC_FOLDER, 'index.html')
  1002. # 404错误处理器 - 解决SPA路由刷新问题
  1003. @app.errorhandler(404)
  1004. def page_not_found(e):
  1005. """处理所有404错误,对于非API路径返回index.html"""
  1006. path = request.path
  1007. # 检查是否是API请求
  1008. if path.startswith('/api/'):
  1009. # 对于API请求,返回404错误
  1010. return jsonify({
  1011. 'success': False,
  1012. 'message': 'API endpoint not found'
  1013. }), 404
  1014. # 对于所有非API路径,返回index.html让前端路由处理
  1015. return send_from_directory(STATIC_FOLDER, 'index.html'), 200
  1016. @app.route('/api/serial/ports', methods=['GET'])
  1017. def get_serial_ports():
  1018. """获取可用串口列表"""
  1019. ports = serial_client.list_ports()
  1020. return jsonify({
  1021. 'success': True,
  1022. 'ports': ports
  1023. })
  1024. @app.route('/api/serial/connect', methods=['POST'])
  1025. def serial_connect():
  1026. """连接串口"""
  1027. try:
  1028. data = request.json
  1029. port = data.get('port')
  1030. baudrate = data.get('baudrate', 9600)
  1031. bytesize = data.get('bytesize', 8)
  1032. parity = data.get('parity', 'N')
  1033. stopbits = data.get('stopbits', 1)
  1034. timeout = data.get('timeout', 0.1)
  1035. if not port:
  1036. logger.warning("连接串口请求缺少串口名称")
  1037. return jsonify({
  1038. 'success': False,
  1039. 'message': '串口名称不能为空',
  1040. 'error_code': ERROR_CODES['CONFIG_ERROR']
  1041. }), 400
  1042. # 先断开之前的连接
  1043. _status = serial_client.get_status()
  1044. if isinstance(_status, dict) and _status.get('connected', False):
  1045. _port = serial_client.current_config.port if serial_client.current_config else 'unknown'
  1046. logger.info(f"断开现有串口连接: {_port}")
  1047. serial_client.disconnect()
  1048. # 连接新的串口
  1049. logger.info(f"尝试连接串口: {port}, 波特率: {baudrate}")
  1050. success, message = serial_client.connect(port, baudrate=baudrate, timeout=timeout, bytesize=bytesize, parity=parity, stopbits=stopbits)
  1051. status_code = 200 if success else 400
  1052. error_code = ERROR_CODES['SUCCESS'] if success else ERROR_CODES['SERIAL_CONNECTION_ERROR']
  1053. response = {
  1054. 'success': success,
  1055. 'message': message,
  1056. 'error_code': error_code
  1057. }
  1058. if success:
  1059. logger.info(f"串口连接成功: {port}")
  1060. # 保存串口配置
  1061. save_serial_config(port, baudrate, timeout, bytesize, parity, stopbits)
  1062. else:
  1063. logger.error(f"串口连接失败: {message}")
  1064. return jsonify(response), status_code
  1065. except Exception as e:
  1066. error_msg = f'连接串口时出错: {str(e)}'
  1067. logger.exception(error_msg) # 使用exception记录完整堆栈
  1068. return jsonify({
  1069. 'success': False,
  1070. 'message': error_msg,
  1071. 'error_code': ERROR_CODES['UNKNOWN_ERROR']
  1072. }), 500
  1073. @app.route('/api/serial/disconnect', methods=['POST'])
  1074. def serial_disconnect():
  1075. """断开串口连接"""
  1076. success, message = serial_client.disconnect()
  1077. return jsonify({
  1078. 'success': success,
  1079. 'message': message
  1080. })
  1081. @app.route('/api/serial/status', methods=['GET'])
  1082. def serial_get_status():
  1083. """获取串口状态"""
  1084. _st = serial_client.get_status()
  1085. saved_config = {}
  1086. try:
  1087. with open(SERIAL_CONFIG_FILE, 'r') as f:
  1088. saved_config = json.load(f)
  1089. except (FileNotFoundError, json.JSONDecodeError):
  1090. pass
  1091. result = {'saved_config': saved_config}
  1092. if isinstance(_st, dict):
  1093. result['connected'] = _st.get('connected', False)
  1094. if result['connected']:
  1095. cfg = _st.get('config')
  1096. result['port'] = cfg.port if cfg else None
  1097. else:
  1098. result['port'] = None
  1099. else:
  1100. result['connected'] = bool(_st)
  1101. result['port'] = None
  1102. return jsonify(result)
  1103. @app.route('/api/serial/send', methods=['POST'])
  1104. def serial_send():
  1105. """发送数据到串口"""
  1106. data = request.json
  1107. message = data.get('message')
  1108. if not message:
  1109. return jsonify({
  1110. 'success': False,
  1111. 'message': '消息内容不能为空'
  1112. })
  1113. success, message = serial_client.send_data(message)
  1114. return jsonify({
  1115. 'success': success,
  1116. 'message': message
  1117. })
  1118. @app.route('/api/mqtt/connect', methods=['POST'])
  1119. def mqtt_connect():
  1120. """连接MQTT服务器"""
  1121. try:
  1122. data = request.json
  1123. host = data.get('broker') or data.get('host', 'localhost')
  1124. port = data.get('port', 1883)
  1125. client_id = data.get('client_id', f'serial_gateway_{int(time.time())}')
  1126. username = data.get('username')
  1127. password = data.get('password')
  1128. keepalive = data.get('keepalive', 60)
  1129. # 先断开之前的连接
  1130. if mqtt_client.get_status():
  1131. logger.info(f"断开现有MQTT连接: {mqtt_client.host}:{mqtt_client.port}")
  1132. mqtt_client.disconnect()
  1133. # 连接新的MQTT服务器
  1134. logger.info(f"尝试连接MQTT服务器: {host}:{port}, 客户端ID: {client_id}")
  1135. success, message = mqtt_client.connect(
  1136. host=host,
  1137. port=port,
  1138. client_id=client_id,
  1139. username=username,
  1140. password=password,
  1141. keepalive=keepalive
  1142. )
  1143. # 如果连接成功,订阅主题
  1144. if success and 'topics' in data:
  1145. mqtt_client.subscribe(data['topics'])
  1146. status_code = 200 if success else 400
  1147. error_code = ERROR_CODES['SUCCESS'] if success else ERROR_CODES['MQTT_CONNECTION_ERROR']
  1148. response = {
  1149. 'success': success,
  1150. 'message': message,
  1151. 'error_code': error_code
  1152. }
  1153. if success:
  1154. logger.info(f"MQTT服务器连接成功: {host}:{port}")
  1155. else:
  1156. logger.error(f"MQTT服务器连接失败: {message}")
  1157. return jsonify(response), status_code
  1158. except Exception as e:
  1159. error_msg = f'连接MQTT服务器时出错: {str(e)}'
  1160. logger.exception(error_msg)
  1161. return jsonify({
  1162. 'success': False,
  1163. 'message': error_msg,
  1164. 'error_code': ERROR_CODES['UNKNOWN_ERROR']
  1165. }), 500
  1166. @app.route('/api/mqtt/disconnect', methods=['POST'])
  1167. def mqtt_disconnect():
  1168. """断开MQTT连接"""
  1169. success, message = mqtt_client.disconnect()
  1170. return jsonify({
  1171. 'success': success,
  1172. 'message': message
  1173. })
  1174. @app.route('/api/mqtt/status', methods=['GET'])
  1175. def mqtt_get_status():
  1176. """获取MQTT状态"""
  1177. _st = mqtt_client.get_status()
  1178. return jsonify({
  1179. 'connected': _st.get('connected', False) if isinstance(_st, dict) else bool(_st)
  1180. })
  1181. @app.route('/api/mqtt/publish', methods=['POST'])
  1182. def mqtt_publish():
  1183. """发布MQTT消息"""
  1184. data = request.json
  1185. topic = data.get('topic')
  1186. message = data.get('message')
  1187. if not topic or not message:
  1188. return jsonify({
  1189. 'success': False,
  1190. 'message': '主题和消息内容不能为空'
  1191. })
  1192. success, message = mqtt_client.publish(topic, message)
  1193. return jsonify({
  1194. 'success': success,
  1195. 'message': message
  1196. })
  1197. @app.route('/api/mqtt/subscribe', methods=['POST'])
  1198. def mqtt_subscribe():
  1199. """订阅MQTT主题"""
  1200. data = request.json
  1201. topics = data.get('topics', [])
  1202. if not topics:
  1203. return jsonify({
  1204. 'success': False,
  1205. 'message': '请至少订阅一个主题'
  1206. })
  1207. success, message = mqtt_client.subscribe(topics)
  1208. return jsonify({
  1209. 'success': success,
  1210. 'message': message
  1211. })
  1212. @app.route('/api/mqtt/broadcast_discover', methods=['POST'])
  1213. def mqtt_broadcast_discover():
  1214. """发送DTU发现广播 (7.14)
  1215. 服务器通过广播主题发送DISCOVER消息,
  1216. 触发所有在线DTU进行注册响应
  1217. """
  1218. data = request.json or {}
  1219. customer_id = data.get('customer_id', dtu_config.get('customer_id', 'default'))
  1220. discover_topic = build_dtu_topic('broadcast', 'dtu', 'discover')
  1221. payload = {
  1222. 'msg_id': f"disc_{int(time.time() * 1000)}",
  1223. 'timestamp': int(time.time() * 1000),
  1224. 'type': 'DISCOVER',
  1225. 'payload': {
  1226. 'customer_id': customer_id,
  1227. 'action': 'register'
  1228. }
  1229. }
  1230. success, message = mqtt_client.publish(discover_topic, json.dumps(payload))
  1231. logger.info(f"发送DTU发现广播: topic={discover_topic}, payload={payload}")
  1232. return jsonify({
  1233. 'success': success,
  1234. 'message': 'DTU发现广播已发送' if success else message,
  1235. 'topic': discover_topic
  1236. })
  1237. @app.route('/api/mqtt/broadcast_config', methods=['POST'])
  1238. def mqtt_broadcast_config():
  1239. """发送DTU批量配置广播 (7.15)
  1240. 服务器通过广播主题发送CONFIG消息,
  1241. 用于批量配置所有在线DTU
  1242. """
  1243. data = request.json or {}
  1244. customer_id = data.get('customer_id', dtu_config.get('customer_id', 'default'))
  1245. config_settings = data.get('config', {})
  1246. config_topic = build_dtu_topic('broadcast', 'dtu', 'config')
  1247. payload = {
  1248. 'msg_id': f"cfg_{int(time.time() * 1000)}",
  1249. 'timestamp': int(time.time() * 1000),
  1250. 'type': 'CONFIG',
  1251. 'payload': {
  1252. 'customer_id': customer_id,
  1253. 'config': config_settings
  1254. }
  1255. }
  1256. success, message = mqtt_client.publish(config_topic, json.dumps(payload))
  1257. logger.info(f"发送DTU批量配置广播: topic={config_topic}, payload={payload}")
  1258. return jsonify({
  1259. 'success': success,
  1260. 'message': 'DTU批量配置广播已发送' if success else message,
  1261. 'topic': config_topic,
  1262. 'config': config_settings
  1263. })
  1264. @app.route('/api/data/serial', methods=['GET'])
  1265. def get_serial_data():
  1266. """获取串口数据"""
  1267. return jsonify({
  1268. 'data': serial_data_buffer
  1269. })
  1270. @app.route('/api/data/mqtt', methods=['GET'])
  1271. def get_mqtt_data():
  1272. """获取MQTT数据"""
  1273. return jsonify({
  1274. 'data': mqtt_data_buffer
  1275. })
  1276. @app.route('/api/forward/config', methods=['POST'])
  1277. def set_forward_config():
  1278. """设置转发配置"""
  1279. global forward_serial_to_mqtt, forward_mqtt_to_serial, mqtt_publish_topic
  1280. data = request.json
  1281. forward_serial_to_mqtt = data.get('serial_to_mqtt', False)
  1282. forward_mqtt_to_serial = data.get('mqtt_to_serial', False)
  1283. if 'publish_topic' in data:
  1284. mqtt_publish_topic = data['publish_topic']
  1285. return jsonify({
  1286. 'success': True,
  1287. 'message': '转发配置已更新',
  1288. 'config': {
  1289. 'serial_to_mqtt': forward_serial_to_mqtt,
  1290. 'mqtt_to_serial': forward_mqtt_to_serial,
  1291. 'publish_topic': mqtt_publish_topic
  1292. }
  1293. })
  1294. @app.route('/api/forward/status', methods=['GET'])
  1295. def get_forward_status():
  1296. """获取转发状态"""
  1297. return jsonify({
  1298. 'serial_to_mqtt': forward_serial_to_mqtt,
  1299. 'mqtt_to_serial': forward_mqtt_to_serial,
  1300. 'publish_topic': mqtt_publish_topic
  1301. })
  1302. # 健康检查端点
  1303. @app.route('/api/health', methods=['GET'])
  1304. def health_check():
  1305. """健康检查端点"""
  1306. try:
  1307. # 获取客户端连接数量
  1308. client_counts = {
  1309. 'data': len(connected_clients['data']),
  1310. 'status': len(connected_clients['status']),
  1311. 'control': len(connected_clients['control'])
  1312. }
  1313. # 获取缓冲区大小
  1314. buffer_sizes = {
  1315. 'serial': len(serial_data_buffer),
  1316. 'mqtt': len(mqtt_data_buffer)
  1317. }
  1318. # 执行系统负载检查
  1319. # 注意:这只是一个简化的负载检查,实际应用中可能需要更复杂的监控
  1320. is_healthy = True
  1321. load_warnings = []
  1322. # 检查缓冲区是否过大
  1323. if buffer_sizes['serial'] > MAX_BUFFER_SIZE * 0.8:
  1324. is_healthy = False
  1325. load_warnings.append(f"串口缓冲区接近最大容量: {buffer_sizes['serial']}/{MAX_BUFFER_SIZE}")
  1326. if buffer_sizes['mqtt'] > MAX_BUFFER_SIZE * 0.8:
  1327. is_healthy = False
  1328. load_warnings.append(f"MQTT缓冲区接近最大容量: {buffer_sizes['mqtt']}/{MAX_BUFFER_SIZE}")
  1329. # 检查WebSocket连接数是否过多
  1330. total_clients = sum(client_counts.values())
  1331. if total_clients > 100: # 设置合理的阈值
  1332. is_healthy = False
  1333. load_warnings.append(f"WebSocket连接数过多: {total_clients}")
  1334. # 尝试获取网络状态信息(使用try-except包装,防止网络模块出错导致健康检查失败)
  1335. network_info = None
  1336. try:
  1337. network_info = network_manager.get_network_status()
  1338. except Exception as e:
  1339. logger.warning(f"获取网络状态时出错: {str(e)}")
  1340. # 不影响整体健康检查,只添加警告
  1341. load_warnings.append(f"网络状态获取失败: {str(e)}")
  1342. response = {
  1343. 'status': 'healthy' if is_healthy else 'warning',
  1344. 'timestamp': time.strftime('%Y-%m-%d %H:%M:%S'),
  1345. 'services': {
  1346. 'serial': serial_status,
  1347. 'mqtt': mqtt_status,
  1348. 'websocket': True
  1349. },
  1350. 'client_counts': client_counts,
  1351. 'buffer_sizes': buffer_sizes,
  1352. 'warnings': load_warnings
  1353. }
  1354. # 仅当获取到网络信息时添加
  1355. if network_info:
  1356. response['network'] = network_info
  1357. return jsonify(response), 200
  1358. except Exception as e:
  1359. # 捕获所有异常,确保健康检查不会返回500错误
  1360. logger.error(f"健康检查端点出错: {str(e)}")
  1361. # 返回一个基础的健康状态,至少显示服务在运行
  1362. return jsonify({
  1363. 'status': 'error',
  1364. 'timestamp': time.strftime('%Y-%m-%d %H:%M:%S'),
  1365. 'error': str(e),
  1366. 'services': {
  1367. 'api': True, # API服务本身是运行的
  1368. 'serial': None,
  1369. 'mqtt': None,
  1370. 'websocket': None
  1371. }
  1372. }), 200 # 仍然返回200,避免健康检查导致的连锁反应
  1373. # 网络配置相关API
  1374. @app.route('/api/network/config', methods=['GET'])
  1375. def get_network_config():
  1376. """获取当前网络配置"""
  1377. try:
  1378. config = network_manager.get_network_config()
  1379. return jsonify(config)
  1380. except Exception as e:
  1381. logger.error(f'获取网络配置失败: {str(e)}')
  1382. return jsonify({'success': False, 'message': f'获取网络配置失败: {str(e)}'}), 500
  1383. @app.route('/api/network/config', methods=['POST'])
  1384. def update_network_config():
  1385. """更新网络配置"""
  1386. try:
  1387. config_data = request.json
  1388. result = network_manager.update_network_config(config_data)
  1389. if result['success']:
  1390. return jsonify(result)
  1391. else:
  1392. return jsonify(result), 400
  1393. except Exception as e:
  1394. logger.error(f'更新网络配置失败: {str(e)}')
  1395. return jsonify({'success': False, 'message': f'更新网络配置失败: {str(e)}'}), 500
  1396. @app.route('/api/network/status', methods=['GET'])
  1397. def get_network_status():
  1398. """获取网络状态信息"""
  1399. try:
  1400. status = network_manager.get_network_status()
  1401. return jsonify(status)
  1402. except Exception as e:
  1403. logger.error(f'获取网络状态失败: {str(e)}')
  1404. return jsonify({'success': False, 'message': f'获取网络状态失败: {str(e)}'}), 500
  1405. @app.route('/api/network/restart', methods=['POST'])
  1406. def restart_network_service():
  1407. """重启网络服务以应用新配置"""
  1408. try:
  1409. result = network_manager.restart_network_service()
  1410. if result['success']:
  1411. return jsonify(result)
  1412. else:
  1413. return jsonify(result), 500
  1414. except Exception as e:
  1415. logger.error(f'重启网络服务失败: {str(e)}')
  1416. return jsonify({'success': False, 'message': f'重启网络服务失败: {str(e)}'}), 500
  1417. # Modbus RTU API
  1418. @app.route('/api/modbus/antenna_addresses', methods=['GET'])
  1419. def get_antenna_addresses():
  1420. """获取天线地址映射表"""
  1421. return jsonify({
  1422. 'success': True,
  1423. 'antennas': {str(k): f"0x{v:04x}" for k, v in ANTENNA_ADDRESSES.items()}
  1424. })
  1425. @app.route('/api/modbus/read_antenna', methods=['POST'])
  1426. def modbus_read_antenna():
  1427. """读取指定天线的卡号
  1428. 请求参数:
  1429. {
  1430. "device_address": 1, // 设备地址 (1-247)
  1431. "antenna": 1, // 天线编号 (1-24)
  1432. "timeout": 1.0 // 可选,超时时间(秒)
  1433. }
  1434. """
  1435. try:
  1436. data = request.json
  1437. device_address = data.get('device_address', 1)
  1438. antenna = data.get('antenna', 1)
  1439. timeout = data.get('timeout')
  1440. if antenna < 1 or antenna > 24:
  1441. return jsonify({
  1442. 'success': False,
  1443. 'message': f'无效的天线编号: {antenna}, 必须是1-24'
  1444. }), 400
  1445. if not (isinstance(serial_client.get_status(), dict) and serial_client.get_status().get("connected", False)):
  1446. return jsonify({
  1447. 'success': False,
  1448. 'message': '串口未连接'
  1449. }), 400
  1450. result = modbus_client.read_antenna_card(
  1451. device_address=device_address,
  1452. antenna_num=antenna,
  1453. timeout=timeout
  1454. )
  1455. if 'error' in result:
  1456. return jsonify({
  1457. 'success': False,
  1458. 'message': result['error'],
  1459. 'raw_data': result.get('raw_data', '')
  1460. }), 400
  1461. return jsonify({
  1462. 'success': True,
  1463. 'data': result
  1464. })
  1465. except Exception as e:
  1466. logger.error(f'读取天线数据失败: {str(e)}')
  1467. return jsonify({
  1468. 'success': False,
  1469. 'message': str(e)
  1470. }), 500
  1471. @app.route('/api/modbus/read_registers', methods=['POST'])
  1472. def modbus_read_registers():
  1473. """读取保持寄存器
  1474. 请求参数:
  1475. {
  1476. "device_address": 1,
  1477. "start_address": 2, // 起始地址 (十六进制如0x0002或十进制如2)
  1478. "quantity": 4, // 寄存器数量
  1479. "timeout": 1.0
  1480. }
  1481. """
  1482. try:
  1483. data = request.json
  1484. device_address = data.get('device_address', 1)
  1485. start_address = data.get('start_address', 0)
  1486. quantity = data.get('quantity', 1)
  1487. timeout = data.get('timeout')
  1488. # 支持十六进制字符串
  1489. if isinstance(start_address, str):
  1490. start_address = int(start_address, 16)
  1491. if isinstance(device_address, str):
  1492. device_address = int(device_address, 16)
  1493. if not (isinstance(serial_client.get_status(), dict) and serial_client.get_status().get("connected", False)):
  1494. return jsonify({
  1495. 'success': False,
  1496. 'message': '串口未连接'
  1497. }), 400
  1498. result = modbus_client.read_holding_registers(
  1499. device_address=device_address,
  1500. start_address=start_address,
  1501. quantity=quantity,
  1502. timeout=timeout
  1503. )
  1504. if 'error' in result:
  1505. return jsonify({
  1506. 'success': False,
  1507. 'message': result['error'],
  1508. 'raw_data': result.get('raw_data', '')
  1509. }), 400
  1510. return jsonify({
  1511. 'success': True,
  1512. 'data': result
  1513. })
  1514. except Exception as e:
  1515. logger.error(f'读取寄存器失败: {str(e)}')
  1516. return jsonify({
  1517. 'success': False,
  1518. 'message': str(e)
  1519. }), 500
  1520. @app.route('/api/modbus/write_register', methods=['POST'])
  1521. def modbus_write_register():
  1522. """写单个寄存器
  1523. 请求参数:
  1524. {
  1525. "device_address": 1,
  1526. "register_address": 1, // 寄存器地址
  1527. "value": 256, // 写入的值
  1528. "timeout": 1.0
  1529. }
  1530. """
  1531. try:
  1532. data = request.json
  1533. device_address = data.get('device_address', 1)
  1534. register_address = data.get('register_address', 1)
  1535. value = data.get('value', 0)
  1536. timeout = data.get('timeout')
  1537. if not (isinstance(serial_client.get_status(), dict) and serial_client.get_status().get("connected", False)):
  1538. return jsonify({
  1539. 'success': False,
  1540. 'message':'串口未连接'
  1541. }), 400
  1542. result = modbus_client.write_single_register(
  1543. device_address=device_address,
  1544. register_address=register_address,
  1545. value=value,
  1546. timeout=timeout
  1547. )
  1548. if 'error' in result:
  1549. return jsonify({
  1550. 'success': False,
  1551. 'message': result['error'],
  1552. 'raw_data': result.get('raw_data', '')
  1553. }), 400
  1554. return jsonify({
  1555. 'success': True,
  1556. 'data': result
  1557. })
  1558. except Exception as e:
  1559. logger.error(f'写寄存器失败: {str(e)}')
  1560. return jsonify({
  1561. 'success': False,
  1562. 'message': str(e)
  1563. }), 500
  1564. @app.route('/api/modbus/set_rgb_led', methods=['POST'])
  1565. def modbus_set_rgb_led():
  1566. """设置RGB灯状态
  1567. 请求参数:
  1568. {
  1569. "device_address": 1,
  1570. "led_number": 1, // 灯编号 (1-24)
  1571. "color": 1, // 颜色: 0=灭, 1=红灯, 2=绿灯, 3=蓝灯
  1572. "timeout": 1.0
  1573. }
  1574. """
  1575. try:
  1576. data = request.json
  1577. device_address = data.get('device_address', 1)
  1578. led_number = data.get('led_number', 1)
  1579. color = data.get('color', 0)
  1580. timeout = data.get('timeout')
  1581. if not (isinstance(serial_client.get_status(), dict) and serial_client.get_status().get("connected", False)):
  1582. return jsonify({
  1583. 'success': False,
  1584. 'message': '串口未连接'
  1585. }), 400
  1586. result = modbus_client.set_rgb_led(
  1587. device_address=device_address,
  1588. led_number=led_number,
  1589. color=color,
  1590. timeout=timeout
  1591. )
  1592. if 'error' in result:
  1593. return jsonify({
  1594. 'success': False,
  1595. 'message': result['error'],
  1596. 'raw_data': result.get('raw_data', '')
  1597. }), 400
  1598. # 更新 LED 状态追踪
  1599. led_states[led_number] = color
  1600. return jsonify({
  1601. 'success': True,
  1602. 'data': result
  1603. })
  1604. except Exception as e:
  1605. logger.error(f'设置RGB灯失败: {str(e)}')
  1606. return jsonify({
  1607. 'success': False,
  1608. 'message': str(e)
  1609. }), 500
  1610. @app.route('/api/modbus/scan', methods=['POST'])
  1611. def modbus_scan_devices():
  1612. """扫描在线设备
  1613. 请求参数:
  1614. {
  1615. "max_address": 247, // 最大设备地址
  1616. "timeout": 0.2 // 单个设备超时时间
  1617. }
  1618. """
  1619. try:
  1620. data = request.json or {}
  1621. max_address = data.get('max_address', 247)
  1622. timeout = data.get('timeout', 0.2)
  1623. if not (isinstance(serial_client.get_status(), dict) and serial_client.get_status().get("connected", False)):
  1624. return jsonify({
  1625. 'success': False,
  1626. 'message': '串口未连接'
  1627. }), 400
  1628. # 异步扫描可能更好,但这里先同步实现
  1629. result = modbus_client.scan_devices(max_address)
  1630. return jsonify({
  1631. 'success': True,
  1632. 'devices': result
  1633. })
  1634. except Exception as e:
  1635. logger.error(f'扫描设备失败: {str(e)}')
  1636. return jsonify({
  1637. 'success': False,
  1638. 'message': str(e)
  1639. }), 500
  1640. # ========== 地址配置协议 API ==========
  1641. @app.route('/api/modbus/broadcast_query', methods=['POST'])
  1642. def modbus_broadcast_query():
  1643. """发送广播查询指令"""
  1644. try:
  1645. data = request.json or {}
  1646. timeout = data.get('timeout')
  1647. if not (isinstance(serial_client.get_status(), dict) and serial_client.get_status().get("connected", False)):
  1648. return jsonify({'success': False, 'message': '串口未连接'}), 400
  1649. responses = address_config.broadcast_query(timeout)
  1650. for r in responses:
  1651. uid = r.get('uid', '').lower()
  1652. if uid and uid not in address_config.get_stored_devices():
  1653. addr = len(address_config.get_stored_devices()) + 1
  1654. address_config.add_stored_device(uid, addr)
  1655. logger.info(f"自动保存发现设备: UID={uid}, 地址={addr}")
  1656. if responses:
  1657. save_device_config()
  1658. return jsonify({'success': True, 'responses': responses, 'count': len(responses)})
  1659. except Exception as e:
  1660. logger.error(f'广播查询失败: {str(e)}')
  1661. return jsonify({'success': False, 'message': str(e)}), 500
  1662. @app.route('/api/modbus/auto_configure', methods=['POST'])
  1663. def modbus_auto_configure():
  1664. """自动配置设备地址"""
  1665. try:
  1666. data = request.json or {}
  1667. timeout = data.get('timeout')
  1668. if not (isinstance(serial_client.get_status(), dict) and serial_client.get_status().get("connected", False)):
  1669. return jsonify({'success': False, 'message': '串口未连接'}), 400
  1670. result = address_config.auto_configure(timeout)
  1671. if result.get('success') and (result.get('discovered', 0) > 0 or result.get('confirmed', 0) > 0 or result.get('assigned', 0) > 0):
  1672. save_device_config()
  1673. return jsonify(result)
  1674. except Exception as e:
  1675. logger.error(f'自动配置失败: {str(e)}')
  1676. return jsonify({'success': False, 'message': str(e)}), 500
  1677. @app.route('/api/modbus/stored_devices', methods=['GET'])
  1678. def get_stored_devices():
  1679. """获取已存储的设备列表"""
  1680. return jsonify({'success': True, 'devices': address_config.get_stored_devices()})
  1681. @app.route('/api/modbus/stored_devices', methods=['POST'])
  1682. def add_stored_device():
  1683. """添加已存储的设备"""
  1684. try:
  1685. data = request.json
  1686. uid = data.get('uid', '').lower()
  1687. address = data.get('address', 1)
  1688. if len(uid) != 24:
  1689. return jsonify({'success': False, 'message': 'UID长度必须是24个十六进制字符'}), 400
  1690. address_config.add_stored_device(uid, address)
  1691. save_device_config()
  1692. return jsonify({'success': True, 'message': f'已添加设备: UID={uid}, 地址={address}'})
  1693. except Exception as e:
  1694. return jsonify({'success': False, 'message': str(e)}), 500
  1695. @app.route('/api/modbus/load_config', methods=['POST'])
  1696. def modbus_load_config():
  1697. """从文件加载设备配置"""
  1698. try:
  1699. data = request.json
  1700. filepath = data.get('filepath', '/tmp/modbus_devices.json')
  1701. success = address_config.load_config(filepath)
  1702. return jsonify({'success': success, 'devices': address_config.get_stored_devices()})
  1703. except Exception as e:
  1704. return jsonify({'success': False, 'message': str(e)}), 500
  1705. @app.route('/api/modbus/save_config', methods=['POST'])
  1706. def modbus_save_config():
  1707. """保存设备配置到文件"""
  1708. try:
  1709. data = request.json
  1710. filepath = data.get('filepath', '/tmp/modbus_devices.json')
  1711. success = address_config.save_config(filepath)
  1712. return jsonify({'success': success, 'message': f'配置已保存到: {filepath}' if success else '保存失败'})
  1713. except Exception as e:
  1714. return jsonify({'success': False, 'message': str(e)}), 500
  1715. @app.route('/api/modbus/confirm_devices', methods=['POST'])
  1716. def confirm_devices():
  1717. """手动触发对所有已存储设备的确认"""
  1718. try:
  1719. devices = address_config.get_stored_devices()
  1720. if not devices:
  1721. return jsonify({'success': False, 'message': '没有已存储的设备'}), 400
  1722. if not (isinstance(serial_client.get_status(), dict) and serial_client.get_status().get("connected", False)):
  1723. return jsonify({'success': False, 'message': '串口未连接'}), 400
  1724. results = []
  1725. for uid_hex, addr in devices.items():
  1726. try:
  1727. uid_bytes = bytes.fromhex(uid_hex)
  1728. cmd = build_confirm_address(addr, uid_bytes)
  1729. success, msg = serial_client.send_raw(cmd)
  1730. logger.info(f"确认设备 地址={addr}: {'成功' if success else '失败 ' + msg}")
  1731. results.append({'address': addr, 'uid': uid_hex, 'success': success, 'message': msg})
  1732. except Exception as e:
  1733. results.append({'address': addr, 'uid': uid_hex, 'success': False, 'message': str(e)})
  1734. time.sleep(0.1)
  1735. return jsonify({'success': True, 'results': results, 'count': len(results)})
  1736. except Exception as e:
  1737. return jsonify({'success': False, 'message': str(e)}), 500
  1738. # 端口状态追踪 - 用于存储每个端口的事件状态
  1739. port_event_history = [] # 存储最近的端口事件
  1740. @app.route('/api/port/status', methods=['GET'])
  1741. def get_port_status():
  1742. """获取所有端口状态"""
  1743. return jsonify({'success': True, 'port_state': port_state})
  1744. @app.route('/api/port/events', methods=['GET'])
  1745. def get_port_events():
  1746. """获取端口事件历史"""
  1747. limit = request.args.get('limit', 100, type=int)
  1748. return jsonify({
  1749. 'success': True,
  1750. 'events': port_event_history[-limit:]
  1751. })
  1752. @app.route('/api/port/clear_events', methods=['POST'])
  1753. def clear_port_events():
  1754. """清除端口事件历史"""
  1755. global port_event_history
  1756. port_event_history = []
  1757. return jsonify({'success': True, 'message': '事件历史已清除'})
  1758. # 设备最后响应时间跟踪
  1759. device_last_seen = {}
  1760. @app.route('/api/panel/status', methods=['GET'])
  1761. def get_panel_status():
  1762. """获取所有面板状态"""
  1763. now = time.time()
  1764. PANEL_OFFLINE_TIMEOUT = 60
  1765. panel_status = {}
  1766. for panel_id, ports in port_state.items():
  1767. port_count = len(ports)
  1768. alarm_count = sum(1 for p in ports.values() if p.get('alarm_count', 0) > 0)
  1769. connected_count = sum(1 for p in ports.values() if p.get('last_uid'))
  1770. panel_status[panel_id] = {
  1771. 'panel_id': panel_id,
  1772. 'port_count': port_count,
  1773. 'connected_count': connected_count,
  1774. 'alarm_count': alarm_count,
  1775. 'status': 'online' if connected_count > 0 else 'offline'
  1776. }
  1777. for panel_id, cfg in panel_config.items():
  1778. last_seen = device_last_seen.get(panel_id, 0)
  1779. is_online = (now - last_seen) < PANEL_OFFLINE_TIMEOUT
  1780. if panel_id in panel_status:
  1781. panel_status[panel_id].update({
  1782. 'address': cfg.get('address'),
  1783. 'position': cfg.get('position'),
  1784. 'panel_uid': cfg.get('panel_uid'),
  1785. 'status': 'online' if is_online else 'offline'
  1786. })
  1787. else:
  1788. panel_status[panel_id] = {
  1789. 'panel_id': panel_id,
  1790. 'address': cfg.get('address'),
  1791. 'position': cfg.get('position'),
  1792. 'panel_uid': cfg.get('panel_uid'),
  1793. 'port_count': 0,
  1794. 'connected_count': 0,
  1795. 'alarm_count': 0,
  1796. 'status': 'online' if is_online else 'offline'
  1797. }
  1798. return jsonify({'success': True, 'panels': panel_status})
  1799. # LED 状态追踪 (内存中跟踪每个灯的最后设置状态)
  1800. led_states = {i: 0 for i in range(1, 25)} # 0=off, 1=red, 2=green, 3=blue
  1801. @app.route('/api/modbus/led_status', methods=['GET'])
  1802. def get_led_status():
  1803. """获取所有 LED 状态"""
  1804. return jsonify({'success': True, 'leds': led_states})
  1805. @app.route('/api/modbus/set_all_leds', methods=['POST'])
  1806. def set_all_leds():
  1807. """批量设置 LED
  1808. 请求: {"device_address": 1, "color": 1}
  1809. 设置所有 24 个 LED 到指定颜色
  1810. """
  1811. try:
  1812. data = request.json
  1813. device_address = data.get('device_address', 1)
  1814. color = data.get('color', 0)
  1815. timeout = data.get('timeout')
  1816. if not (isinstance(serial_client.get_status(), dict) and serial_client.get_status().get("connected", False)):
  1817. return jsonify({'success': False, 'message': '串口未连接'}), 400
  1818. results = []
  1819. for led_num in range(1, 25):
  1820. result = modbus_client.set_rgb_led(
  1821. device_address=device_address,
  1822. led_number=led_num,
  1823. color=color,
  1824. timeout=timeout
  1825. )
  1826. if 'error' in result:
  1827. results.append({'led': led_num, 'success': False, 'error': result['error']})
  1828. else:
  1829. results.append({'led': led_num, 'success': True})
  1830. led_states[led_num] = color
  1831. return jsonify({'success': True, 'results': results})
  1832. except Exception as e:
  1833. logger.error(f'批量设置LED失败: {str(e)}')
  1834. return jsonify({'success': False, 'message': str(e)}), 500
  1835. # 在 set_rgb_led 中更新 LED 状态追踪
  1836. # 不再需要静态文件目录,前端由nginx提供服务
  1837. # ========== DTU MQTT协议配置API ==========
  1838. @app.route('/api/dtu/config', methods=['GET'])
  1839. def get_dtu_config():
  1840. """获取DTU配置"""
  1841. return jsonify({
  1842. 'success': True,
  1843. 'data': dtu_config
  1844. })
  1845. @app.route('/api/dtu/config', methods=['POST'])
  1846. def update_dtu_config():
  1847. """更新DTU配置"""
  1848. try:
  1849. data = request.json
  1850. old_config = dtu_config.copy()
  1851. # 更新配置项
  1852. if 'topic_prefix' in data:
  1853. dtu_config['topic_prefix'] = data['topic_prefix']
  1854. if 'customer_id' in data:
  1855. dtu_config['customer_id'] = data['customer_id']
  1856. if 'dtu_id' in data:
  1857. dtu_config['dtu_id'] = data['dtu_id']
  1858. if 'firmware_version' in data:
  1859. dtu_config['firmware_version'] = data['firmware_version']
  1860. if 'hardware_version' in data:
  1861. dtu_config['hardware_version'] = data['hardware_version']
  1862. if 'heartbeat_interval' in data:
  1863. dtu_config['heartbeat_interval'] = data['heartbeat_interval']
  1864. if 'enabled' in data:
  1865. dtu_config['enabled'] = data['enabled']
  1866. # 如果MQTT已连接且主题配置发生变化,重新订阅
  1867. if mqtt_status and dtu_config.get('enabled'):
  1868. old_control_topic = build_dtu_topic(
  1869. old_config.get('customer_id', DEFAULT_CUSTOMER_ID),
  1870. 'dtu',
  1871. old_config.get('dtu_id', DEFAULT_DTU_ID),
  1872. 'control'
  1873. )
  1874. new_control_topic = build_dtu_topic(
  1875. dtu_config['customer_id'],
  1876. 'dtu',
  1877. dtu_config['dtu_id'],
  1878. 'control'
  1879. )
  1880. if old_control_topic != new_control_topic:
  1881. # 取消旧订阅,订阅新主题
  1882. mqtt_client.unsubscribe(old_control_topic)
  1883. mqtt_client.subscribe(new_control_topic)
  1884. logger.info(f"控制主题已更新: {old_control_topic} -> {new_control_topic}")
  1885. # 重新发送注册消息
  1886. dtu_register()
  1887. return jsonify({
  1888. 'success': True,
  1889. 'message': 'DTU配置已更新',
  1890. 'data': dtu_config
  1891. })
  1892. except Exception as e:
  1893. logger.error(f"更新DTU配置失败: {str(e)}")
  1894. return jsonify({'success': False, 'message': str(e)}), 500
  1895. @app.route('/api/dtu/register', methods=['POST'])
  1896. def manual_dtu_register():
  1897. """手动触发DTU注册"""
  1898. try:
  1899. if not mqtt_status:
  1900. return jsonify({'success': False, 'message': 'MQTT未连接'}), 400
  1901. success = dtu_register()
  1902. if success:
  1903. return jsonify({'success': True, 'message': '注册消息已发送'})
  1904. else:
  1905. return jsonify({'success': False, 'message': '注册消息发送失败'}), 500
  1906. except Exception as e:
  1907. logger.error(f"手动触发DTU注册失败: {str(e)}")
  1908. return jsonify({'success': False, 'message': str(e)}), 500
  1909. @app.route('/api/dtu/status', methods=['GET'])
  1910. def get_dtu_status():
  1911. """获取DTU状态"""
  1912. try:
  1913. # 获取串口状态
  1914. serial_st = serial_client.get_status()
  1915. # 获取面板状态
  1916. devices = address_config.get_stored_devices()
  1917. status = {
  1918. 'dtu_id': dtu_config.get('dtu_id'),
  1919. 'mqtt_connected': mqtt_status,
  1920. 'serial_connected': isinstance(serial_st, dict) and serial_st.get('connected', False),
  1921. 'dtu_enabled': dtu_config.get('enabled', True),
  1922. 'topic_prefix': dtu_config.get('topic_prefix'),
  1923. 'customer_id': dtu_config.get('customer_id'),
  1924. 'panel_count': len(devices),
  1925. 'topics': {
  1926. 'register': build_dtu_topic(dtu_config['customer_id'], 'dtu', dtu_config['dtu_id'], 'register'),
  1927. 'status': build_dtu_topic(dtu_config['customer_id'], 'dtu', dtu_config['dtu_id'], 'status'),
  1928. 'control': build_dtu_topic(dtu_config['customer_id'], 'dtu', dtu_config['dtu_id'], 'control'),
  1929. 'response': build_dtu_topic(dtu_config['customer_id'], 'dtu', dtu_config['dtu_id'], 'response'),
  1930. 'event': build_dtu_topic(dtu_config['customer_id'], 'dtu', dtu_config['dtu_id'], 'event'),
  1931. 'alarm': build_dtu_topic(dtu_config['customer_id'], 'dtu', dtu_config['dtu_id'], 'alarm')
  1932. }
  1933. }
  1934. return jsonify({'success': True, 'data': status})
  1935. except Exception as e:
  1936. logger.error(f"获取DTU状态失败: {str(e)}")
  1937. return jsonify({'success': False, 'message': str(e)}), 500
  1938. @app.route('/api/dtu/control', methods=['POST'])
  1939. def dtu_control():
  1940. """发送DTU控制命令"""
  1941. try:
  1942. data = request.json
  1943. command = data.get('command')
  1944. if command == 'REBOOT':
  1945. return jsonify({'success': True, 'message': '重启命令已发送'})
  1946. return jsonify({'success': False, 'message': f'未知命令: {command}'}), 400
  1947. except Exception as e:
  1948. logger.error(f"DTU控制命令失败: {str(e)}")
  1949. return jsonify({'success': False, 'message': str(e)}), 500
  1950. # OTA状态存储
  1951. ota_status = {
  1952. 'status': 'IDLE', # IDLE, DOWNLOADING, VERIFYING, FLASHING, SUCCESS, FAILED
  1953. 'progress': 0,
  1954. 'firmware_version': None,
  1955. 'target_version': None,
  1956. 'error_code': None,
  1957. 'error_message': None,
  1958. 'last_update': None
  1959. }
  1960. @app.route('/api/dtu/ota_status', methods=['GET'])
  1961. def get_ota_status():
  1962. """获取OTA升级状态"""
  1963. try:
  1964. return jsonify({
  1965. 'success': True,
  1966. 'data': {
  1967. 'current_firmware': dtu_config.get('firmware_version', 'v1.0.0'),
  1968. 'ota_status': ota_status.get('status', 'IDLE'),
  1969. 'ota_progress': ota_status.get('progress', 0),
  1970. 'target_version': ota_status.get('target_version'),
  1971. 'error_code': ota_status.get('error_code'),
  1972. 'error_message': ota_status.get('error_message'),
  1973. 'last_update': ota_status.get('last_update')
  1974. }
  1975. })
  1976. except Exception as e:
  1977. logger.error(f"获取OTA状态失败: {str(e)}")
  1978. return jsonify({'success': False, 'message': str(e)}), 500
  1979. @app.route('/api/dtu/ota_upgrade', methods=['POST'])
  1980. def trigger_ota_upgrade():
  1981. """触发OTA升级"""
  1982. try:
  1983. if not mqtt_status:
  1984. return jsonify({'success': False, 'message': 'MQTT未连接'}), 400
  1985. data = request.json
  1986. if not data:
  1987. return jsonify({'success': False, 'message': '请求体不能为空'}), 400
  1988. # 验证必填字段
  1989. required_fields = ['firmware_url', 'firmware_version', 'file_size', 'checksum', 'checksum_type']
  1990. missing = [f for f in required_fields if not data.get(f)]
  1991. if missing:
  1992. return jsonify({'success': False, 'message': f'缺少必填字段: {", ".join(missing)}'}), 400
  1993. # 构建OTA升级命令
  1994. msg_id = f"ota_{int(time.time() * 1000)}"
  1995. control_topic = build_dtu_topic(
  1996. dtu_config['customer_id'],
  1997. 'dtu',
  1998. dtu_config['dtu_id'],
  1999. 'control'
  2000. )
  2001. payload = {
  2002. 'msg_id': msg_id,
  2003. 'timestamp': int(time.time() * 1000),
  2004. 'dtu_id': dtu_config['dtu_id'],
  2005. 'type': 'CONTROL',
  2006. 'payload': {
  2007. 'command': 'OTA_UPGRADE',
  2008. 'target': 'dtu',
  2009. 'params': {
  2010. 'firmware_url': data['firmware_url'],
  2011. 'firmware_version': data['firmware_version'],
  2012. 'file_size': data['file_size'],
  2013. 'checksum': data['checksum'],
  2014. 'checksum_type': data['checksum_type'],
  2015. 'force_upgrade': data.get('force_upgrade', False)
  2016. }
  2017. }
  2018. }
  2019. # 发布OTA升级命令
  2020. mqtt_client.publish(control_topic, json.dumps(payload))
  2021. logger.info(f"OTA升级命令已发送: {data['firmware_version']}")
  2022. # 更新OTA状态
  2023. ota_status['status'] = 'DOWNLOADING'
  2024. ota_status['progress'] = 0
  2025. ota_status['target_version'] = data['firmware_version']
  2026. ota_status['error_code'] = None
  2027. ota_status['error_message'] = None
  2028. ota_status['last_update'] = time.strftime('%Y-%m-%d %H:%M:%S')
  2029. return jsonify({
  2030. 'success': True,
  2031. 'message': 'OTA升级命令已发送',
  2032. 'data': {
  2033. 'msg_id': msg_id,
  2034. 'target_version': data['firmware_version'],
  2035. 'ota_status': 'DOWNLOADING'
  2036. }
  2037. })
  2038. except Exception as e:
  2039. logger.error(f"触发OTA升级失败: {str(e)}")
  2040. return jsonify({'success': False, 'message': str(e)}), 500
  2041. # 处理OTA状态上报
  2042. def handle_ota_status(dtu_id, payload):
  2043. """处理OTA状态上报"""
  2044. ota_status['status'] = payload.get('ota_status', 'IDLE')
  2045. ota_status['progress'] = payload.get('ota_progress', 0)
  2046. ota_status['firmware_version'] = payload.get('firmware_version')
  2047. ota_status['last_update'] = time.strftime('%Y-%m-%d %H:%M:%S')
  2048. error_code = payload.get('error_code')
  2049. if error_code and error_code != 0:
  2050. ota_status['error_code'] = error_code
  2051. ota_status['error_message'] = payload.get('error_message', get_ota_error_message(error_code))
  2052. ota_status['status'] = 'FAILED'
  2053. elif ota_status['status'] == 'SUCCESS':
  2054. ota_status['error_code'] = None
  2055. ota_status['error_message'] = None
  2056. # 更新DTU配置中的固件版本
  2057. dtu_config['firmware_version'] = ota_status['firmware_version']
  2058. logger.info(f"OTA状态更新: {ota_status['status']}, 进度: {ota_status['progress']}%")
  2059. def get_ota_error_message(error_code):
  2060. """获取OTA错误码对应的错误信息"""
  2061. error_messages = {
  2062. 1021: '已是目标版本',
  2063. 1022: '校验失败',
  2064. 1023: '下载失败',
  2065. 1024: '写入失败',
  2066. 1025: '存储空间不足'
  2067. }
  2068. return error_messages.get(error_code, f'未知错误码: {error_code}')
  2069. # ==================== 环境传感器 API ====================
  2070. # 环境传感器数据存储
  2071. env_sensor_data = {
  2072. 'temperature': None, # 环境温度 (℃)
  2073. 'humidity': None, # 环境湿度 (%)
  2074. 'dtu_temperature': None, # DTU主板温度 (℃)
  2075. 'update_time': None, # 数据更新时间
  2076. 'sensor_update_time': None, # 传感器更新时间
  2077. 'connected': False # 传感器连接状态
  2078. }
  2079. # 环境传感器告警阈值
  2080. env_sensor_threshold = {
  2081. 'temp_high': 45.0, # 环境温度上限 (℃)
  2082. 'temp_low': -10.0, # 环境温度下限 (℃)
  2083. 'humidity_high': 80.0, # 环境湿度上限 (%)
  2084. 'humidity_low': 20.0, # 环境湿度下限 (%)
  2085. 'dtu_temp_high': 70.0 # DTU主板温度上限 (℃)
  2086. }
  2087. # 环境传感器历史数据 (保留最近1000条)
  2088. env_sensor_history = []
  2089. @app.route('/api/sensor/env', methods=['GET'])
  2090. def get_env_sensor_data():
  2091. """获取环境传感器当前数据"""
  2092. return jsonify({
  2093. 'success': True,
  2094. 'data': env_sensor_data
  2095. })
  2096. @app.route('/api/sensor/threshold', methods=['GET', 'POST'])
  2097. def handle_env_sensor_threshold():
  2098. """获取或设置环境传感器告警阈值"""
  2099. if request.method == 'GET':
  2100. return jsonify({
  2101. 'success': True,
  2102. 'data': env_sensor_threshold
  2103. })
  2104. else:
  2105. try:
  2106. data = request.get_json()
  2107. for key in env_sensor_threshold.keys():
  2108. if key in data:
  2109. env_sensor_threshold[key] = float(data[key])
  2110. logger.info(f"更新环境传感器告警阈值: {env_sensor_threshold}")
  2111. return jsonify({
  2112. 'success': True,
  2113. 'message': '阈值设置成功',
  2114. 'data': env_sensor_threshold
  2115. })
  2116. except Exception as e:
  2117. logger.error(f"设置告警阈值失败: {str(e)}")
  2118. return jsonify({'success': False, 'message': str(e)}), 400
  2119. @app.route('/api/sensor/history', methods=['GET'])
  2120. def get_env_sensor_history():
  2121. """获取环境传感器历史数据"""
  2122. try:
  2123. import datetime
  2124. # 获取查询参数
  2125. range_param = request.args.get('range', '24h')
  2126. start_time = request.args.get('start_time')
  2127. end_time = request.args.get('end_time')
  2128. limit = request.args.get('limit', 100, type=int)
  2129. filtered_history = env_sensor_history
  2130. # 解析range参数
  2131. if not start_time and not end_time:
  2132. now = datetime.datetime.now()
  2133. if range_param == '1h':
  2134. start_time = (now - datetime.timedelta(hours=1)).strftime('%Y-%m-%d %H:%M:%S')
  2135. elif range_param == '6h':
  2136. start_time = (now - datetime.timedelta(hours=6)).strftime('%Y-%m-%d %H:%M:%S')
  2137. elif range_param == '24h':
  2138. start_time = (now - datetime.timedelta(hours=24)).strftime('%Y-%m-%d %H:%M:%S')
  2139. elif range_param == '7d':
  2140. start_time = (now - datetime.timedelta(days=7)).strftime('%Y-%m-%d %H:%M:%S')
  2141. elif range_param == '30d':
  2142. start_time = (now - datetime.timedelta(days=30)).strftime('%Y-%m-%d %H:%M:%S')
  2143. # 按时间过滤
  2144. if start_time:
  2145. filtered_history = [h for h in filtered_history if h.get('update_time') >= start_time]
  2146. if end_time:
  2147. filtered_history = [h for h in filtered_history if h.get('update_time') <= end_time]
  2148. # 按时间排序 (新到旧)
  2149. filtered_history = sorted(filtered_history, key=lambda x: x.get('update_time', ''), reverse=True)
  2150. # 限制返回数量
  2151. filtered_history = filtered_history[:limit]
  2152. return jsonify({
  2153. 'success': True,
  2154. 'data': filtered_history,
  2155. 'total': len(filtered_history)
  2156. })
  2157. except Exception as e:
  2158. logger.error(f"获取历史数据失败: {str(e)}")
  2159. return jsonify({'success': False, 'message': str(e)}), 500
  2160. def update_env_sensor_data(temperature, humidity, dtu_temperature, sensor_update_time):
  2161. """更新环境传感器数据 (由MQTT消息触发)"""
  2162. import datetime
  2163. now = datetime.datetime.now().strftime('%Y-%m-%d %H:%M:%S')
  2164. env_sensor_data['temperature'] = temperature
  2165. env_sensor_data['humidity'] = humidity
  2166. env_sensor_data['dtu_temperature'] = dtu_temperature
  2167. env_sensor_data['sensor_update_time'] = sensor_update_time
  2168. env_sensor_data['update_time'] = now
  2169. env_sensor_data['connected'] = True
  2170. # 添加到历史记录
  2171. history_record = {
  2172. 'temperature': temperature,
  2173. 'humidity': humidity,
  2174. 'dtu_temperature': dtu_temperature,
  2175. 'sensor_update_time': sensor_update_time,
  2176. 'update_time': now
  2177. }
  2178. env_sensor_history.append(history_record)
  2179. # 保留最近1000条
  2180. if len(env_sensor_history) > 1000:
  2181. env_sensor_history[:] = env_sensor_history[-1000:]
  2182. # 检查告警
  2183. check_env_sensor_alarms(temperature, humidity, dtu_temperature)
  2184. logger.info(f"环境传感器数据更新: 温度={temperature}℃, 湿度={humidity}%, DTU温度={dtu_temperature}℃")
  2185. def check_env_sensor_alarms(temperature, humidity, dtu_temperature):
  2186. """检查环境传感器告警"""
  2187. alarms = []
  2188. if temperature is not None:
  2189. if temperature > env_sensor_threshold['temp_high']:
  2190. alarms.append({
  2191. 'type': 'temperature_high',
  2192. 'message': f'环境温度过高: {temperature}℃ (阈值: {env_sensor_threshold["temp_high"]}℃)',
  2193. 'level': 'warning'
  2194. })
  2195. elif temperature < env_sensor_threshold['temp_low']:
  2196. alarms.append({
  2197. 'type': 'temperature_low',
  2198. 'message': f'环境温度过低: {temperature}℃ (阈值: {env_sensor_threshold["temp_low"]}℃)',
  2199. 'level': 'warning'
  2200. })
  2201. if humidity is not None:
  2202. if humidity > env_sensor_threshold['humidity_high']:
  2203. alarms.append({
  2204. 'type': 'humidity_high',
  2205. 'message': f'环境湿度过高: {humidity}% (阈值: {env_sensor_threshold["humidity_high"]}%)',
  2206. 'level': 'warning'
  2207. })
  2208. elif humidity < env_sensor_threshold['humidity_low']:
  2209. alarms.append({
  2210. 'type': 'humidity_low',
  2211. 'message': f'环境湿度过低: {humidity}% (阈值: {env_sensor_threshold["humidity_low"]}%)',
  2212. 'level': 'warning'
  2213. })
  2214. if dtu_temperature is not None:
  2215. if dtu_temperature > env_sensor_threshold['dtu_temp_high']:
  2216. alarms.append({
  2217. 'type': 'dtu_temp_high',
  2218. 'message': f'DTU主板温度过高: {dtu_temperature}℃ (阈值: {env_sensor_threshold["dtu_temp_high"]}℃)',
  2219. 'level': 'critical'
  2220. })
  2221. # 发送告警通知
  2222. for alarm in alarms:
  2223. logger.warning(f"环境传感器告警: {alarm['message']}")
  2224. # 可以通过WebSocket发送告警
  2225. socketio.emit('env_sensor_alarm', alarm)
  2226. if __name__ == '__main__':
  2227. try:
  2228. # 启动前的初始化工作
  2229. logger.info('启动串口-MQTT网关服务...')
  2230. logger.info(f"配置信息: 主机={FLASK_HOST}, 端口={FLASK_PORT}, 调试模式={FLASK_DEBUG}")
  2231. # 加载设备配置
  2232. loaded = load_device_config()
  2233. logger.info(f"已加载 {len(loaded)} 个设备配置")
  2234. # 启动自动发现循环
  2235. def auto_discover_loop():
  2236. while True:
  2237. time.sleep(30)
  2238. try:
  2239. _st = serial_client.get_status()
  2240. if not (isinstance(_st, dict) and _st.get('connected', False)):
  2241. continue
  2242. if not dtu_config.get('enabled'):
  2243. continue
  2244. result = address_config.auto_configure(timeout=2.0)
  2245. if result.get('discovered', 0) > 0:
  2246. logger.info(f"自动发现: {result.get('discovered')} 个设备")
  2247. save_device_config()
  2248. now = time.time()
  2249. for uid in address_config.get_stored_devices():
  2250. device_last_seen[uid] = now
  2251. except Exception as e:
  2252. logger.error(f"自动发现异常: {str(e)}")
  2253. import threading
  2254. t = threading.Thread(target=auto_discover_loop, daemon=True)
  2255. t.start()
  2256. logger.info("启动自动发现线程 (间隔30秒)")
  2257. # 自动连接上次使用的串口
  2258. logger.info("尝试自动连接串口...")
  2259. auto_connect_serial()
  2260. # 启动服务
  2261. socketio.run(
  2262. app,
  2263. host=FLASK_HOST,
  2264. port=FLASK_PORT,
  2265. debug=FLASK_DEBUG,
  2266. use_reloader=False, # 禁用重载器以避免重复初始化问题
  2267. log_output=False, # 禁用Flask的日志输出,使用我们自己的日志配置
  2268. allow_unsafe_werkzeug=True
  2269. )
  2270. except KeyboardInterrupt:
  2271. # 优雅退出
  2272. logger.info('正在关闭应用...')
  2273. try:
  2274. if isinstance(serial_client.get_status(), dict) and serial_client.get_status().get("connected", False):
  2275. serial_client.disconnect()
  2276. logger.info('串口连接已断开')
  2277. if mqtt_client.get_status():
  2278. mqtt_client.disconnect()
  2279. logger.info('MQTT连接已断开')
  2280. except Exception as e:
  2281. logger.error(f'关闭连接时出错: {str(e)}')
  2282. # 清理WebSocket连接
  2283. for client_type in connected_clients:
  2284. connected_clients[client_type].clear()
  2285. logger.info('应用已安全关闭')
  2286. except Exception as e:
  2287. logger.exception(f'应用启动失败') # 使用exception记录完整堆栈
  2288. # 确保资源被释放
  2289. try:
  2290. serial_client.disconnect()
  2291. mqtt_client.disconnect()
  2292. except:
  2293. pass