FuncOrdinary.cpp 18 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391
  1. #include "FuncOrdinary.h"
  2. #include "GlobalVariable.h"
  3. #include "FromRedis.h"
  4. #include "FromWebAPI.h"
  5. #include "UniversalFunc.h"
  6. #include "GlobalConfig.h"
  7. FuncOrdinary::FuncOrdinary()
  8. {
  9. m_logger = spdlog::get("SPAServer");
  10. if(m_logger == nullptr)
  11. {
  12. SPDLOG_LOGGER_ERROR(m_logger, "SPAServer logger is nullptr");
  13. return;
  14. }
  15. }
  16. FuncOrdinary::~FuncOrdinary()
  17. {
  18. m_listSrcAlarm.clearAlarmInfo();
  19. m_listAlarmBuffer.clearAlarmInfo();
  20. }
  21. /* 任务线程 */
  22. void FuncOrdinary::task()
  23. {
  24. /* 初始化WebAPI和Redis接口 */
  25. if(!initDataInterface())
  26. {
  27. return;
  28. }
  29. /* 保存上次获取时间段的时间 */
  30. QDateTime lastTime = QDateTime::fromString("2025-05-07 00:00:00", "yyyy-MM-dd hh:mm:ss");
  31. while (GThreadInfo.getRunning())
  32. {
  33. /* 休眠设置的时间 */
  34. std::this_thread::sleep_for(std::chrono::milliseconds(GVariable.ThreadSleepMS));
  35. /* 从全局的信息块中更新功能信息,如是否需要退出线程,主要是更新摄像机关联的算法信息 */
  36. GThreadInfo.updateFuncInfo(m_funcThreadInfo);
  37. /* 如果线程的运行状态为停止 */
  38. if(m_funcThreadInfo.RunState == RunTimeState::RUN_STATE_STOP)
  39. {
  40. /* 退出线程 */
  41. break;
  42. }
  43. /* 判断是否在检测时间段内 */
  44. if(!isInDetectTime(m_periodInfo))
  45. {
  46. continue;
  47. }
  48. // SPDLOG_LOGGER_DEBUG(m_logger, "{} 在检测时间段内", m_baseLog);
  49. /* ========================= 读取Redis数据 ========================= */
  50. readRedisData(m_listSrcAlarm);
  51. // SPDLOG_LOGGER_DEBUG(m_logger, "读取Redis数据成功, 报警数据数目: {}", m_listAlarm.listAlarmInfo.size());
  52. // continue;
  53. /* ========================= 处理报警数据 ========================= */
  54. for(auto newAlarm : m_listSrcAlarm.listAlarmInfo)
  55. {
  56. /* 找到该报警相关的房间信息 */
  57. auto roomInfo = m_funcThreadInfo.findRoomInfo(newAlarm->RoomID);
  58. if(roomInfo.RoomID == -1)
  59. {
  60. SPDLOG_LOGGER_ERROR(m_logger, "{} 房间ID: {} 不存在", m_baseLog, newAlarm->RoomID);
  61. continue;
  62. }
  63. /* 创建基础日志信息 */
  64. std::string tmpBaseLog = fmt::format("{} 房间: {}, 摄像机ID: {}, 算法ID: {}", m_baseLog, roomInfo.strRoomName, newAlarm->DeviceID, newAlarm->ActionID);
  65. /* -------------------------------------------------------------------------------
  66. * 判断有无报警记录,新的报警就先创建一个报警信息,写入报警列表,超过报警的最小误报时间后,
  67. * 再写入到EQM的tAlarmInfo表中,正在报警的记录就判断和更新,结束报警就更新tAlarmInfo的
  68. * 报警数据的结束时间,并清空pAlarmInfo
  69. * ------------------------------------------------------------------------------- */
  70. /* 找出数组中与当前报警ID相同的的报警信息 */
  71. auto pAlarmInfo = m_listAlarmBuffer.findAlarmInfo(*newAlarm);
  72. /* 没有找到报警记录,就新建一个 */
  73. if(pAlarmInfo == nullptr)
  74. {
  75. /* 判断这条记录有没有报警,没有就不做任何处理 */
  76. if(!newAlarm->Is_Alarm)
  77. {
  78. // SPDLOG_LOGGER_TRACE(m_logger, "{}, 没有报警,直接跳过", tmpBaseLog);
  79. continue;
  80. }
  81. /* 新记录有报警,新建报警记录 */
  82. if(newAlarm->Is_Alarm)
  83. {
  84. /* 图片不能是空,如果是空的,就不写入数据库,现在是先加入缓存中 */
  85. if(!newAlarm->ImageInfo.empty())
  86. {
  87. /* 违禁品检测,开始时间是事件时间 */
  88. newAlarm->StartTime = newAlarm->EventTime;
  89. newAlarm->EndTime = GVariable.nullTime;
  90. /* 保存新的报警记录 */
  91. newAlarm->Is_InsertEQM = false;
  92. m_listAlarmBuffer.addAlarmInfo(*newAlarm);
  93. SPDLOG_LOGGER_INFO(m_logger, "{} 新的报警,加入到缓存中", tmpBaseLog);
  94. }else
  95. {
  96. SPDLOG_LOGGER_WARN(m_logger, "{}, 有报警区域, 但是没有图片信息", tmpBaseLog);
  97. }
  98. }
  99. } else
  100. {
  101. /* 已有报警记录 */
  102. /* 更新图片信息 */
  103. if(!newAlarm->ImageInfo.empty())
  104. {
  105. pAlarmInfo->ImageInfo = newAlarm->ImageInfo;
  106. }
  107. /* 判断这条记录有无报警 */
  108. if(newAlarm->Is_Alarm)
  109. {
  110. /* 正在报警中,检查是否已经写入到数据库中 */
  111. if(pAlarmInfo->Is_InsertEQM)
  112. {
  113. /* 已经写入到数据库中,不做处理 */
  114. }else
  115. {
  116. /* 判断报警时长间隔是否足够 */
  117. if(!isOutAlarmTimeVaild(*pAlarmInfo))
  118. {
  119. /* 不够报警时间,直接进行下一个循环 */
  120. continue;
  121. }
  122. /* 写入到数据库 */
  123. pAlarmInfo->Is_InsertEQM = false;
  124. int pkid = 0;
  125. if(m_fromWebAPI->insertAlarmInfo(*pAlarmInfo, pkid))
  126. {
  127. pAlarmInfo->Is_InsertEQM = true;
  128. pAlarmInfo->PKID = pkid;
  129. }else {
  130. SPDLOG_LOGGER_ERROR(m_logger, "{} 写入tAlarmInfo报警数据失败! ", tmpBaseLog);
  131. }
  132. SPDLOG_LOGGER_INFO(m_logger, "{}, 报警开始,写入数据库", tmpBaseLog);
  133. }
  134. } else
  135. {
  136. /* 报警结束,先判断有无写入数据库,已经写入,直接结束报警
  137. * 没有写入数据库,判断时长,如果小于设置的最小间隔,可能是误报警
  138. * 更新数据库结束时间,结束时间是此时电脑时间 */
  139. pAlarmInfo->EndTime = m_nowTime;
  140. /* 判断是否已经写入到数据库中 */
  141. if(pAlarmInfo->Is_InsertEQM)
  142. {
  143. /* 已经写入到数据库中,更新结束时间 */
  144. m_fromWebAPI->endAlarmInfoByPKID(pAlarmInfo->PKID, m_nowTime);
  145. }else
  146. {
  147. /* 判断报警时长间隔是否足够 */
  148. if(isOutAlarmTimeVaild(*pAlarmInfo))
  149. {
  150. /* 时长足够且没有写入到数据库中,写入到数据库中 */
  151. m_fromWebAPI->insertAlarmInfo(*pAlarmInfo);
  152. }
  153. }
  154. SPDLOG_LOGGER_INFO(m_logger, "{} 报警结束", tmpBaseLog);
  155. /* 从缓存中删除这个报警信息 */
  156. m_listAlarmBuffer.deleteAlarmInfo(*pAlarmInfo);
  157. }
  158. }
  159. }
  160. }
  161. m_listSrcAlarm.clearAlarmInfo();
  162. m_listAlarmBuffer.clearAlarmInfo();
  163. }
  164. /* old处理方法 */
  165. void FuncOrdinary::oldTask()
  166. {
  167. // /* 读取Redis信息,处理数据,第一层循环是根据房间读取,第二个循环是根据房间内的摄像机读取 */
  168. // for(const auto& roomInfo : m_funcThreadInfo.listRoomCamActInfo)
  169. // {
  170. // for(const auto& it : roomInfo.mapCameraAction)
  171. // {
  172. // for(const auto act : it.second)
  173. // {
  174. // /* 读取Redis数据 */
  175. // std::string strKey = std::to_string(it.first) + ":" + act;
  176. // std::string strRetValue;
  177. // if(!m_fromRedis->getRedisString(strKey, strRetValue))
  178. // {
  179. // SPDLOG_LOGGER_ERROR(m_logger, "读取Redis数据失败, Key:{}", strKey);
  180. // std::this_thread::sleep_for(std::chrono::milliseconds(10));
  181. // continue;
  182. // }
  183. // /* 解析数据,先设置基础数据 */
  184. // AlarmInfo newAlarmInfo;
  185. // newAlarmInfo.ActionID = act;
  186. // newAlarmInfo.appFunction = m_funcThreadInfo.appFunction;
  187. // parseRedisBaseData(strRetValue, newAlarmInfo);
  188. // parseRedisBBoxesData(strRetValue, newAlarmInfo);
  189. // /* 判断事件的时效性,超过多少秒不更新就可能是超脑挂了 */
  190. // if(isEventTimeVaild(newAlarmInfo.EventTime))
  191. // {
  192. // SPDLOG_LOGGER_WARN(m_logger, "Redis Key:{} 数据长时间没有更新,EventTime:{}",strKey, newAlarmInfo.EventTime);
  193. // std::this_thread::sleep_for(std::chrono::milliseconds(100));
  194. // continue;
  195. // }
  196. // /* -------------------------------------------------------------------------------
  197. // * 判断有无报警记录,新的报警就先创建一个报警信息,写入报警列表,超过报警的最小误报时间后,
  198. // * 再写入到EQM的tAlarmInfo表中,正在报警的记录就判断和更新,结束报警就更新tAlarmInfo的
  199. // * 报警数据的结束时间,并清空pAlarmInfo
  200. // * ------------------------------------------------------------------------------- */
  201. // /* 找出数组中与当前报警ID相同的的报警信息 */
  202. // auto pAlarmInfo = m_listAlarm.findAlarmInfo(newAlarmInfo);
  203. // /* 没有找到报警记录,就新建一个 */
  204. // if(pAlarmInfo == nullptr)
  205. // {
  206. // /* 新记录有报警,新建报警记录 */
  207. // if(newAlarmInfo.Is_Alarm)
  208. // {
  209. // /* 图片不能是空,如果是空的,就不写入数据库 */
  210. // if(!newAlarmInfo.ImageInfo.empty())
  211. // {
  212. // /* 违禁品检测,开始时间是事件时间 */
  213. // newAlarmInfo.StartTime = newAlarmInfo.EventTime;
  214. // newAlarmInfo.EndTime = "";
  215. // /* 保存新的报警记录 */
  216. // newAlarmInfo.Is_InsertEQM = false;
  217. // m_listAlarm.addAlarmInfo(newAlarmInfo);
  218. // }else
  219. // {
  220. // SPDLOG_LOGGER_WARN(m_logger, "频道:{}, 房间:{}, 摄像机:{}, 算法:{}, 有报警区域, 但是没有图片信息", m_funcThreadInfo.ChannelID, roomInfo.RoomID, it.first, act);
  221. // }
  222. // }
  223. // }
  224. // /* 已有报警记录 */
  225. // else
  226. // {
  227. // /* 更新图片信息 */
  228. // if(!newAlarmInfo.ImageInfo.empty())
  229. // {
  230. // pAlarmInfo->ImageInfo = newAlarmInfo.ImageInfo;
  231. // }
  232. // /* 判断是否还在报警中 */
  233. // if(newAlarmInfo.Is_Alarm)
  234. // {
  235. // /* 正在报警中,检查是否已经写入到数据库中 */
  236. // if(pAlarmInfo->Is_InsertEQM)
  237. // {
  238. // /* 已经写入到数据库中,不做处理 */
  239. // }else
  240. // {
  241. // /* 判断报警时长间隔是否足够 */
  242. // if(!isAlarmTimeVaild(*pAlarmInfo))
  243. // {
  244. // /* 不够报警时间,直接进行下一个循环 */
  245. // continue;
  246. // }
  247. // /* 写入到数据库 */
  248. // pAlarmInfo->Is_InsertEQM = false;
  249. // if(m_fromWebAPI->insertAlarmInfo(*pAlarmInfo))
  250. // {
  251. // pAlarmInfo->Is_InsertEQM = true;
  252. // }else {
  253. // SPDLOG_LOGGER_ERROR(m_logger, "写入tAlarmInfo报警数据失败, 频率:{}, 房间:{}, 算法:{} ", m_funcThreadInfo.ChannelID, roomInfo.RoomID, act);
  254. // }
  255. // }
  256. // } else
  257. // {
  258. // /* 报警结束,判断时长,如果小于设置的最小间隔,可能是误报警
  259. // * 更新数据库结束时间,结束时间是此时电脑时间 */
  260. // if(isAlarmTimeVaild(*pAlarmInfo))
  261. // {
  262. // pAlarmInfo->EndTime = chronoToStrTime(std::chrono::system_clock::now());
  263. // /* 判断是否已经写入到数据库中 */
  264. // if(pAlarmInfo->Is_InsertEQM)
  265. // {
  266. // /* 已经写入到数据库中,更新结束时间 */
  267. // m_fromWebAPI->updateAlarmEndTime(*pAlarmInfo);
  268. // }else {
  269. // /* 没有写入到数据库中,写入到数据库中 */
  270. // m_fromWebAPI->insertAlarmInfo(*pAlarmInfo);
  271. // }
  272. // SPDLOG_LOGGER_INFO(m_logger, "频率:{}, 房间:{}, 摄像机:{}, 算法:{}, 报警结束", m_funcThreadInfo.ChannelID, roomInfo.RoomID, it.first, act);
  273. // }
  274. // /* 删除这个报警信息 */
  275. // m_listAlarm.deleteAlarmInfo(*pAlarmInfo);
  276. // }
  277. // }
  278. // }
  279. // }
  280. // }
  281. }
  282. /* 判断报警时长是否符合不同应用的最小间隔 */
  283. bool FuncOrdinary::isOutAlarmTimeVaild(const AlarmInfo& info)
  284. {
  285. int alarmTime = 0;
  286. /* 违禁品物品 */
  287. if(info.appFunction == AppFunction::APP_Contraband)
  288. {
  289. alarmTime = GVariable.AppContraband_Time;
  290. }
  291. /* 玩手机识别 */
  292. else if(info.appFunction == AppFunction::APP_PlayPhone)
  293. {
  294. alarmTime = GVariable.AppPlayPhone;
  295. }
  296. /* 老鼠识别 */
  297. else if(info.appFunction == AppFunction::APP_Mouse)
  298. {
  299. alarmTime = GVariable.AppMouse;
  300. }
  301. /* 未戴口罩识别 */
  302. else if(info.appFunction == AppFunction::APP_NoMask)
  303. {
  304. alarmTime = GVariable.AppMask;
  305. }
  306. /* 疲劳检测 */
  307. else if(info.appFunction == AppFunction::APP_Fatigue)
  308. {
  309. alarmTime = GVariable.AppTired_Time;
  310. }
  311. /* 摔倒识别 */
  312. else if(info.appFunction == AppFunction::APP_AllDown)
  313. {
  314. return true;
  315. }
  316. if(info.EventTime.secsTo(m_nowTime) < alarmTime)
  317. {
  318. return false;
  319. }
  320. return true;
  321. }
  322. /* 读取Redis数据 */
  323. // void FuncOrdinary::readRedisData(std::list<AlarmInfo>& listAlarmTemp)
  324. // {
  325. // listAlarmTemp.clear();
  326. // /* 读取Redis数据 */
  327. // for(const auto& roomInfo : m_funcThreadInfo.listRoomCamActInfo)
  328. // {
  329. // for(const auto& cam : roomInfo.mapCameraAction)
  330. // {
  331. // for(const auto act : cam.second)
  332. // {
  333. // /* 读取Redis数据 */
  334. // std::string strKey = std::to_string(cam.first) + ":" + act;
  335. // std::string strRetValue;
  336. // // SPDLOG_LOGGER_DEBUG(m_logger, "读取Redis数据, Key:{}", strKey);
  337. // if(!m_fromRedis->getRedisString(strKey, strRetValue))
  338. // {
  339. // // SPDLOG_LOGGER_ERROR(m_logger, "读取Redis数据失败, Key:{}", strKey);
  340. // std::this_thread::sleep_for(std::chrono::milliseconds(10));
  341. // continue;
  342. // }
  343. // /* 解析数据,先设置基础数据 */
  344. // AlarmInfo newAlarmInfo;
  345. // newAlarmInfo.ChannelID = m_funcThreadInfo.ChannelID;
  346. // newAlarmInfo.appFunction = m_funcThreadInfo.appFunction;
  347. // newAlarmInfo.RoomID = roomInfo.RoomID;
  348. // newAlarmInfo.DeviceID = cam.first;
  349. // newAlarmInfo.ActionID = act;
  350. // parseRedisBaseData(strRetValue, newAlarmInfo);
  351. // parseRedisBBoxesData(strRetValue, newAlarmInfo);
  352. // /* 判断事件的时效性,超过多少秒不更新就可能是超脑挂了 */
  353. // if(!isEventTimeVaild(newAlarmInfo.EventTime))
  354. // {
  355. // /* 事件时间超过600秒,可能是超脑挂了 */
  356. // SPDLOG_LOGGER_WARN(m_logger, "Redis Key:{} 数据长时间没有更新,EventTime:{}",strKey, newAlarmInfo.EventTime.toString("yyyy-MM-dd hh:mm:ss").toStdString());
  357. // std::this_thread::sleep_for(std::chrono::milliseconds(100));
  358. // continue;
  359. // }
  360. // listAlarmTemp.push_back(newAlarmInfo);
  361. // printRawDataInfo(roomInfo, strKey, newAlarmInfo);
  362. // }
  363. // }
  364. // }
  365. // }