alarm_plan.py 47 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200
  1. from typing import List, Tuple, Optional
  2. from datetime import datetime, time, date
  3. from threading import Thread, Lock
  4. from enum import Enum
  5. import uuid
  6. import json
  7. import traceback
  8. from datetime import datetime, timezone, timedelta
  9. import common.sys_comm as sys_comm
  10. from common.sys_comm import (
  11. LOGDBG, LOGINFO, LOGWARN, LOGERR,
  12. get_utc_time_ms, get_utc_time_s, get_bj_time_ms, get_bj_time_s,
  13. utc_to_bj_ms, bj_to_utc_ms, utc_to_bj_s, bj_to_utc_s
  14. )
  15. from core.time_plan import TimePlan
  16. from core.event_type import EventType, event_desc_map
  17. import core.g_LAS as g_las
  18. import core.alarm_plan_helper as helper
  19. from core.linkage_action import LinkageAction
  20. from db.db_process import (
  21. db_req_que, DBRequest_Async
  22. )
  23. from db.db_process import (
  24. db_execute_sync, db_execute_async
  25. )
  26. import db.db_sqls as sqls
  27. import mqtt.mqtt_send as mqtt_send
  28. import device.dev_mng as g_Dev
  29. from device.dev_mng import (
  30. Device, g_dev_mgr
  31. )
  32. class EventAttr_Base:
  33. def __init__(self):
  34. return
  35. # 事件属性 事件事件
  36. class EventAttr_StayDetection(EventAttr_Base):
  37. def __init__(self, event_type):
  38. self.event_type_ = event_type
  39. self.enter_ts_: int = -1 # 进入时间(s)
  40. self.leave_ts_: int = -1 # 离开时间(s)
  41. self.stay_time_: int = -1 # 停留时长(s)
  42. return
  43. def reset(self):
  44. self.enter_ts_ = -1
  45. self.leave_ts_ = -1
  46. self.stay_time_ = -1
  47. # 事件属性 滞留事件
  48. class EventAttr_RetentionDetection(EventAttr_Base):
  49. def __init__(self, event_type):
  50. self.event_type_ = event_type
  51. self.enter_ts_: int = -1 # 进入时间(s)
  52. self.leave_ts_: int = -1 # 离开时间(s)
  53. self.stay_time_: int = -1 # 停留时长(s)
  54. return
  55. def reset(self):
  56. self.enter_ts_ = -1
  57. self.leave_ts_ = -1
  58. self.stay_time_ = -1
  59. # 事件属性 如厕事件
  60. class EventAttr_ToiletingDetection(EventAttr_Base):
  61. def __init__(self, event_type):
  62. self.event_type_ = event_type
  63. self.enter_ts_: int = -1 # 进入时间(ms)
  64. self.leave_ts_: int = -1 # 离开时间(ms)
  65. self.stay_time_: int = -1 # 停留时长(ms)
  66. return
  67. def reset(self):
  68. self.enter_ts_ = -1
  69. self.leave_ts_ = -1
  70. self.stay_time_ = -1
  71. # 事件属性 如厕频次统计
  72. class EventAttr_ToiletingFrequency(EventAttr_Base):
  73. def __init__(self, event_type):
  74. self.event_type_ = event_type
  75. self.count_: int = 0 # 统计次数
  76. self.event_list: list = []
  77. return
  78. def reset(self):
  79. self.count_ = 0
  80. self.event_list = []
  81. # 事件属性 夜间如厕频次统计
  82. class EventAttr_NightToiletingFrequency(EventAttr_Base):
  83. def __init__(self, event_type):
  84. self.event_type_ = event_type
  85. self.count_: int = 0 # 统计次数
  86. self.event_list: list = []
  87. return
  88. def reset(self):
  89. self.count_ = 0
  90. self.event_list = []
  91. # 事件属性 如厕频次异常
  92. class EventAttr_ToiletingFrequencyAbnormal(EventAttr_Base):
  93. def __init__(self, event_type):
  94. self.event_type_ = event_type
  95. self.count_: int = 0 # 统计次数
  96. self.threshold_count_: int = 0 # 异常阈值
  97. self.event_list: list = []
  98. return
  99. def reset(self):
  100. self.count_ = 0
  101. self.event_list = []
  102. # 事件属性 起夜异常
  103. class EventAttr_NightToiletingFrequencyAbnormal(EventAttr_Base):
  104. def __init__(self, event_type):
  105. self.event_type_ = event_type
  106. self.count_: int = 0 # 统计次数
  107. self.threshold_count_: int = 0 # 异常阈值
  108. self.event_list: list = []
  109. return
  110. def reset(self):
  111. self.count_ = 0
  112. self.event_list = []
  113. # 事件属性 卫生间频次统计
  114. class EventAttr_BathroomStayFrequency(EventAttr_Base):
  115. def __init__(self, event_type):
  116. self.event_type_ = event_type
  117. self.count_: int = 0 # 统计次数
  118. self.event_list: list = []
  119. return
  120. def reset(self):
  121. self.count_ = 0
  122. self.event_list = []
  123. # 事件属性 异常消失
  124. class EventAttr_TargetAbsence(EventAttr_Base):
  125. def __init__(self, event_type):
  126. self.event_type_ = event_type
  127. self.leave_ts_: int = -1 # 离开时间(ms)
  128. self.enter_ts_: int = -1 # 进入时间(ms)
  129. self.absence_time_: int = -1 # 消失时长(ms)
  130. self.time_threshold_: int = 300 # 触发消失时间阈值(ms)
  131. return
  132. def reset(self):
  133. self.leave_ts_ = -1
  134. self.enter_ts_ = -1
  135. self.absence_time_ = -1
  136. # 事件属性 清理过期事件
  137. class EventAttr_CleanExpireEvents(EventAttr_Base):
  138. def __init__(self, event_type):
  139. self.event_type_ = event_type
  140. self.expire_range_ = 90
  141. # 事件属性表
  142. event_attr_map = {
  143. EventType.STAY_DETECTION.value : EventAttr_StayDetection,
  144. EventType.RETENTION_DETECTION.value : EventAttr_RetentionDetection,
  145. EventType.TOILETING_DETECTION.value : EventAttr_ToiletingDetection,
  146. EventType.TOILETING_FREQUENCY.value : EventAttr_ToiletingFrequency,
  147. EventType.NIGHT_TOILETING_FREQUENCY.value : EventAttr_NightToiletingFrequency,
  148. EventType.TOILETING_FREQUENCY_ABNORMAL.value : EventAttr_ToiletingFrequencyAbnormal,
  149. EventType.NIGHT_TOILETING_FREQUENCY_ABNORMAL.value: EventAttr_NightToiletingFrequencyAbnormal,
  150. EventType.BATHROOM_STAY_FREQUENCY.value : EventAttr_BathroomStayFrequency,
  151. EventType.TARGET_ABSENCE.value : EventAttr_TargetAbsence,
  152. EventType.CLEAN_EXPIRE_EVENTS.value : EventAttr_CleanExpireEvents,
  153. }
  154. class Cron:
  155. def __init__(self, h, m, s):
  156. self.h_ = None
  157. self.m_ = None
  158. self.s_ = None
  159. class AlarmPlan:
  160. handles_map = {
  161. EventType.STAY_DETECTION.value : lambda plan: AlarmPlan.handle_stay_detection(plan),
  162. EventType.RETENTION_DETECTION.value : lambda plan: AlarmPlan.handle_retention_detection(plan),
  163. EventType.TOILETING_DETECTION.value : lambda plan: AlarmPlan.handle_toileting_detection(plan),
  164. EventType.TOILETING_FREQUENCY.value : lambda plan: AlarmPlan.handle_toileting_frequency(plan),
  165. EventType.NIGHT_TOILETING_FREQUENCY.value : lambda plan: AlarmPlan.handle_night_toileting_frequency(plan),
  166. EventType.TOILETING_FREQUENCY_ABNORMAL.value : lambda plan: AlarmPlan.handle_toileting_frequency_abnormal(plan),
  167. EventType.NIGHT_TOILETING_FREQUENCY_ABNORMAL.value : lambda plan: AlarmPlan.handle_night_toileting_frequency_abnormal(plan),
  168. EventType.BATHROOM_STAY_FREQUENCY.value : lambda plan: AlarmPlan.handle_bathroom_stay_frequency(plan),
  169. EventType.TARGET_ABSENCE.value : lambda plan: AlarmPlan.handle_target_absence(plan),
  170. EventType.CLEAN_EXPIRE_EVENTS.value : lambda plan: AlarmPlan.handle_clear_expire_events(plan),
  171. }
  172. def __init__(self,
  173. plan_uuid: str,
  174. name: str,
  175. dev_id: str,
  176. dev_name: str,
  177. enable: bool,
  178. time_plan: TimePlan,
  179. rect: list,
  180. event_type: int,
  181. threshold_time: int,
  182. merge_time: int,
  183. param: dict,
  184. cron: Optional[dict] = None,
  185. linkage_action: LinkageAction = LinkageAction(),
  186. tenant_id:int = 0
  187. ):
  188. self.lock_ = Lock()
  189. self.plan_uuid_ = plan_uuid # 计划id
  190. self.name_ = name # 计划名称
  191. self.dev_id_ = dev_id # 设备id
  192. self.dev_name_ = dev_name # 设备名称
  193. self.enable_ = enable # 是否启用
  194. self.time_plan_ = time_plan # 时间计划
  195. self.param_ = param # 参数
  196. self.linkage_action_ = linkage_action # 联动动作
  197. self.tenant_id_ = tenant_id # 租户id
  198. # 维护状态(根据TimePlanu判断)
  199. self.status_ = 0 # 0未激活,1激活,-1过期
  200. self.status_update_ts_ = -1 # 状态更新时间,初始值为-1
  201. # 事件触发参数
  202. self.rect_ = rect # 检测区域 [left, top, width, height]
  203. self.threshold_time_ = threshold_time # 触发时间阈值
  204. self.merge_time_ = merge_time # 归并时间窗口
  205. self.event_type_ = event_type # 事件类型
  206. self.event_attr_ = self.init_event_attr() # 事件属性
  207. if self.event_attr_ is None:
  208. raise ValueError(f"Invalid event_type: {event_type}")
  209. self.handle_func_ = self.handles_map.get(event_type)
  210. if self.handle_func_ is None:
  211. raise ValueError(f"Invalic event_type: {event_type}")
  212. # 计划任务的开始时间
  213. self.cron_ = cron # {“hour": 7, "minute": 0}
  214. def execute(self):
  215. if self.status_ != 1:
  216. return
  217. g_las.g_alarm_plan_disp.dispatch(self)
  218. # 初始化事件属性
  219. def init_event_attr(self):
  220. try:
  221. event_cls = event_attr_map.get(self.event_type_)
  222. if event_cls is None:
  223. return None
  224. event_attr = event_cls(self.event_type_)
  225. if ((self.event_type_ == EventType.TOILETING_FREQUENCY_ABNORMAL.value) or
  226. (self.event_type_ == EventType.NIGHT_TOILETING_FREQUENCY_ABNORMAL.value)):
  227. event_attr.threshold_count_ = int(self.param_.get("count", 0))
  228. elif ((self.event_type_ == EventType.TARGET_ABSENCE.value)):
  229. event_attr.time_threshold_ = int(self.param_.get("time_threshold", 0))
  230. elif ((self.event_type_ == EventType.CLEAN_EXPIRE_EVENTS.value)):
  231. event_attr.expire_range_ = 90
  232. return event_attr
  233. except json.JSONDecodeError as e:
  234. tb_info = traceback.extract_tb(e.__traceback__)
  235. for frame in tb_info:
  236. LOGERR(f"[{frame.filename}:{frame.lineno}] @{frame.name}(), error:{e}, {e.doc}")
  237. except Exception as e:
  238. tb_info = traceback.extract_tb(e.__traceback__)
  239. for frame in tb_info:
  240. LOGERR(f"[{frame.filename}:{frame.lineno}] @{frame.name}(), error: {e}")
  241. # 更新激活状态
  242. def update_status(self, now: Optional[datetime] = None) -> None:
  243. now = now or datetime.now()
  244. old_status = self.status_
  245. if not self.enable_:
  246. self.status_ = 0
  247. else:
  248. now_fmt = now.strftime("%Y-%m-%d")
  249. # 过期
  250. if now_fmt > self.time_plan_.stop_date_:
  251. self.status_ = -1
  252. elif now_fmt < self.time_plan_.start_date_:
  253. self.status_ = 0
  254. elif self.time_plan_.is_active_now(now):
  255. self.status_ = 1
  256. else:
  257. self.status_ = 0
  258. if self.status_ != old_status:
  259. self.status_update_ts = int(now.timestamp())
  260. LOGINFO(f"[Status] plan {self.plan_uuid_} status_ changed {old_status} -> {self.status_}")
  261. def is_point_in_rect(self, x:float, y:float, rect:list) -> bool:
  262. rx, ry, rw, rh = rect
  263. x_min = min(rx, rx + rw)
  264. x_max = max(rx, rx + rw)
  265. y_min = min(ry, ry - rh)
  266. y_max = max(ry, ry - rh)
  267. bRet: bool = x_min <= x <= x_max and y_min <= y <= y_max
  268. return bRet
  269. # 查找最近 t 秒内,最后一个落在 rect 内的 target_point 的 rtd_unit
  270. def find_latest_rtd_in_region(self, device: Device, rect: list, now: int=None, t: int=1):
  271. now_s = now if now else get_utc_time_s()
  272. rtd_que_copy = device.get_rtd_que_copy()
  273. with self.lock_:
  274. for rtd_unit in reversed(rtd_que_copy): # 倒序扫描
  275. ts_s = int(rtd_unit.get("timestamp", 0))
  276. if now_s - ts_s > t:
  277. break # 已经超过 t 秒,可以直接结束
  278. # 检查点是否在区域内
  279. for pt in rtd_unit.get("target_point", []):
  280. if len(pt) >= 2:
  281. x, y = pt[0], pt[1]
  282. if self.is_point_in_rect(x, y, rect):
  283. return rtd_unit
  284. return None
  285. # 停留事件
  286. def handle_stay_detection(self):
  287. try:
  288. dev_id = self.dev_id_
  289. device:Device = g_Dev.g_dev_mgr.find_dev_map(dev_id)
  290. if not device:
  291. return
  292. now = get_utc_time_s()
  293. # 查找最新的落在检测区域的目标
  294. rtd_unit = self.find_latest_rtd_in_region(device, self.rect_, now, 3)
  295. if rtd_unit:
  296. timestamp = rtd_unit["timestamp"]
  297. pose = rtd_unit["pose"]
  298. target_point = rtd_unit["target_point"]
  299. if self.event_attr_.enter_ts_ == -1:
  300. self.event_attr_.enter_ts_ = timestamp
  301. else:
  302. self.event_attr_.leave_ts_ = timestamp
  303. if self.event_attr_.enter_ts_ == -1 or self.event_attr_.leave_ts_ == -1:
  304. return
  305. # 归并时间内,不认为事件结束
  306. if now - self.event_attr_.leave_ts_ < self.merge_time_:
  307. return
  308. self.event_attr_.stay_time_ = self.event_attr_.leave_ts_ - self.event_attr_.enter_ts_
  309. stay_time =self.event_attr_.stay_time_
  310. # 时间小于触发时间阈值,忽略并重置
  311. if stay_time < self.threshold_time_ :
  312. self.event_attr_.reset()
  313. LOGINFO(f"less than threshold_time, alarm_plan: {self.plan_uuid_}, event_type: {self.event_type_}")
  314. return
  315. # 构造事件
  316. # 入库
  317. info = {
  318. "start_time": utc_to_bj_s(self.event_attr_.enter_ts_),
  319. "end_time": utc_to_bj_s(self.event_attr_.leave_ts_),
  320. "stay_time": stay_time
  321. }
  322. event_uuid = str(uuid.uuid4())
  323. params = {
  324. "dev_id": dev_id,
  325. "uuid": event_uuid,
  326. "plan_uuid": self.plan_uuid_,
  327. "event_type": self.event_type_,
  328. "info": json.dumps(info),
  329. "linkage_push_wechat_service": self.linkage_action_.wechat_service_,
  330. "is_handle": 0,
  331. "create_time": get_bj_time_s(),
  332. "is_deleted": 0,
  333. "tenant_id": self.tenant_id_,
  334. "remark": json.dumps({}, ensure_ascii=False)
  335. }
  336. db_req_que.put(DBRequest_Async(sql=sqls.sql_insert_events, params=params, callback=None))
  337. # 通知
  338. linkage_action = {
  339. "linkage_push_wechat_service": self.linkage_action_.wechat_service_
  340. }
  341. mqtt_send.alarm_event(dev_id, self.dev_name_, event_uuid, self.plan_uuid_, event_desc_map[self.event_type_], info, linkage_action, "events")
  342. LOGDBG(f"new event: {event_desc_map[self.event_type_]}, stay_time: {stay_time}")
  343. self.event_attr_.reset()
  344. except json.JSONDecodeError as e:
  345. tb_info = traceback.extract_tb(e.__traceback__)
  346. for frame in tb_info:
  347. LOGERR(f"[{frame.filename}:{frame.lineno}] @{frame.name}(), error:{e}, {e.doc}")
  348. except Exception as e:
  349. tb_info = traceback.extract_tb(e.__traceback__)
  350. for frame in tb_info:
  351. LOGERR(f"[{frame.filename}:{frame.lineno}] @{frame.name}(), error: {e}")
  352. # 滞留事件
  353. def handle_retention_detection(self):
  354. try:
  355. dev_id = self.dev_id_
  356. device:Device = g_Dev.g_dev_mgr.find_dev_map(dev_id)
  357. if not device:
  358. return
  359. now = get_utc_time_s()
  360. # 查找最新的落在检测区域的目标
  361. rtd_unit = self.find_latest_rtd_in_region(device, self.rect_, now, 3)
  362. if rtd_unit:
  363. timestamp = rtd_unit["timestamp"]
  364. pose = rtd_unit["pose"]
  365. target_point = rtd_unit["target_point"]
  366. if self.event_attr_.enter_ts_ == -1:
  367. self.event_attr_.enter_ts_ = timestamp
  368. else:
  369. self.event_attr_.leave_ts_ = timestamp
  370. if self.event_attr_.enter_ts_ == -1 or self.event_attr_.leave_ts_ == -1:
  371. return
  372. # 归并时间内,不认为事件结束
  373. if now - self.event_attr_.leave_ts_ < self.merge_time_:
  374. return
  375. self.event_attr_.stay_time_ = self.event_attr_.leave_ts_ - self.event_attr_.enter_ts_
  376. stay_time =self.event_attr_.stay_time_
  377. # 时间小于触发时间阈值,忽略并重置
  378. if stay_time < self.threshold_time_ :
  379. self.event_attr_.reset()
  380. LOGINFO(f"less than threshold_time, alarm_plan: {self.plan_uuid_}, event_type: {self.event_type_}")
  381. return
  382. # 构造事件
  383. # 入库
  384. info = {
  385. "start_time": utc_to_bj_s(self.event_attr_.enter_ts_),
  386. "end_time": utc_to_bj_s(self.event_attr_.leave_ts_),
  387. "stay_time": stay_time
  388. }
  389. event_uuid = str(uuid.uuid4())
  390. params = {
  391. "dev_id": dev_id,
  392. "uuid": event_uuid,
  393. "plan_uuid": self.plan_uuid_,
  394. "event_type": self.event_type_,
  395. "info": json.dumps(info),
  396. "linkage_push_wechat_service": self.linkage_action_.wechat_service_,
  397. "is_handle": 0,
  398. "create_time": get_bj_time_s(),
  399. "is_deleted": 0,
  400. "tenant_id": self.tenant_id_,
  401. "remark": json.dumps({}, ensure_ascii=False)
  402. }
  403. db_req_que.put(DBRequest_Async(sql=sqls.sql_insert_events, params=params, callback=None))
  404. # 通知
  405. linkage_action = {
  406. "linkage_push_wechat_service": self.linkage_action_.wechat_service_
  407. }
  408. mqtt_send.alarm_event(dev_id, self.dev_name_, event_uuid, self.plan_uuid_, event_desc_map[self.event_type_], info, linkage_action, "events")
  409. LOGDBG(f"new event: {event_desc_map[self.event_type_]}, stay_time: {stay_time}")
  410. self.event_attr_.reset()
  411. except json.JSONDecodeError as e:
  412. tb_info = traceback.extract_tb(e.__traceback__)
  413. for frame in tb_info:
  414. LOGERR(f"[{frame.filename}:{frame.lineno}] @{frame.name}(), error:{e}, {e.doc}")
  415. except Exception as e:
  416. tb_info = traceback.extract_tb(e.__traceback__)
  417. for frame in tb_info:
  418. LOGERR(f"[{frame.filename}:{frame.lineno}] @{frame.name}(), error: {e}")
  419. # 如厕事件
  420. def handle_toileting_detection(self):
  421. try:
  422. dev_id = self.dev_id_
  423. device:Device = g_Dev.g_dev_mgr.find_dev_map(dev_id)
  424. if not device:
  425. return
  426. now = get_utc_time_s()
  427. # 查找最新的落在检测区域的目标
  428. rtd_unit = self.find_latest_rtd_in_region(device, self.rect_, now, 3)
  429. if rtd_unit:
  430. timestamp = rtd_unit["timestamp"]
  431. pose = rtd_unit["pose"]
  432. target_point = rtd_unit["target_point"]
  433. if self.event_attr_.enter_ts_ == -1:
  434. self.event_attr_.enter_ts_ = timestamp
  435. else:
  436. self.event_attr_.leave_ts_ = timestamp
  437. if self.event_attr_.enter_ts_ == -1 or self.event_attr_.leave_ts_ == -1:
  438. return
  439. # 归并时间内,不认为事件结束
  440. if now - self.event_attr_.leave_ts_ < self.merge_time_:
  441. return
  442. self.event_attr_.stay_time_ = self.event_attr_.leave_ts_ - self.event_attr_.enter_ts_
  443. stay_time =self.event_attr_.stay_time_
  444. # 时间小于触发时间阈值,忽略并重置
  445. if stay_time < self.threshold_time_ :
  446. self.event_attr_.reset()
  447. LOGINFO(f"less than threshold_time, alarm_plan: {self.plan_uuid_}, event_type: {self.event_type_}")
  448. return
  449. # 构造事件
  450. # 入库
  451. info = {
  452. "start_time": utc_to_bj_s(self.event_attr_.enter_ts_),
  453. "end_time": utc_to_bj_s(self.event_attr_.leave_ts_),
  454. "stay_time": stay_time
  455. }
  456. event_uuid = str(uuid.uuid4())
  457. params = {
  458. "dev_id": dev_id,
  459. "uuid": event_uuid,
  460. "plan_uuid": self.plan_uuid_,
  461. "event_type": self.event_type_,
  462. "info": json.dumps(info),
  463. "linkage_push_wechat_service": self.linkage_action_.wechat_service_,
  464. "is_handle": 0,
  465. "create_time": get_bj_time_s(),
  466. "is_deleted": 0,
  467. "tenant_id": self.tenant_id_,
  468. "remark": json.dumps({}, ensure_ascii=False)
  469. }
  470. db_req_que.put(DBRequest_Async(sql=sqls.sql_insert_events, params=params, callback=None))
  471. # 通知
  472. linkage_action = {
  473. "linkage_push_wechat_service": self.linkage_action_.wechat_service_
  474. }
  475. mqtt_send.alarm_event(dev_id, self.dev_name_, event_uuid, self.plan_uuid_, event_desc_map[self.event_type_], info, linkage_action, "events")
  476. LOGDBG(f"new event: {event_desc_map[self.event_type_]}, stay_time: {stay_time}")
  477. self.event_attr_.reset()
  478. except json.JSONDecodeError as e:
  479. tb_info = traceback.extract_tb(e.__traceback__)
  480. for frame in tb_info:
  481. LOGERR(f"[{frame.filename}:{frame.lineno}] @{frame.name}(), error:{e}, {e.doc}")
  482. except Exception as e:
  483. tb_info = traceback.extract_tb(e.__traceback__)
  484. for frame in tb_info:
  485. LOGERR(f"[{frame.filename}:{frame.lineno}] @{frame.name}(), error: {e}")
  486. # 如厕频次统计
  487. def handle_toileting_frequency(self):
  488. try:
  489. dev_id = self.dev_id_
  490. device:Device = g_Dev.g_dev_mgr.find_dev_map(dev_id)
  491. if not device:
  492. return
  493. start_dt, end_dt = helper.get_query_time_range(self.param_)
  494. params = {
  495. "dev_id" : self.dev_id_,
  496. "event_type": EventType.TOILETING_DETECTION.value,
  497. "start_dt": start_dt,
  498. "end_dt": end_dt
  499. }
  500. userdata = {
  501. "start_dt" : start_dt,
  502. "end_dt" : end_dt
  503. }
  504. db_req_que.put(DBRequest_Async(
  505. sql=sqls.sql_query_events_by_datetime,
  506. params=params,
  507. callback=self.cb_toileting_frequency,
  508. userdata=userdata))
  509. except json.JSONDecodeError as e:
  510. tb_info = traceback.extract_tb(e.__traceback__)
  511. for frame in tb_info:
  512. LOGERR(f"[{frame.filename}:{frame.lineno}] @{frame.name}(), error:{e}, {e.doc}")
  513. except Exception as e:
  514. tb_info = traceback.extract_tb(e.__traceback__)
  515. for frame in tb_info:
  516. LOGERR(f"[{frame.filename}:{frame.lineno}] @{frame.name}(), error: {e}")
  517. # 夜间如厕频次统计
  518. def handle_night_toileting_frequency(self):
  519. try:
  520. dev_id = self.dev_id_
  521. device:Device = g_Dev.g_dev_mgr.find_dev_map(dev_id)
  522. if not device:
  523. return
  524. start_dt, end_dt = helper.get_query_time_range(self.param_)
  525. params = {
  526. "dev_id" : self.dev_id_,
  527. "event_type": EventType.TOILETING_DETECTION.value,
  528. "start_dt": start_dt,
  529. "end_dt": end_dt
  530. }
  531. userdata = {
  532. "start_dt" : start_dt,
  533. "end_dt" : end_dt
  534. }
  535. db_req_que.put(DBRequest_Async(
  536. sql=sqls.sql_query_events_by_datetime,
  537. params=params,
  538. callback=self.cb_night_toileting_frequency,
  539. userdata=userdata))
  540. except json.JSONDecodeError as e:
  541. tb_info = traceback.extract_tb(e.__traceback__)
  542. for frame in tb_info:
  543. LOGERR(f"[{frame.filename}:{frame.lineno}] @{frame.name}(), error:{e}, {e.doc}")
  544. except Exception as e:
  545. tb_info = traceback.extract_tb(e.__traceback__)
  546. for frame in tb_info:
  547. LOGERR(f"[{frame.filename}:{frame.lineno}] @{frame.name}(), error: {e}")
  548. # 如厕频次异常
  549. def handle_toileting_frequency_abnormal(self):
  550. try:
  551. dev_id = self.dev_id_
  552. device:Device = g_Dev.g_dev_mgr.find_dev_map(dev_id)
  553. if not device:
  554. return
  555. start_dt, end_dt = helper.get_query_time_range(self.param_)
  556. params = {
  557. "dev_id" : self.dev_id_,
  558. "event_type": EventType.TOILETING_DETECTION.value,
  559. "start_dt": start_dt,
  560. "end_dt": end_dt
  561. }
  562. userdata = {
  563. "start_dt" : start_dt,
  564. "end_dt" : end_dt
  565. }
  566. db_req_que.put(DBRequest_Async(
  567. sql=sqls.sql_query_events_by_datetime,
  568. params=params,
  569. callback=self.cb_toileting_frequency_abnormal,
  570. userdata=userdata))
  571. except json.JSONDecodeError as e:
  572. tb_info = traceback.extract_tb(e.__traceback__)
  573. for frame in tb_info:
  574. LOGERR(f"[{frame.filename}:{frame.lineno}] @{frame.name}(), error:{e}, {e.doc}")
  575. except Exception as e:
  576. tb_info = traceback.extract_tb(e.__traceback__)
  577. for frame in tb_info:
  578. LOGERR(f"[{frame.filename}:{frame.lineno}] @{frame.name}(), error: {e}")
  579. # 起夜异常
  580. def handle_night_toileting_frequency_abnormal(self):
  581. try:
  582. dev_id = self.dev_id_
  583. device:Device = g_Dev.g_dev_mgr.find_dev_map(dev_id)
  584. if not device:
  585. return
  586. start_dt, end_dt = helper.get_query_time_range(self.param_)
  587. params = {
  588. "dev_id" : self.dev_id_,
  589. "event_type": EventType.TOILETING_DETECTION.value,
  590. "start_dt": start_dt,
  591. "end_dt": end_dt
  592. }
  593. userdata = {
  594. "start_dt" : start_dt,
  595. "end_dt" : end_dt
  596. }
  597. db_req_que.put(DBRequest_Async(
  598. sql=sqls.sql_query_events_by_datetime,
  599. params=params,
  600. callback=self.cb_night_toileting_frequency_abnormal,
  601. userdata=userdata))
  602. except json.JSONDecodeError as e:
  603. tb_info = traceback.extract_tb(e.__traceback__)
  604. for frame in tb_info:
  605. LOGERR(f"[{frame.filename}:{frame.lineno}] @{frame.name}(), error:{e}, {e.doc}")
  606. except Exception as e:
  607. tb_info = traceback.extract_tb(e.__traceback__)
  608. for frame in tb_info:
  609. LOGERR(f"[{frame.filename}:{frame.lineno}] @{frame.name}(), error: {e}")
  610. # 卫生间频次统计
  611. def handle_bathroom_stay_frequency(self):
  612. try:
  613. dev_id = self.dev_id_
  614. device:Device = g_Dev.g_dev_mgr.find_dev_map(dev_id)
  615. if not device:
  616. return
  617. start_dt, end_dt = helper.get_query_time_range(self.param_)
  618. params = {
  619. "dev_id" : self.dev_id_,
  620. "event_type": EventType.STAY_DETECTION.value,
  621. "start_dt": start_dt,
  622. "end_dt": end_dt
  623. }
  624. userdata = {
  625. "start_dt" : start_dt,
  626. "end_dt" : end_dt
  627. }
  628. db_req_que.put(DBRequest_Async(
  629. sql=sqls.sql_query_events_by_datetime,
  630. params=params,
  631. callback=self.cb_bathroom_stay_frequency,
  632. userdata=userdata))
  633. except json.JSONDecodeError as e:
  634. tb_info = traceback.extract_tb(e.__traceback__)
  635. for frame in tb_info:
  636. LOGERR(f"[{frame.filename}:{frame.lineno}] @{frame.name}(), error:{e}, {e.doc}")
  637. except Exception as e:
  638. tb_info = traceback.extract_tb(e.__traceback__)
  639. for frame in tb_info:
  640. LOGERR(f"[{frame.filename}:{frame.lineno}] @{frame.name}(), error: {e}")
  641. # 异常消失
  642. def handle_target_absence(self):
  643. try:
  644. dev_id = self.dev_id_
  645. device:Device = g_Dev.g_dev_mgr.find_dev_map(dev_id)
  646. if not device:
  647. return
  648. now = get_utc_time_s()
  649. threshold: int = self.event_attr_.time_threshold_
  650. # 查找最新的落在检测区域的目标
  651. rtd_unit = self.find_latest_rtd_in_region(device, self.rect_, now, 3)
  652. if rtd_unit:
  653. return
  654. timestamp = get_utc_time_s()
  655. if self.event_attr_.leave_ts_ == -1:
  656. self.event_attr_.leave_ts_ = timestamp
  657. else:
  658. self.event_attr_.enter_ts_ = timestamp
  659. if self.event_attr_.enter_ts_ == -1 or self.event_attr_.leave_ts_ == -1:
  660. return
  661. # 归并时间内,不认为事件结束
  662. if now - self.event_attr_.enter_ts_ < self.merge_time_:
  663. return
  664. self.event_attr_.absence_time_ = self.event_attr_.enter_ts_ - self.event_attr_.leave_ts_
  665. absence_time = self.event_attr_.absence_time_
  666. # 时间小于触发时间阈值,忽略并重置
  667. if absence_time < threshold :
  668. self.event_attr_.reset()
  669. LOGINFO(f"less than threshold_time, alarm_plan: {self.plan_uuid_}, event_type: {self.event_type_}")
  670. return
  671. # 构造事件
  672. # 入库
  673. info = {
  674. "start_time": utc_to_bj_s(self.event_attr_.leave_ts_),
  675. "end_time": utc_to_bj_s(self.event_attr_.enter_ts_),
  676. "absence_time": absence_time
  677. }
  678. event_uuid = str(uuid.uuid4())
  679. params = {
  680. "dev_id": dev_id,
  681. "uuid": event_uuid,
  682. "plan_uuid": self.plan_uuid_,
  683. "event_type": self.event_type_,
  684. "info": json.dumps(info),
  685. "linkage_push_wechat_service": self.linkage_action_.wechat_service_,
  686. "is_handle": 0,
  687. "create_time": get_bj_time_s(),
  688. "is_deleted": 0,
  689. "tenant_id": self.tenant_id_,
  690. "remark": json.dumps({}, ensure_ascii=False)
  691. }
  692. db_req_que.put(DBRequest_Async(sql=sqls.sql_insert_events, params=params, callback=None))
  693. # 通知
  694. linkage_action = {
  695. "linkage_push_wechat_service": self.linkage_action_.wechat_service_
  696. }
  697. mqtt_send.alarm_event(dev_id, self.dev_name_, event_uuid, self.plan_uuid_, event_desc_map[self.event_type_], info, linkage_action, "events")
  698. LOGDBG(f"new event: {event_desc_map[self.event_type_]}, absence_time: {absence_time}")
  699. self.event_attr_.reset()
  700. except json.JSONDecodeError as e:
  701. tb_info = traceback.extract_tb(e.__traceback__)
  702. for frame in tb_info:
  703. LOGERR(f"[{frame.filename}:{frame.lineno}] @{frame.name}(), error:{e}, {e.doc}")
  704. except Exception as e:
  705. tb_info = traceback.extract_tb(e.__traceback__)
  706. for frame in tb_info:
  707. LOGERR(f"[{frame.filename}:{frame.lineno}] @{frame.name}(), error: {e}")
  708. # 清理过期事件
  709. def handle_clear_expire_events(self):
  710. try:
  711. params = {
  712. "save_days": self.event_attr_.expire_range_
  713. }
  714. db_execute_async(sqls.sql_delete_expire_events, params=params)
  715. except json.JSONDecodeError as e:
  716. tb_info = traceback.extract_tb(e.__traceback__)
  717. for frame in tb_info:
  718. LOGERR(f"[{frame.filename}:{frame.lineno}] @{frame.name}(), error:{e}, {e.doc}")
  719. except Exception as e:
  720. tb_info = traceback.extract_tb(e.__traceback__)
  721. for frame in tb_info:
  722. LOGERR(f"[{frame.filename}:{frame.lineno}] @{frame.name}(), error: {e}")
  723. # ========== 一些回调 ==========
  724. # 如厕频次统计回调
  725. def cb_toileting_frequency(self, result, userdata):
  726. try:
  727. if result:
  728. count = 0
  729. event_list = []
  730. for row in result:
  731. dev_id: str = row.get("dev_id")
  732. event_uuid: str = row.get("uuid")
  733. plan_uuid: str = row.get("plan_uuid")
  734. event_type: int = row.get("event_type")
  735. info: dict = json.loads(row["info"]) if row.get("info") else {}
  736. is_handle: str = row.get("is_handle")
  737. create_time: str = row.get("create_time")
  738. update_time: str = row.get("update_time")
  739. is_deleted: str = row.get("is_deleted")
  740. remark: str = row.get("remark")
  741. event_list.append(info)
  742. this_event_uuid = str(uuid.uuid4())
  743. last_info = {
  744. "start_time" : userdata["start_dt"],
  745. "end_time" : userdata["end_dt"],
  746. "count" : len(event_list),
  747. "event_list" : event_list
  748. }
  749. # 入库
  750. event_uuid = str(uuid.uuid4())
  751. params = {
  752. "dev_id": dev_id,
  753. "uuid": this_event_uuid,
  754. "plan_uuid": self.plan_uuid_,
  755. "event_type": self.event_type_,
  756. "info": json.dumps(last_info, ensure_ascii=False),
  757. "linkage_push_wechat_service": self.linkage_action_.wechat_service_,
  758. "is_handle": 0,
  759. "create_time": get_bj_time_s(),
  760. "is_deleted": 0,
  761. "tenant_id": self.tenant_id_,
  762. "remark": json.dumps({}, ensure_ascii=False)
  763. }
  764. db_req_que.put(DBRequest_Async(sql=sqls.sql_insert_events, params=params, callback=None))
  765. # 通知
  766. linkage_action = {
  767. "linkage_push_wechat_service": self.linkage_action_.wechat_service_
  768. }
  769. mqtt_send.alarm_event(dev_id, self.dev_name_, this_event_uuid, plan_uuid, event_desc_map[self.event_type_], last_info, linkage_action, "events")
  770. LOGINFO(f"cb_toileting_frequency succeed")
  771. else:
  772. LOGDBG("cb_toileting_frequency, empty result")
  773. except json.JSONDecodeError as e:
  774. tb_info = traceback.extract_tb(e.__traceback__)
  775. for frame in tb_info:
  776. LOGERR(f"[{frame.filename}:{frame.lineno}] @{frame.name}(), error:{e}, {e.doc}")
  777. except Exception as e:
  778. tb_info = traceback.extract_tb(e.__traceback__)
  779. for frame in tb_info:
  780. LOGERR(f"[{frame.filename}:{frame.lineno}] @{frame.name}(), error: {e}")
  781. return
  782. # 夜间如厕频次统计回调
  783. def cb_night_toileting_frequency(self, result, userdata):
  784. try:
  785. if result:
  786. count = 0
  787. event_list = []
  788. for row in result:
  789. dev_id: str = row.get("dev_id")
  790. event_uuid: str = row.get("uuid")
  791. plan_uuid: str = row.get("plan_uuid")
  792. event_type: int = row.get("event_type")
  793. info: dict = json.loads(row["info"]) if row.get("info") else {}
  794. is_handle: str = row.get("is_handle")
  795. create_time: str = row.get("create_time")
  796. update_time: str = row.get("update_time")
  797. is_deleted: str = row.get("is_deleted")
  798. remark: str = row.get("remark")
  799. event_list.append(info)
  800. this_event_uuid = str(uuid.uuid4())
  801. last_info = {
  802. "start_time" : userdata["start_dt"],
  803. "end_time" : userdata["end_dt"],
  804. "count" : len(event_list),
  805. "event_list" : event_list
  806. }
  807. # 入库
  808. event_uuid = str(uuid.uuid4())
  809. params = {
  810. "dev_id": dev_id,
  811. "uuid": this_event_uuid,
  812. "plan_uuid": self.plan_uuid_,
  813. "event_type": self.event_type_,
  814. "info": json.dumps(last_info, ensure_ascii=False),
  815. "linkage_push_wechat_service": self.linkage_action_.wechat_service_,
  816. "is_handle": 0,
  817. "create_time": get_bj_time_s(),
  818. "is_deleted": 0,
  819. "tenant_id": self.tenant_id_,
  820. "remark": json.dumps({}, ensure_ascii=False)
  821. }
  822. db_req_que.put(DBRequest_Async(sql=sqls.sql_insert_events, params=params, callback=None))
  823. # 通知
  824. linkage_action = {
  825. "linkage_push_wechat_service": self.linkage_action_.wechat_service_
  826. }
  827. mqtt_send.alarm_event(dev_id, self.dev_name_, this_event_uuid, plan_uuid, event_desc_map[self.event_type_], last_info, linkage_action, "events")
  828. LOGINFO(f"cb_night_toileting_frequency succeed")
  829. else:
  830. LOGDBG("cb_night_toileting_frequency, empty result")
  831. except json.JSONDecodeError as e:
  832. tb_info = traceback.extract_tb(e.__traceback__)
  833. for frame in tb_info:
  834. LOGERR(f"[{frame.filename}:{frame.lineno}] @{frame.name}(), error:{e}, {e.doc}")
  835. except Exception as e:
  836. tb_info = traceback.extract_tb(e.__traceback__)
  837. for frame in tb_info:
  838. LOGERR(f"[{frame.filename}:{frame.lineno}] @{frame.name}(), error: {e}")
  839. return
  840. # 如厕频次异常回调
  841. def cb_toileting_frequency_abnormal(self, result, userdata):
  842. try:
  843. if result:
  844. count = 0
  845. event_list = []
  846. for row in result:
  847. dev_id: str = row.get("dev_id")
  848. event_uuid: str = row.get("uuid")
  849. plan_uuid: str = row.get("plan_uuid")
  850. event_type: int = row.get("event_type")
  851. info: dict = json.loads(row["info"]) if row.get("info") else {}
  852. is_handle: str = row.get("is_handle")
  853. create_time: str = row.get("create_time")
  854. update_time: str = row.get("update_time")
  855. is_deleted: str = row.get("is_deleted")
  856. remark: str = row.get("remark")
  857. event_list.append(info)
  858. if len(event_list) < self.event_attr_.threshold_count_:
  859. return
  860. this_event_uuid = str(uuid.uuid4())
  861. last_info = {
  862. "start_time" : userdata["start_dt"],
  863. "end_time" : userdata["end_dt"],
  864. "count" : len(event_list),
  865. "event_list" : event_list
  866. }
  867. # 入库
  868. event_uuid = str(uuid.uuid4())
  869. params = {
  870. "dev_id": dev_id,
  871. "uuid": this_event_uuid,
  872. "plan_uuid": self.plan_uuid_,
  873. "event_type": self.event_type_,
  874. "info": json.dumps(last_info, ensure_ascii=False),
  875. "linkage_push_wechat_service": self.linkage_action_.wechat_service_,
  876. "is_handle": 0,
  877. "create_time": get_bj_time_s(),
  878. "is_deleted": 0,
  879. "tenant_id": self.tenant_id_,
  880. "remark": json.dumps({}, ensure_ascii=False)
  881. }
  882. db_req_que.put(DBRequest_Async(sql=sqls.sql_insert_events, params=params, callback=None))
  883. # 通知
  884. linkage_action = {
  885. "linkage_push_wechat_service": self.linkage_action_.wechat_service_
  886. }
  887. mqtt_send.alarm_event(dev_id, self.dev_name_, this_event_uuid, self.plan_uuid_, event_desc_map[self.event_type_], last_info, linkage_action, "events")
  888. LOGINFO(f"cb_toileting_frequency_abnormal succeed")
  889. else:
  890. LOGDBG("cb_toileting_frequency_abnormal, empty result")
  891. except json.JSONDecodeError as e:
  892. tb_info = traceback.extract_tb(e.__traceback__)
  893. for frame in tb_info:
  894. LOGERR(f"[{frame.filename}:{frame.lineno}] @{frame.name}(), error:{e}, {e.doc}")
  895. except Exception as e:
  896. tb_info = traceback.extract_tb(e.__traceback__)
  897. for frame in tb_info:
  898. LOGERR(f"[{frame.filename}:{frame.lineno}] @{frame.name}(), error: {e}")
  899. return
  900. # 起夜异常回调
  901. def cb_night_toileting_frequency_abnormal(self, result, userdata):
  902. try:
  903. if result:
  904. count = 0
  905. event_list = []
  906. for row in result:
  907. dev_id: str = row.get("dev_id")
  908. event_uuid: str = row.get("uuid")
  909. plan_uuid: str = row.get("plan_uuid")
  910. event_type: int = row.get("event_type")
  911. info: dict = json.loads(row["info"]) if row.get("info") else {}
  912. is_handle: str = row.get("is_handle")
  913. create_time: str = row.get("create_time")
  914. update_time: str = row.get("update_time")
  915. is_deleted: str = row.get("is_deleted")
  916. remark: str = row.get("remark")
  917. event_list.append(info)
  918. if len(event_list) < self.event_attr_.threshold_count_:
  919. return
  920. this_event_uuid = str(uuid.uuid4())
  921. last_info = {
  922. "start_time" : userdata["start_dt"],
  923. "end_time" : userdata["end_dt"],
  924. "count" : len(event_list),
  925. "event_list" : event_list
  926. }
  927. # 入库
  928. event_uuid = str(uuid.uuid4())
  929. params = {
  930. "dev_id": dev_id,
  931. "uuid": this_event_uuid,
  932. "plan_uuid": self.plan_uuid_,
  933. "event_type": self.event_type_,
  934. "info": json.dumps(last_info, ensure_ascii=False),
  935. "linkage_push_wechat_service": self.linkage_action_.wechat_service_,
  936. "is_handle": 0,
  937. "create_time": get_bj_time_s(),
  938. "is_deleted": 0,
  939. "tenant_id": self.tenant_id_,
  940. "remark": json.dumps({}, ensure_ascii=False)
  941. }
  942. db_req_que.put(DBRequest_Async(sql=sqls.sql_insert_events, params=params, callback=None))
  943. # 通知
  944. linkage_action = {
  945. "linkage_push_wechat_service": self.linkage_action_.wechat_service_
  946. }
  947. mqtt_send.alarm_event(dev_id, self.dev_name_, this_event_uuid, self.plan_uuid_, event_desc_map[self.event_type_], last_info, linkage_action, "events")
  948. LOGINFO(f"cb_night_toileting_frequency_abnormal succeed")
  949. else:
  950. LOGDBG("cb_night_toileting_frequency_abnormal, empty result")
  951. except json.JSONDecodeError as e:
  952. tb_info = traceback.extract_tb(e.__traceback__)
  953. for frame in tb_info:
  954. LOGERR(f"[{frame.filename}:{frame.lineno}] @{frame.name}(), error:{e}, {e.doc}")
  955. except Exception as e:
  956. tb_info = traceback.extract_tb(e.__traceback__)
  957. for frame in tb_info:
  958. LOGERR(f"[{frame.filename}:{frame.lineno}] @{frame.name}(), error: {e}")
  959. return
  960. # 卫生间频次统计回调
  961. def cb_bathroom_stay_frequency(self, result, userdata):
  962. try:
  963. if result:
  964. count = 0
  965. event_list = []
  966. for row in result:
  967. dev_id: str = row.get("dev_id")
  968. event_uuid: str = row.get("uuid")
  969. plan_uuid: str = row.get("plan_uuid")
  970. event_type: int = row.get("event_type")
  971. info: dict = json.loads(row["info"]) if row.get("info") else {}
  972. is_handle: str = row.get("is_handle")
  973. create_time: str = row.get("create_time")
  974. update_time: str = row.get("update_time")
  975. is_deleted: str = row.get("is_deleted")
  976. remark: str = row.get("remark")
  977. event_list.append(info)
  978. this_event_uuid = str(uuid.uuid4())
  979. last_info = {
  980. "start_time" : userdata["start_dt"],
  981. "end_time" : userdata["end_dt"],
  982. "count" : len(event_list),
  983. "event_list" : event_list
  984. }
  985. # 入库
  986. event_uuid = str(uuid.uuid4())
  987. params = {
  988. "dev_id": dev_id,
  989. "uuid": this_event_uuid,
  990. "plan_uuid": self.plan_uuid_,
  991. "event_type": self.event_type_,
  992. "info": json.dumps(last_info, ensure_ascii=False),
  993. "linkage_push_wechat_service": self.linkage_action_.wechat_service_,
  994. "is_handle": 0,
  995. "create_time": get_bj_time_s(),
  996. "is_deleted": 0,
  997. "tenant_id": self.tenant_id_,
  998. "remark": json.dumps({}, ensure_ascii=False)
  999. }
  1000. db_req_que.put(DBRequest_Async(sql=sqls.sql_insert_events, params=params, callback=None))
  1001. # 通知
  1002. linkage_action = {
  1003. "linkage_push_wechat_service": self.linkage_action_.wechat_service_
  1004. }
  1005. mqtt_send.alarm_event(dev_id, self.dev_name_, this_event_uuid, plan_uuid, event_desc_map[self.event_type_], last_info, linkage_action, "events")
  1006. LOGINFO(f"cb_bathroom_stay_frequency+- succeed")
  1007. else:
  1008. LOGDBG("cb_bathroom_stay_frequency, empty result")
  1009. except json.JSONDecodeError as e:
  1010. tb_info = traceback.extract_tb(e.__traceback__)
  1011. for frame in tb_info:
  1012. LOGERR(f"[{frame.filename}:{frame.lineno}] @{frame.name}(), error:{e}, {e.doc}")
  1013. except Exception as e:
  1014. tb_info = traceback.extract_tb(e.__traceback__)
  1015. for frame in tb_info:
  1016. LOGERR(f"[{frame.filename}:{frame.lineno}] @{frame.name}(), error: {e}")
  1017. return