app.py 102 KB

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