ResistAlarm.cpp 121 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360136113621363136413651366136713681369137013711372137313741375137613771378137913801381138213831384138513861387138813891390139113921393139413951396139713981399140014011402140314041405140614071408140914101411141214131414141514161417141814191420142114221423142414251426142714281429143014311432143314341435143614371438143914401441144214431444144514461447144814491450145114521453145414551456145714581459146014611462146314641465146614671468146914701471147214731474147514761477147814791480148114821483148414851486148714881489149014911492149314941495149614971498149915001501150215031504150515061507150815091510151115121513151415151516151715181519152015211522152315241525152615271528152915301531153215331534153515361537153815391540154115421543154415451546154715481549155015511552155315541555155615571558155915601561156215631564156515661567156815691570157115721573157415751576157715781579158015811582158315841585158615871588158915901591159215931594159515961597159815991600160116021603160416051606160716081609161016111612161316141615161616171618161916201621162216231624162516261627162816291630163116321633163416351636163716381639164016411642164316441645164616471648164916501651165216531654165516561657165816591660166116621663166416651666166716681669167016711672167316741675167616771678167916801681168216831684168516861687168816891690169116921693169416951696169716981699170017011702170317041705170617071708170917101711171217131714171517161717171817191720172117221723172417251726172717281729173017311732173317341735173617371738173917401741174217431744174517461747174817491750175117521753175417551756175717581759176017611762176317641765176617671768176917701771177217731774177517761777177817791780178117821783178417851786178717881789179017911792179317941795179617971798179918001801180218031804180518061807180818091810181118121813181418151816181718181819182018211822182318241825182618271828182918301831183218331834183518361837183818391840184118421843184418451846184718481849185018511852185318541855185618571858185918601861186218631864186518661867186818691870187118721873187418751876187718781879188018811882188318841885188618871888188918901891189218931894189518961897189818991900190119021903190419051906190719081909191019111912191319141915191619171918191919201921192219231924192519261927192819291930193119321933193419351936193719381939194019411942194319441945194619471948194919501951195219531954195519561957195819591960196119621963196419651966196719681969197019711972197319741975197619771978197919801981198219831984198519861987198819891990199119921993199419951996199719981999200020012002200320042005200620072008200920102011201220132014201520162017201820192020202120222023202420252026202720282029203020312032203320342035203620372038203920402041204220432044204520462047204820492050205120522053205420552056205720582059206020612062206320642065206620672068206920702071207220732074207520762077207820792080208120822083208420852086208720882089209020912092209320942095209620972098209921002101210221032104210521062107210821092110211121122113211421152116211721182119212021212122212321242125212621272128212921302131213221332134213521362137213821392140214121422143214421452146214721482149215021512152215321542155215621572158215921602161216221632164216521662167216821692170217121722173217421752176217721782179218021812182218321842185218621872188218921902191219221932194219521962197219821992200220122022203220422052206220722082209221022112212221322142215221622172218221922202221222222232224222522262227222822292230223122322233223422352236223722382239224022412242224322442245224622472248224922502251225222532254225522562257225822592260226122622263226422652266226722682269227022712272227322742275227622772278227922802281228222832284228522862287228822892290229122922293229422952296229722982299230023012302230323042305230623072308230923102311231223132314231523162317231823192320232123222323232423252326232723282329233023312332233323342335233623372338233923402341234223432344234523462347234823492350235123522353235423552356235723582359236023612362236323642365236623672368236923702371237223732374237523762377237823792380238123822383238423852386238723882389239023912392239323942395239623972398239924002401240224032404240524062407240824092410241124122413241424152416241724182419242024212422242324242425242624272428242924302431243224332434243524362437243824392440244124422443244424452446244724482449245024512452245324542455245624572458245924602461246224632464246524662467246824692470247124722473247424752476247724782479248024812482248324842485248624872488248924902491249224932494249524962497249824992500250125022503250425052506250725082509251025112512251325142515251625172518251925202521252225232524252525262527252825292530253125322533253425352536253725382539254025412542254325442545254625472548254925502551255225532554255525562557255825592560256125622563256425652566256725682569257025712572257325742575257625772578257925802581258225832584258525862587258825892590259125922593259425952596259725982599260026012602260326042605260626072608260926102611261226132614261526162617261826192620262126222623262426252626262726282629263026312632263326342635263626372638263926402641264226432644264526462647264826492650265126522653265426552656265726582659266026612662266326642665266626672668266926702671267226732674267526762677267826792680268126822683268426852686268726882689269026912692269326942695269626972698269927002701270227032704270527062707270827092710271127122713271427152716271727182719272027212722272327242725272627272728272927302731273227332734273527362737273827392740274127422743274427452746274727482749275027512752275327542755275627572758275927602761276227632764276527662767276827692770277127722773277427752776277727782779278027812782278327842785278627872788278927902791279227932794279527962797279827992800280128022803280428052806280728082809281028112812281328142815281628172818281928202821282228232824282528262827282828292830283128322833283428352836283728382839284028412842284328442845284628472848284928502851285228532854285528562857285828592860286128622863286428652866286728682869287028712872287328742875287628772878287928802881288228832884288528862887288828892890289128922893289428952896289728982899290029012902290329042905290629072908290929102911291229132914291529162917291829192920292129222923292429252926292729282929293029312932293329342935293629372938293929402941294229432944294529462947294829492950295129522953295429552956295729582959296029612962296329642965296629672968296929702971297229732974297529762977297829792980298129822983298429852986298729882989299029912992299329942995299629972998299930003001300230033004300530063007300830093010301130123013301430153016301730183019302030213022302330243025302630273028302930303031303230333034303530363037303830393040304130423043304430453046304730483049305030513052305330543055305630573058305930603061306230633064306530663067306830693070307130723073307430753076307730783079308030813082308330843085308630873088308930903091309230933094309530963097309830993100310131023103310431053106310731083109311031113112311331143115311631173118311931203121312231233124312531263127312831293130313131323133313431353136313731383139314031413142314331443145314631473148314931503151315231533154315531563157315831593160316131623163316431653166316731683169317031713172317331743175317631773178317931803181318231833184318531863187318831893190319131923193319431953196319731983199320032013202320332043205320632073208320932103211321232133214321532163217321832193220322132223223322432253226322732283229323032313232323332343235323632373238323932403241324232433244324532463247324832493250325132523253325432553256325732583259326032613262326332643265326632673268326932703271327232733274327532763277327832793280328132823283328432853286328732883289329032913292329332943295329632973298329933003301330233033304330533063307330833093310331133123313331433153316331733183319332033213322332333243325332633273328332933303331333233333334333533363337333833393340334133423343334433453346334733483349335033513352335333543355335633573358335933603361336233633364336533663367336833693370337133723373337433753376337733783379338033813382338333843385338633873388338933903391339233933394339533963397339833993400340134023403340434053406340734083409341034113412341334143415341634173418341934203421342234233424342534263427342834293430343134323433343434353436343734383439344034413442344334443445344634473448344934503451345234533454345534563457
  1. #include "stdafx.h"
  2. #include "ResistAlarm.h"
  3. #include "MonitorObject.h"
  4. #include "Device.h"
  5. #include <ODBC/DBConnectPool.h>
  6. #include <rapidjson/document.h>
  7. #include <rapidjson/writer.h>
  8. #include <rapidjson/stringbuffer.h>
  9. #include <Simplelog.h>
  10. #include <sstream>
  11. #include <gbk2utf8.h>
  12. #include "AppService.h"
  13. #include <yyjson.h>
  14. #include <315ClientManager.h>
  15. #include "SkylightMng.h"
  16. CResistAlarm::CResistAlarm()
  17. {
  18. }
  19. CResistAlarm::~CResistAlarm()
  20. {
  21. }
  22. CResistAlarmMng::CResistAlarmMng()
  23. {
  24. }
  25. CResistAlarmMng::~CResistAlarmMng()
  26. {
  27. }
  28. BOOL CResistAlarmMng::Start()
  29. {
  30. LoadAlarmSet();
  31. if (FALSE == LoadAlarmID()) return FALSE;
  32. LoadUnAck();
  33. m_bWork = true;
  34. m_pThreadAlarmSet = new thread(CResistAlarmMng::ThreadProcAlarmSet, (DWORD_PTR)this);
  35. if (m_pThreadAlarmSet == nullptr) return FALSE;
  36. m_pThreadDevice = new thread(CResistAlarmMng::ThreadProcDevice, (DWORD_PTR)this);
  37. if (m_pThreadDevice == nullptr) return FALSE;
  38. m_pThreadMove = new thread(CResistAlarmMng::ThreadProcMove, (DWORD_PTR)this);
  39. if (m_pThreadMove == nullptr) return FALSE;
  40. return TRUE;
  41. }
  42. void CResistAlarmMng::Stop()
  43. {
  44. m_bWork = false;
  45. if (m_pThreadAlarmSet)
  46. {
  47. m_pThreadAlarmSet->join();
  48. delete m_pThreadAlarmSet;
  49. m_pThreadAlarmSet = nullptr;
  50. }
  51. if (m_pThreadDevice)
  52. {
  53. m_pThreadDevice->join();
  54. delete m_pThreadDevice;
  55. m_pThreadDevice = nullptr;
  56. }
  57. if (m_pThreadMove)
  58. {
  59. m_pThreadMove->join();
  60. delete m_pThreadMove;
  61. m_pThreadMove = nullptr;
  62. }
  63. for (auto& it : m_alarm_set)
  64. {
  65. if (it.second)
  66. {
  67. if (it.second->type == eZL_ALARMTYPE::MAX_OVER_LIMIT)
  68. {
  69. delete (MAX_OVER_LIMIT_INFO*)it.second;
  70. it.second = nullptr;
  71. }
  72. else if (it.second->type == eZL_ALARMTYPE::FRICTION_OVER_LIMIT)
  73. {
  74. delete (FRICTION_OVER_LIMIT_INFO*)it.second;
  75. it.second = nullptr;
  76. }
  77. else if (it.second->type == eZL_ALARMTYPE::CONVERT_LIMIT)
  78. {
  79. delete (CONVERT_RESIST_OVER_LIMIT*)it.second;
  80. it.second = nullptr;
  81. }
  82. else if (it.second->type == eZL_ALARMTYPE::SUOBI_LOCK_LIMIT)
  83. {
  84. delete (SUOBI_OVER_LIMIT_INFO*)it.second;
  85. it.second = nullptr;
  86. }
  87. else
  88. {
  89. assert(0);
  90. }
  91. }
  92. }
  93. m_alarm_set.clear();
  94. for (auto& it : m_lstUnConfirmAlarm) delete it;
  95. m_lstUnConfirmAlarm.clear();
  96. }
  97. //BASE_INFO* CResistAlarmMng::Find(string momp, uint8_t no, uint8_t type)
  98. //{
  99. // stringstream ss;
  100. // ss << momp << '.' << to_string(no) << '.' << to_string(type);
  101. // auto it = m_alarm_set.find(ss.str());
  102. // if (it != m_alarm_set.end())
  103. // {
  104. // return it->second;
  105. // }
  106. // return nullptr;
  107. //}
  108. BASE_INFO* CResistAlarmMng::Find(string mo, string mp, uint8_t no, eZL_ALARMTYPE type)
  109. {
  110. stringstream ss;
  111. ss << mo << '.' << mp << '.' << to_string(no) << '.' << to_string((uint8_t)type);
  112. auto it = m_alarm_set.find(ss.str());
  113. if (it != m_alarm_set.end())
  114. {
  115. return it->second;
  116. }
  117. return nullptr;
  118. }
  119. //bool CResistAlarmMng::Insert(const string& momp, uint8_t no, uint8_t type, BASE_INFO* info)
  120. //{
  121. // auto it = m_alarm_set.insert(make_pair(momp + '.' + to_string(no) + '.' + to_string(type), info));
  122. // return it.second;
  123. //}
  124. bool CResistAlarmMng::Insert(string mo, string mp, uint8_t no, uint8_t type, BASE_INFO* info)
  125. {
  126. auto it = m_alarm_set.insert(make_pair(mo + '.' + mp + '.' + to_string(no) + '.' + to_string(type), info));
  127. return it.second;
  128. }
  129. //bool CResistAlarmMng::ConfirmAlarm(string mo, string mp, uint8_t no, uint8_t type, const SYSTEMTIME& st)
  130. //{
  131. // lock_guard<mutex> lock(m_mtxAlarm);
  132. // auto it = m_lstUnConfirmAlarm.begin();
  133. // for (it; it != m_lstUnConfirmAlarm.end(); ++it)
  134. // {
  135. // const auto& pInfo = *it;
  136. // if (pInfo->mo.compare(mo) == 0 && pInfo->mp.compare(mp) == 0 && pInfo->no == no && pInfo->type == type && pInfo->time.wYear == st.wYear &&
  137. // pInfo->time.wMonth == st.wMonth && pInfo->time.wDay == st.wDay && pInfo->time.wHour == st.wHour && pInfo->time.wMinute == st.wMinute)
  138. // {
  139. // m_lstUnConfirmAlarm.erase(it);
  140. // return true;
  141. // }
  142. // }
  143. // return false;
  144. //}
  145. bool CResistAlarmMng::AckAlarm(int alarm_id, string& name, CTime& time)
  146. {
  147. lock_guard<mutex> lock(m_mtxAlarm);
  148. auto it = m_lstUnConfirmAlarm.begin();
  149. for (it; it != m_lstUnConfirmAlarm.end(); ++it)
  150. {
  151. auto& pInfo = *it;
  152. if (alarm_id == pInfo->id)
  153. {
  154. pInfo->ack_result = 1;
  155. pInfo->ack_name = name;
  156. pInfo->ack_time = time;
  157. //g_p315ClientManager->GetTcpClient()->SendAlarmData(pInfo->no, 2, CTime(pInfo->time).GetTime(), time.GetTime(), (WORD)pInfo->type, pInfo->val, 0, 0);
  158. return true;
  159. }
  160. }
  161. return false;
  162. }
  163. bool CResistAlarmMng::HandleAlarm(int alarm_id)
  164. {
  165. lock_guard<mutex> lock(m_mtxAlarm);
  166. auto it = m_lstUnConfirmAlarm.begin();
  167. for (it; it != m_lstUnConfirmAlarm.end(); ++it)
  168. {
  169. const auto& pInfo = *it;
  170. if (pInfo->id == alarm_id && pInfo->ack_result == 1 && pInfo->recoveryTime.wYear > 2000)
  171. {
  172. m_lstUnConfirmAlarm.erase(it);
  173. return true;
  174. }
  175. }
  176. return true;
  177. }
  178. void CResistAlarmMng::GeneralAlarm(string mo, string mp, uint8_t no, eZL_ALARMTYPE type, uint8_t level, SYSTEMTIME& tAlarm)
  179. {
  180. ALARM_INFO* pAlarmInfo = nullptr;
  181. bool bNew = false;
  182. {
  183. std::lock_guard<mutex> lock(m_mtxAlarm);
  184. for (const auto& alarm : m_lstUnConfirmAlarm)
  185. {
  186. if (alarm->type == type && alarm->mo.compare(mo) == 0 && alarm->recoveryTime.wYear < 2000
  187. && alarm->mp.compare(mp) == 0 && no == alarm->no) //跟上次报警时间超过1小时,则算新报警
  188. {
  189. pAlarmInfo = alarm;
  190. break;
  191. }
  192. }
  193. }
  194. if (pAlarmInfo == nullptr)
  195. {
  196. bNew = true;
  197. pAlarmInfo = new ALARM_INFO;
  198. pAlarmInfo->event_id = hjfunc_GetGUID();
  199. pAlarmInfo->id = ++m_nAlarmID;
  200. pAlarmInfo->level = 1;
  201. pAlarmInfo->mo = mo;
  202. pAlarmInfo->mp = mp;
  203. pAlarmInfo->no = no;
  204. pAlarmInfo->time = tAlarm;
  205. pAlarmInfo->type = type;
  206. pAlarmInfo->val = 0;
  207. char szInfo[200];
  208. sprintf_s(szInfo, sizeof(szInfo), "传感器异常时间:%04d-%02d-%02d %02d:%02d:%02d",
  209. tAlarm.wYear, tAlarm.wMonth, tAlarm.wDay, tAlarm.wHour, tAlarm.wMinute, tAlarm.wSecond);
  210. pAlarmInfo->desc = szInfo;
  211. lock_guard<mutex> lock(m_mtxAlarm);
  212. m_lstUnConfirmAlarm.push_back(pAlarmInfo);
  213. }
  214. //send
  215. if (bNew) //不再推送
  216. {
  217. rapidjson::StringBuffer buffer;
  218. auto ret = AlarmInfo2Pack(pAlarmInfo, buffer);
  219. const char* output = buffer.GetString();
  220. //CAppService::Instance()->GetLwsServer()->SendPackToALLClient_with_noEncode((uint8_t*)output, buffer.GetLength());
  221. CAppService::Instance()->GetMgServer()->SendToAllClient(output, buffer.GetLength());
  222. }
  223. //save
  224. if (bNew)
  225. {
  226. CTime ctAlarmTime(tAlarm);
  227. CString sql;
  228. sql.Format("INSERT INTO [rm_alarm]([ID],[mo],[mp],[no],[type],[occur_time],[level],[desc],[suggest],[val],[event_id],[rel_id],posi,loworhigh,referval,[sunroof]) "\
  229. "VALUES(%d, '%s', '%s', %d, %d, '%s', %d, '%s', '%s', %d, '%s', '%s',%d,%d,%d,%d);",
  230. pAlarmInfo->id, pAlarmInfo->mo.c_str(), pAlarmInfo->mp.c_str(), pAlarmInfo->no, (int)pAlarmInfo->type,
  231. (LPCSTR)(ctAlarmTime.Format("%Y-%m-%d %H:%M:%S")), pAlarmInfo->level, pAlarmInfo->desc.c_str(), pAlarmInfo->suggest.c_str(), pAlarmInfo->val,
  232. pAlarmInfo->event_id.c_str(), pAlarmInfo->rel_id.c_str(), uint8_t(pAlarmInfo->posi), uint8_t(pAlarmInfo->loworhigh), pAlarmInfo->refer_val, pAlarmInfo->sunroof);
  233. if (false == CDBConnectPool::Instance()->DBExecuteSQL(sql))
  234. CSimpleLog::Error("执行语句失败" + sql);
  235. else
  236. {
  237. uint8_t* pack = nullptr; int len = 0;
  238. auto packno = CResistAlarmMng::GeneralNewAlarmData(pAlarmInfo, &pack, &len);
  239. if (packno) CSuperManager::Instance()->SendPack(pack, len, packno, E_ZL_PROTOCAL::ZL_ALARM_DATA, true);
  240. //g_p315ClientManager->GetTcpClient()->SendAlarmData(pAlarmInfo->no, 1, ctAlarmTime.GetTime(), 0xFFFFFFFF, (WORD)pAlarmInfo->type, pAlarmInfo->val, 0, 0);
  241. delete[] pack;
  242. pack = nullptr;
  243. }
  244. }
  245. }
  246. //报警恢复, 报警恢复逻辑与受理确认逻辑分开
  247. void CResistAlarmMng::RecoverAlarm(const string mo, const string mp, const uint8_t no, const eZL_ALARMTYPE type, const uint8_t level, const SYSTEMTIME& tAlarm)
  248. {
  249. vector<uint32_t> vctAlarmID;
  250. vector<string> vctEventID;
  251. //ALARM_INFO tmpalarm;
  252. {
  253. std::lock_guard<mutex> lock(m_mtxAlarm);
  254. for (auto it = m_lstUnConfirmAlarm.begin(); it != m_lstUnConfirmAlarm.end();)
  255. {
  256. auto pAlarm = *it;
  257. if (pAlarm->type == type && pAlarm->mo.compare(mo) == 0 && pAlarm->mp.compare(mp) == 0 && no == pAlarm->no && level == pAlarm->level
  258. && pAlarm->recoveryTime.wYear < 2000)
  259. {
  260. vctAlarmID.emplace_back(pAlarm->id);
  261. vctEventID.emplace_back(pAlarm->event_id);
  262. //CopyMemory(&tmpalarm, *it, sizeof(tmpalarm)); 有string 不能拷贝
  263. //tmpalarm = *pAlarm;
  264. if (pAlarm->ack_result == 1)//如果已确认则可以删除
  265. {
  266. it = m_lstUnConfirmAlarm.erase(it);
  267. continue;
  268. }
  269. else
  270. memcpy(&pAlarm->recoveryTime, &tAlarm, sizeof(SYSTEMTIME));
  271. }
  272. it++;
  273. }
  274. }
  275. if (vctAlarmID.size() == 0)
  276. return;
  277. CTime ctAlarm(tAlarm);
  278. string recovery_time = ctAlarm.Format("%Y-%m-%d %H:%M:%S");
  279. //g_p315ClientManager->GetTcpClient()->SendAlarmData(tmpalarm.no, 2, CTime(tmpalarm.time).GetTime(), ctAlarm.GetTime(), (WORD)tmpalarm.type, tmpalarm.val, 0, 0);
  280. //更新数据库
  281. {
  282. CString sql;
  283. for (auto& it : vctAlarmID)
  284. {
  285. sql.Format("UPDATE [rm_alarm] SET [recovery_time]='%s' WHERE ID = %d;", recovery_time.c_str(), it);
  286. CDBConnectPool::Instance()->DBExecuteSQL(sql);
  287. }
  288. //for (auto& it : vctEventID)
  289. //{
  290. // sql.Format("UPDATE [rm_alarm] SET [recovery_time]='%s' WHERE [event_id] = '%s';", recovery_time.c_str(), it);
  291. // CDBConnectPool::Instance()->DBExecuteSQL(sql);
  292. //}
  293. }
  294. //恢复暂时不通知
  295. //auto doc = yyjson_mut_doc_new(nullptr);
  296. //auto root = yyjson_mut_obj(doc);
  297. //yyjson_mut_doc_set_root(doc, root);
  298. //yyjson_mut_obj_add_str(doc, root, "cmd", "alm_ack");
  299. //yyjson_mut_obj_add_uint(doc, root, "alarm_id", alarm_id);
  300. //yyjson_mut_obj_add_strcpy(doc, root, "ack_time", recovery_time.c_str());
  301. //yyjson_mut_obj_add_strcpy(doc, root, "ack_name", ANSItoUTF8("系统").c_str());
  302. //size_t len;
  303. //auto json = yyjson_mut_write(doc, 0, &len);
  304. //if (json)
  305. //{
  306. // CAppService::Instance()->GetMgServer()->SendToAllClient(json, len);
  307. // free(json);
  308. //}
  309. //yyjson_mut_doc_free(doc);
  310. }
  311. eZL_MP_STAT CResistAlarmMng::GetAlarmStat(string& mo, string& mp, SYSTEMTIME& stAlarm)
  312. {
  313. eZL_MP_STAT stat = eZL_MP_STAT::MP_STAT_UNKNOW;
  314. lock_guard<mutex> lock(m_mtxAlarm);
  315. auto it = m_lstUnConfirmAlarm.rbegin();
  316. for (it; it != m_lstUnConfirmAlarm.rend(); ++it)
  317. {
  318. const auto& pInfo = *it;
  319. if (pInfo->ack_result) continue;//受理了暂时放过
  320. if (pInfo->mo.compare(mo) == 0 && pInfo->mp.compare(mp) == 0 && pInfo->recoveryTime.wYear < 2000)
  321. {
  322. if (pInfo->type == eZL_ALARMTYPE::EQUIP_OFFLINE || pInfo->type == eZL_ALARMTYPE::SENSOR_ABNORMAL)
  323. {
  324. stat = eZL_MP_STAT::MP_STAT_OFFLINE_GRAY;
  325. break;
  326. }
  327. else if (pInfo->level == 1)
  328. {
  329. stAlarm = pInfo->time;
  330. stat = eZL_MP_STAT::MP_STAT_ALARM_RED;
  331. break;
  332. }
  333. else if (pInfo->level == 0)
  334. {
  335. stAlarm = pInfo->time;
  336. stat = eZL_MP_STAT::MP_STAT_WARN_ORANGE;
  337. break;
  338. }
  339. }
  340. }
  341. return stat;
  342. }
  343. uint32_t CResistAlarmMng::GeneralAlarmSet(uint8_t** pack, int* len)
  344. {
  345. auto doc = yyjson_mut_doc_new(nullptr);
  346. auto root = yyjson_mut_arr(doc);
  347. yyjson_mut_doc_set_root(doc, root);
  348. for (const auto& it : m_alarm_set)
  349. {
  350. auto obj = yyjson_mut_obj(doc);
  351. yyjson_mut_arr_add_val(root, obj);
  352. auto ss = it.second;
  353. auto kk = it.first;
  354. int nPos = kk.find('.');
  355. int nPos2 = kk.find('.', nPos + 1);
  356. string mo = kk.substr(0, nPos);
  357. string mp = kk.substr(nPos + 1, nPos2 - nPos - 1);
  358. string momp = kk.substr(0, nPos2);
  359. string mmomp_name, station;
  360. CMonitorObjectMng::Instance()->GetStationNameByMomP(momp, station, mmomp_name);
  361. yyjson_mut_obj_add_strcpy(doc, obj, "mo", mo.c_str());
  362. yyjson_mut_obj_add_strcpy(doc, obj, "mp", mp.c_str());
  363. yyjson_mut_obj_add_strcpy(doc, obj, "station", station.c_str());
  364. yyjson_mut_obj_add_int(doc, obj, "no", ss->no);
  365. yyjson_mut_obj_add_int(doc, obj, "type", uint8_t(ss->type));
  366. switch (ss->type)
  367. {
  368. case eZL_ALARMTYPE::MAX_OVER_LIMIT:
  369. {
  370. auto conf = yyjson_mut_arr(doc);
  371. yyjson_mut_obj_add_val(doc, obj, "conf", conf);
  372. {
  373. auto object = yyjson_mut_obj(doc);
  374. yyjson_mut_arr_add_val(conf, object);
  375. yyjson_mut_obj_add_str(doc, object, "name", "enable");
  376. yyjson_mut_obj_add_str(doc, object, "val", ss->enable ? "true" : "false");
  377. }
  378. auto pInfo = (MAX_OVER_LIMIT_INFO*)ss;
  379. {
  380. auto object = yyjson_mut_obj(doc);
  381. yyjson_mut_arr_add_val(conf, object);
  382. if (ss->no == 2)
  383. yyjson_mut_obj_add_str(doc, object, "name", "d_alarm_high_limit");
  384. else
  385. yyjson_mut_obj_add_str(doc, object, "name", "lock_alarm_high_limit");
  386. yyjson_mut_obj_add_strcpy(doc, object, "val", to_string(pInfo->alarm_high_limit).c_str());
  387. }
  388. {
  389. auto object = yyjson_mut_obj(doc);
  390. yyjson_mut_arr_add_val(conf, object);
  391. if (ss->no == 2)
  392. yyjson_mut_obj_add_str(doc, object, "name", "d_warn_high_limit");
  393. else
  394. yyjson_mut_obj_add_str(doc, object, "name", "lock_warn_high_limit");
  395. yyjson_mut_obj_add_strcpy(doc, object, "val", to_string(pInfo->warn_high_limit).c_str());
  396. }
  397. {
  398. auto object = yyjson_mut_obj(doc);
  399. yyjson_mut_arr_add_val(conf, object);
  400. if (ss->no == 2)
  401. yyjson_mut_obj_add_str(doc, object, "name", "f_alarm_high_limit");
  402. else
  403. yyjson_mut_obj_add_str(doc, object, "name", "keep_alarm_high_limit");
  404. yyjson_mut_obj_add_strcpy(doc, object, "val", to_string(pInfo->f_alarm_high_limit).c_str());
  405. }
  406. {
  407. auto object = yyjson_mut_obj(doc);
  408. yyjson_mut_arr_add_val(conf, object);
  409. if (ss->no == 2)
  410. yyjson_mut_obj_add_str(doc, object, "name", "keep_warn_high_limit");
  411. else
  412. yyjson_mut_obj_add_str(doc, object, "name", "f_warn_high_limit");
  413. yyjson_mut_obj_add_strcpy(doc, object, "val", to_string(pInfo->f_warn_high_limit).c_str());
  414. }
  415. }
  416. break;
  417. case eZL_ALARMTYPE::FRICTION_OVER_LIMIT:
  418. {
  419. auto conf = yyjson_mut_arr(doc);
  420. yyjson_mut_obj_add_val(doc, obj, "conf", conf);
  421. {
  422. auto object = yyjson_mut_obj(doc);
  423. yyjson_mut_arr_add_val(conf, object);
  424. yyjson_mut_obj_add_str(doc, object, "name", "enable");
  425. yyjson_mut_obj_add_str(doc, object, "val", ss->enable ? "true" : "false");
  426. }
  427. auto pInfo = (FRICTION_OVER_LIMIT_INFO*)ss;
  428. {
  429. auto object = yyjson_mut_obj(doc);
  430. yyjson_mut_arr_add_val(conf, object);
  431. yyjson_mut_obj_add_str(doc, object, "name", "up_alarm_low_limit");
  432. yyjson_mut_obj_add_strcpy(doc, object, "val", to_string(pInfo->up_alarm_low_limit).c_str());
  433. }
  434. {
  435. auto object = yyjson_mut_obj(doc);
  436. yyjson_mut_arr_add_val(conf, object);
  437. yyjson_mut_obj_add_str(doc, object, "name", "up_warn_low_limit");
  438. yyjson_mut_obj_add_strcpy(doc, object, "val", to_string(pInfo->up_warn_low_limit).c_str());
  439. }
  440. {
  441. auto object = yyjson_mut_obj(doc);
  442. yyjson_mut_arr_add_val(conf, object);
  443. yyjson_mut_obj_add_str(doc, object, "name", "dw_alarm_high_limit");
  444. yyjson_mut_obj_add_strcpy(doc, object, "val", to_string(pInfo->dw_alarm_high_limit).c_str());
  445. }
  446. {
  447. auto object = yyjson_mut_obj(doc);
  448. yyjson_mut_arr_add_val(conf, object);
  449. yyjson_mut_obj_add_str(doc, object, "name", "dw_warn_high_limit");
  450. yyjson_mut_obj_add_strcpy(doc, object, "val", to_string(pInfo->dw_warn_high_limit).c_str());
  451. }
  452. }
  453. break;
  454. case eZL_ALARMTYPE::SUOBI_LOCK_LIMIT:
  455. {
  456. auto conf = yyjson_mut_obj(doc);
  457. yyjson_mut_obj_add_val(doc, obj, "conf", conf);
  458. auto pInfo = (SUOBI_OVER_LIMIT_INFO*)ss;
  459. yyjson_mut_obj_add_bool(doc, conf, "enable", pInfo->enable);
  460. yyjson_mut_obj_add_int(doc, conf, "alarm_low_limit", pInfo->alarm_low_limit);
  461. yyjson_mut_obj_add_int(doc, conf, "warn_low_limit", pInfo->warn_low_limit);
  462. yyjson_mut_obj_add_int(doc, conf, "alarm_high_limit", pInfo->alarm_high_limit);
  463. yyjson_mut_obj_add_int(doc, conf, "warn_high_limit", pInfo->warn_high_limit);
  464. }
  465. break;
  466. case eZL_ALARMTYPE::CONVERT_LIMIT:
  467. {
  468. auto conf = yyjson_mut_obj(doc);
  469. yyjson_mut_obj_add_val(doc, obj, "conf", conf);
  470. auto pInfo = (CONVERT_RESIST_OVER_LIMIT*)ss;
  471. yyjson_mut_obj_add_bool(doc, conf, "enable", pInfo->enable);
  472. yyjson_mut_obj_add_int(doc, conf, "dw_alarm_low_limit", pInfo->dw_alarm_low_limit);
  473. yyjson_mut_obj_add_int(doc, conf, "dw_warn_low_limit", pInfo->dw_warn_low_limit);
  474. yyjson_mut_obj_add_int(doc, conf, "up_alarm_high_limit", pInfo->up_alarm_high_limit);
  475. yyjson_mut_obj_add_int(doc, conf, "up_warn_high_limit", pInfo->up_warn_high_limit);
  476. }
  477. break;
  478. default:
  479. ASSERT(0);
  480. break;
  481. }
  482. }
  483. size_t json_len;
  484. auto json = yyjson_mut_write(doc, 0, &json_len);
  485. yyjson_mut_doc_free(doc);
  486. *len = json_len + sizeof(HJDATAHEAD2) + 2;
  487. *pack = new uint8_t[*len];
  488. auto no = CAppService::Instance()->GetPackNo();
  489. CHjDataConver::conver_sendpack(*pack, (LPBYTE)json, json_len, 0, no, E_ZL_PROTOCAL::ZL_ALARM_SET,
  490. OR_DATA_INFO(0, 0, 1, 2, OPT_TYPE::OPT_SYNC));
  491. free(json);
  492. return no;
  493. }
  494. uint32_t CResistAlarmMng::GeneralUnAck(uint8_t** pack, int* len)
  495. {
  496. auto doc = yyjson_mut_doc_new(nullptr);
  497. auto root = yyjson_mut_arr(doc);
  498. yyjson_mut_doc_set_root(doc, root);
  499. {
  500. //查询数据库近一天的数据
  501. lock_guard<mutex> lock(m_mtxAlarm);
  502. auto it = m_lstUnConfirmAlarm.begin();
  503. for (it; it != m_lstUnConfirmAlarm.end(); ++it)
  504. {
  505. auto& pInfo = *it;
  506. auto obj = yyjson_mut_obj(doc);
  507. yyjson_mut_arr_add_val(root, obj);
  508. yyjson_mut_obj_add_uint(doc, obj, "id", pInfo->id);
  509. yyjson_mut_obj_add_strcpy(doc, obj, "mo", pInfo->mo.c_str());
  510. yyjson_mut_obj_add_strcpy(doc, obj, "mp", pInfo->mp.c_str());
  511. yyjson_mut_obj_add_int(doc, obj, "no", pInfo->no);
  512. yyjson_mut_obj_add_int(doc, obj, "type", int(pInfo->type));
  513. yyjson_mut_obj_add_int(doc, obj, "level", pInfo->level);
  514. yyjson_mut_obj_add_strcpy(doc, obj, "time", fmt::format("{}-{:0>2}-{:0>2} {:0>2}:{:0>2}:{:0>2}",
  515. pInfo->time.wYear, pInfo->time.wMonth, pInfo->time.wDay, pInfo->time.wHour, pInfo->time.wMinute, pInfo->time.wSecond).c_str());
  516. yyjson_mut_obj_add_strcpy(doc, obj, "desc", pInfo->desc.c_str());
  517. yyjson_mut_obj_add_strcpy(doc, obj, "desc", pInfo->suggest.c_str());
  518. yyjson_mut_obj_add_int(doc, obj, "val", pInfo->val);
  519. yyjson_mut_obj_add_int(doc, obj, "ack_result", pInfo->ack_result);
  520. yyjson_mut_obj_add_strcpy(doc, obj, "ack_name", pInfo->ack_name.c_str());
  521. yyjson_mut_obj_add_int(doc, obj, "ack_time", pInfo->ack_time.GetTime());
  522. yyjson_mut_obj_add_strcpy(doc, obj, "event_id", pInfo->event_id.c_str());
  523. yyjson_mut_obj_add_strcpy(doc, obj, "rel_id", pInfo->rel_id.c_str());
  524. yyjson_mut_obj_add_strcpy(doc, obj, "recoverytime", fmt::format("{}-{:0>2}-{:0>2} {:0>2}:{:0>2}:{:0>2}",
  525. pInfo->recoveryTime.wYear, pInfo->recoveryTime.wMonth, pInfo->recoveryTime.wDay, pInfo->recoveryTime.wHour, pInfo->recoveryTime.wMinute, pInfo->recoveryTime.wSecond).c_str());
  526. }
  527. }
  528. size_t json_len;
  529. auto json = yyjson_mut_write(doc, 0, &json_len);
  530. yyjson_mut_doc_free(doc);
  531. *len = json_len + sizeof(HJDATAHEAD2) + 2;
  532. *pack = new uint8_t[*len];
  533. auto no = CAppService::Instance()->GetPackNo();
  534. CHjDataConver::conver_sendpack(*pack, (LPBYTE)json, json_len, 0, no, E_ZL_PROTOCAL::ZL_ALARM_UNACK,
  535. OR_DATA_INFO(0, 0, 1, 2, OPT_TYPE::OPT_SYNC));
  536. free(json);
  537. return no;
  538. }
  539. uint32_t CResistAlarmMng::GeneralNewAlarmData(ALARM_INFO* pInfo, uint8_t** pack, int* len)
  540. {
  541. auto doc = yyjson_mut_doc_new(nullptr);
  542. auto root = yyjson_mut_arr(doc);
  543. yyjson_mut_doc_set_root(doc, root);
  544. auto obj = yyjson_mut_obj(doc);
  545. yyjson_mut_arr_add_val(root, obj);
  546. yyjson_mut_obj_add_uint(doc, obj, "id", pInfo->id);
  547. yyjson_mut_obj_add_strcpy(doc, obj, "mo", pInfo->mo.c_str());
  548. yyjson_mut_obj_add_strcpy(doc, obj, "mp", pInfo->mp.c_str());
  549. yyjson_mut_obj_add_int(doc, obj, "no", pInfo->no);
  550. yyjson_mut_obj_add_int(doc, obj, "type", int(pInfo->type));
  551. yyjson_mut_obj_add_int(doc, obj, "level", pInfo->level);
  552. yyjson_mut_obj_add_strcpy(doc, obj, "time", fmt::format("{}-{:0>2}-{:0>2} {:0>2}:{:0>2}:{:0>2}",
  553. pInfo->time.wYear, pInfo->time.wMonth, pInfo->time.wDay, pInfo->time.wHour, pInfo->time.wMinute, pInfo->time.wSecond).c_str());
  554. yyjson_mut_obj_add_strcpy(doc, obj, "desc", pInfo->desc.c_str());
  555. yyjson_mut_obj_add_strcpy(doc, obj, "suggest", pInfo->suggest.c_str());
  556. yyjson_mut_obj_add_int(doc, obj, "val", pInfo->val);
  557. yyjson_mut_obj_add_int(doc, obj, "ack_result", pInfo->ack_result);
  558. yyjson_mut_obj_add_strcpy(doc, obj, "ack_name", pInfo->ack_name.c_str());
  559. yyjson_mut_obj_add_int(doc, obj, "ack_time", pInfo->ack_time.GetTime());
  560. yyjson_mut_obj_add_strcpy(doc, obj, "event_id", pInfo->event_id.c_str());
  561. yyjson_mut_obj_add_strcpy(doc, obj, "rel_id", pInfo->rel_id.c_str());
  562. yyjson_mut_obj_add_strcpy(doc, obj, "recoverytime", fmt::format("{}-{:0>2}-{:0>2} {:0>2}:{:0>2}:{:0>2}",
  563. pInfo->recoveryTime.wYear, pInfo->recoveryTime.wMonth, pInfo->recoveryTime.wDay, pInfo->recoveryTime.wHour, pInfo->recoveryTime.wMinute, pInfo->recoveryTime.wSecond).c_str());
  564. size_t json_len;
  565. auto json = yyjson_mut_write(doc, 0, &json_len);
  566. yyjson_mut_doc_free(doc);
  567. *len = json_len + sizeof(HJDATAHEAD2) + 2;
  568. *pack = new uint8_t[*len];
  569. auto no = CAppService::Instance()->GetPackNo();
  570. CHjDataConver::conver_sendpack(*pack, (LPBYTE)json, json_len, 0, no, E_ZL_PROTOCAL::ZL_ALARM_DATA,
  571. OR_DATA_INFO(0, 0, 1, 2, OPT_TYPE::OPT_ADD));
  572. free(json);
  573. return no;
  574. }
  575. bool CResistAlarmMng::AlarmInfo2Pack(const ALARM_INFO* pAlarmInfo, rapidjson::StringBuffer& buffer, bool bGb2312 /*= false*/)
  576. {
  577. char szInfo[200];
  578. char szUtf[200];
  579. using namespace rapidjson;
  580. Writer<StringBuffer> writer(buffer);
  581. writer.StartObject();
  582. writer.Key("cmd");
  583. writer.String("new_alarm");
  584. writer.Key("type");
  585. //if (pAlarmInfo->type == MAX_OVER_LIMIT) writer.String("monitor.alarm.max_over_limit");
  586. //else if (pAlarmInfo->type == EQUIP_OFFLINE) writer.String("monitor.alarm.equip_offline");
  587. //else if (pAlarmInfo->type == SENSOR_ABNORMAL) writer.String("monitor.alarm.sensor_abnormal");
  588. //else writer.String("");
  589. writer.Int((int)pAlarmInfo->type);
  590. writer.Key("tag");
  591. writer.String((pAlarmInfo->mo + '.' + pAlarmInfo->mp + '.' + to_string(pAlarmInfo->no + 1) + "#").c_str());
  592. string up, momp_name, up_name;
  593. string momp = pAlarmInfo->mo + "." + pAlarmInfo->mp;
  594. CMonitorObjectMng::Instance()->GetStationNameByMomP(momp, up, momp_name);
  595. auto pObject = CMonitorObjectMng::Instance()->GetTreeByID(up);
  596. if (pObject) up_name = pObject->name;
  597. string name1, name2, name3;
  598. CMonitorObjectMng::Instance()->GetNameByMoMp(momp, name1, name2, name3);
  599. if (pAlarmInfo->no == 0 && name1.length() == 0) name1 = "1号测力点";
  600. else if (pAlarmInfo->no == 1 && name2.length() == 0) name2 = "2号测力点";
  601. else if (pAlarmInfo->no == 2 && name3.length() == 0) name3 = "3号测力点";
  602. sprintf_s(szInfo, 200, "%s.%s.%s.%s%s", up_name.c_str(), momp_name.c_str(),
  603. pAlarmInfo->no == 0 ? name1.c_str() : pAlarmInfo->no == 1 ? name2.c_str() : pAlarmInfo->no == 2 ? name3.c_str() : "",
  604. GetAlarmName(pAlarmInfo->type).c_str(),
  605. pAlarmInfo->level == 1 ? "报警" : "预警"
  606. );
  607. gbk2utf8(szUtf, sizeof(szUtf), szInfo);
  608. writer.Key("title");
  609. writer.String(szUtf);
  610. gbk2utf8(szUtf, sizeof(szUtf), up.c_str());
  611. writer.Key("up");
  612. writer.String(szUtf);
  613. gbk2utf8(szUtf, sizeof(szUtf), up_name.c_str());
  614. writer.Key("up_name");
  615. writer.String(szUtf);
  616. writer.Key("alarm_id");
  617. writer.Uint(pAlarmInfo->id);
  618. writer.Key("event_id");
  619. writer.String(pAlarmInfo->event_id.c_str());
  620. writer.Key("rel_id");
  621. writer.String(pAlarmInfo->rel_id.c_str());
  622. writer.Key("sunroof");
  623. writer.Int(pAlarmInfo->sunroof);
  624. writer.Key("level");
  625. if (pAlarmInfo->level)
  626. writer.String("alarm");
  627. else
  628. writer.String("warn");
  629. writer.Key("occur_time");
  630. sprintf_s(szInfo, sizeof(szInfo), "%04d-%02d-%02d %02d:%02d:%02d.%03d",
  631. pAlarmInfo->time.wYear, pAlarmInfo->time.wMonth, pAlarmInfo->time.wDay,
  632. pAlarmInfo->time.wHour, pAlarmInfo->time.wMinute, pAlarmInfo->time.wSecond, pAlarmInfo->time.wMilliseconds);
  633. writer.String(szInfo);
  634. writer.Key("val");
  635. if (pAlarmInfo->type == eZL_ALARMTYPE::EQUIP_OFFLINE || pAlarmInfo->type == eZL_ALARMTYPE::SENSOR_ABNORMAL)
  636. writer.String("-");
  637. else
  638. writer.String(to_string(pAlarmInfo->val).c_str());
  639. writer.String("desc");
  640. gbk2utf8(szInfo, sizeof(szInfo), pAlarmInfo->desc.c_str());
  641. writer.String(szInfo);
  642. writer.Key("suggest");
  643. writer.String("");
  644. if (pAlarmInfo->ack_result)
  645. {
  646. //已受理的报警
  647. writer.String("ack_result");
  648. writer.Int(pAlarmInfo->ack_result);
  649. writer.Key("ack_name");
  650. writer.String(ANSItoUTF8(pAlarmInfo->ack_name).c_str());
  651. writer.Key("ack_time");
  652. writer.String(pAlarmInfo->ack_time.Format("%Y-%m-%d %H:%M:%S"));
  653. }
  654. if (pAlarmInfo->recoveryTime.wYear != 1900)
  655. {
  656. writer.Key("recoverytime");
  657. sprintf_s(szInfo, sizeof(szInfo), "%04d-%02d-%02d %02d:%02d:%02d.%03d",
  658. pAlarmInfo->recoveryTime.wYear, pAlarmInfo->recoveryTime.wMonth, pAlarmInfo->recoveryTime.wDay,
  659. pAlarmInfo->recoveryTime.wHour, pAlarmInfo->recoveryTime.wMinute, pAlarmInfo->recoveryTime.wSecond, pAlarmInfo->recoveryTime.wMilliseconds);
  660. writer.String(szInfo);
  661. }
  662. writer.EndObject();
  663. return true;
  664. }
  665. CResistAlarmMng CResistAlarmMng::obj;
  666. void CResistAlarmMng::InsertToDBByMove(string mo, string mp, time_t show_time, time_t start_time, time_t end_time,
  667. int curr_val, int show_val, uint8_t idx, uint8_t posi, uint8_t type,
  668. string mark)
  669. {
  670. if (show_time == 0) return;
  671. CTime ctShowTime(show_time / 1000);
  672. uint8_t sunroof = CSkylightMng::GetInstance()->IsSkylight(&ctShowTime); //2024年6月9日 添加天窗
  673. string table_name = fmt::format("rm_move_{:0>4}{:0>2}", ctShowTime.GetYear(), ctShowTime.GetMonth());
  674. auto strShowTime = (LPCSTR)ctShowTime.Format("%Y-%m-%d %H:%M:%S.") + to_string(show_time % 1000);
  675. auto strStartTime = start_time ? CTime(start_time).Format("%Y-%m-%d %H:%M:%S") : "";
  676. auto strEndTime = end_time ? CTime(end_time).Format("%Y-%m-%d %H:%M:%S") : "";
  677. CString sql = fmt::format("INSERT INTO {} (mo,mp,show_time,start_time,end_time,curr_val,show_val,idx,posi,[type],sunroof,mark) values ('{}', '{}', '{}', '{}','{}', {}, {}, {}, {}, {}, {}, '{}');",
  678. table_name, mo, mp, strShowTime, strStartTime, strEndTime, curr_val, show_val, idx, posi, type, sunroof, mark).c_str();
  679. if (CDBConnectPool::Instance()->DBExecuteSQL(sql) == FALSE)
  680. CSimpleLog::Error(sql);
  681. //上送
  682. //组装数据
  683. auto doc = yyjson_mut_doc_new(nullptr);
  684. auto root = yyjson_mut_arr(doc);
  685. yyjson_mut_doc_set_root(doc, root);
  686. //添加数据
  687. auto obj = yyjson_mut_obj(doc);
  688. yyjson_mut_arr_add_val(root, obj);
  689. yyjson_mut_obj_add_strcpy(doc, obj, "mo", mo.c_str());
  690. yyjson_mut_obj_add_strcpy(doc, obj, "mp", mp.c_str());
  691. yyjson_mut_obj_add_strcpy(doc, obj, "show_time", strShowTime.c_str());
  692. yyjson_mut_obj_add_strcpy(doc, obj, "start_time", strStartTime);
  693. yyjson_mut_obj_add_strcpy(doc, obj, "end_time", strEndTime);
  694. yyjson_mut_obj_add_strcpy(doc, obj, "mark", mark.c_str());
  695. yyjson_mut_obj_add_int(doc, obj, "curr_val", curr_val);
  696. yyjson_mut_obj_add_int(doc, obj, "show_val", show_val);
  697. yyjson_mut_obj_add_int(doc, obj, "idx", idx);
  698. yyjson_mut_obj_add_int(doc, obj, "posi", posi);
  699. yyjson_mut_obj_add_int(doc, obj, "sunroof", sunroof);
  700. size_t json_len;
  701. auto json = yyjson_mut_write(doc, 0, &json_len);
  702. auto send_len = json_len + sizeof(HJDATAHEAD2) + 2;
  703. auto pack = new uint8_t[send_len];
  704. auto packno = CAppService::Instance()->GetPackNo();
  705. CHjDataConver::conver_sendpack(pack, (LPBYTE)json, json_len, 0, packno, E_ZL_PROTOCAL::ZL_MOVE,
  706. OR_DATA_INFO(0, 0, 1, 2, OPT_TYPE::OPT_ADD));
  707. CSuperManager::Instance()->SendPack(pack, send_len, packno, E_ZL_PROTOCAL::ZL_MOVE, true);
  708. delete[] pack;
  709. pack = nullptr;
  710. free(json);
  711. yyjson_mut_doc_free(doc);
  712. }
  713. //判断锁闭力是否超限
  714. void CResistAlarmMng::JudgeAlarm(CResistAlarmMng* pService, SUOBI_OVER_LIMIT_INFO* pSuobiOverInfo,
  715. time_t show_time, int show_val, eSuoBiPosi posi, string mo, string mp, uint32_t zzjno)
  716. {
  717. if (pSuobiOverInfo == nullptr || pSuobiOverInfo->enable == false) return;
  718. bool bNew = false;
  719. ALARM_INFO* pAlarmInfo = nullptr;
  720. bool bAlarm = false, bWarn = false; int alarm_value = 0, alarm_refer; time_t alarm_time = show_time;
  721. CTime ctAlarmTime = CTime(alarm_time / 1000);
  722. eLowHigh loworhigh = eLowHigh::LH_UNKNOWN; //0:最小值 1:最大值
  723. {
  724. if (show_val < pSuobiOverInfo->alarm_low_limit)
  725. {
  726. bAlarm = true;
  727. alarm_refer = pSuobiOverInfo->alarm_low_limit;
  728. alarm_value = show_val;
  729. loworhigh = eLowHigh::LH_LOW;
  730. }
  731. else if (show_val < pSuobiOverInfo->warn_low_limit)
  732. {
  733. bWarn = true;
  734. alarm_refer = pSuobiOverInfo->warn_low_limit;
  735. alarm_value = show_val;
  736. loworhigh = eLowHigh::LH_LOW;
  737. }
  738. if (show_val > pSuobiOverInfo->alarm_high_limit)
  739. {
  740. bAlarm = true;
  741. alarm_refer = pSuobiOverInfo->alarm_high_limit;
  742. alarm_value = show_val;
  743. loworhigh = eLowHigh::LH_HIGH;
  744. }
  745. else if (show_val > pSuobiOverInfo->warn_high_limit)
  746. {
  747. bWarn = true;
  748. alarm_refer = pSuobiOverInfo->warn_high_limit;
  749. alarm_value = show_val;
  750. loworhigh = eLowHigh::LH_HIGH;
  751. }
  752. if (bAlarm == false && bWarn == false)
  753. {
  754. {
  755. std::lock_guard<mutex> lock(pService->m_mtxAlarm);
  756. for (auto it = pService->m_lstUnConfirmAlarm.begin(); it != pService->m_lstUnConfirmAlarm.end();)
  757. {
  758. auto pAlarm = *it;
  759. if (pAlarm->no == (uint8_t)posi && pAlarm->type == eZL_ALARMTYPE::SUOBI_LOCK_LIMIT && pAlarm->recoveryTime.wYear < 2000 //预警和报警单独计算
  760. && pAlarm->mo.compare(mo) == 0 && pAlarm->mp.compare(mp) == 0)
  761. {
  762. pAlarmInfo = pAlarm;
  763. if (pAlarmInfo)
  764. {
  765. //恢复
  766. //1. 更新数据库
  767. //2. 更新内存数据
  768. //3. 上送恢复消息
  769. if (ctAlarmTime.GetTime() == 0) ctAlarmTime = CTime::GetCurrentTime();
  770. string recovery_time = ctAlarmTime.Format("%Y-%m-%d %H:%M:%S");
  771. CString sql;
  772. sql.Format("UPDATE [rm_alarm] SET [recovery_time]='%s' WHERE ID = %d;", recovery_time.c_str(), pAlarmInfo->id);
  773. CDBConnectPool::Instance()->DBExecuteSQL(sql);
  774. if (pAlarmInfo->ack_result == 1)
  775. {
  776. it = pService->m_lstUnConfirmAlarm.erase(it);
  777. continue;
  778. }
  779. else
  780. {
  781. ctAlarmTime.GetAsSystemTime(pAlarm->recoveryTime);
  782. }
  783. g_p315ClientManager->GetTcpClient()->SendAlarmData(zzjno, 2, CTime(pAlarmInfo->time).GetTime(), alarm_time / 1000,
  784. pAlarmInfo->type, posi == eSuoBiPosi::SB_FIX ? eDaoChaPosi::DCP_FIX : eDaoChaPosi::DCP_INVERT, pAlarmInfo->level, pAlarm->loworhigh,
  785. TIEDA_ACQ_VALUE(pAlarm->val, TIEDA_VAL_STATE::TVS_MOVEING), TIEDA_ACQ_VALUE(pAlarm->refer_val, TIEDA_VAL_STATE::TVS_MOVEING), 0);
  786. }
  787. }
  788. it++;
  789. }
  790. }
  791. return;
  792. }
  793. //查找是否原有的报警已存在
  794. uint8_t level = 0;
  795. if (bAlarm) level = 1;
  796. {
  797. std::lock_guard<mutex> lock(pService->m_mtxAlarm);
  798. for (const auto& alarm : pService->m_lstUnConfirmAlarm)
  799. {
  800. if (alarm->no == (uint8_t)posi && alarm->type == eZL_ALARMTYPE::SUOBI_LOCK_LIMIT && level == alarm->level && alarm->recoveryTime.wYear < 2000 //预警和报警单独计算
  801. && alarm->mo.compare(mo) == 0 && alarm->mp.compare(mp) == 0)
  802. {
  803. pAlarmInfo = alarm;
  804. break;
  805. }
  806. }
  807. }
  808. ALARM_INFO *pNewAlarmInfo = nullptr;
  809. {
  810. bNew = true;
  811. pNewAlarmInfo = new ALARM_INFO;
  812. pNewAlarmInfo->event_id = hjfunc_GetGUID();
  813. pNewAlarmInfo->rel_id = pAlarmInfo ? pAlarmInfo->event_id : "";
  814. pNewAlarmInfo->id = ++pService->m_nAlarmID;
  815. pNewAlarmInfo->level = (bAlarm == 1);
  816. pNewAlarmInfo->mo = mo;
  817. pNewAlarmInfo->mp = mp;
  818. pNewAlarmInfo->no = (uint8_t)posi;
  819. ctAlarmTime.GetAsSystemTime(pNewAlarmInfo->time);
  820. pNewAlarmInfo->time.wMilliseconds = alarm_time % 1000;
  821. pNewAlarmInfo->type = eZL_ALARMTYPE::SUOBI_LOCK_LIMIT;
  822. pNewAlarmInfo->val = alarm_value;
  823. pNewAlarmInfo->refer_val = alarm_refer;
  824. pNewAlarmInfo->posi = (posi == eSuoBiPosi::SB_FIX ? eDaoChaPosi::DCP_FIX : eDaoChaPosi::DCP_INVERT);
  825. pNewAlarmInfo->loworhigh = loworhigh;
  826. pNewAlarmInfo->zzjno = zzjno;
  827. pNewAlarmInfo->sunroof = CSkylightMng::GetInstance()->IsSkylight(&pNewAlarmInfo->time);
  828. char szInfo[200] = { 0 };
  829. sprintf_s(szInfo, sizeof(szInfo), "锁闭力超限,位置:%s, 报警值为%dN, 参考值为%dN",
  830. posi == eSuoBiPosi::SB_FIX ? "定位" : "反位", alarm_value, alarm_refer);
  831. pNewAlarmInfo->desc = szInfo;
  832. lock_guard<mutex> lock(pService->m_mtxAlarm);
  833. pService->m_lstUnConfirmAlarm.push_back(pNewAlarmInfo);
  834. }
  835. //send
  836. //if (bNew) //不再推送
  837. {
  838. rapidjson::StringBuffer buffer;
  839. auto ret = AlarmInfo2Pack(pNewAlarmInfo, buffer);
  840. const char* output = buffer.GetString();
  841. //CAppService::Instance()->GetLwsServer()->SendPackToALLClient_with_noEncode((uint8_t*)output, buffer.GetLength());
  842. CAppService::Instance()->GetMgServer()->SendToAllClient(output, buffer.GetLength());
  843. }
  844. //主动推送315
  845. //if (bNew)
  846. {
  847. g_p315ClientManager->GetTcpClient()->SendAlarmData(pNewAlarmInfo->zzjno, 1, CTime(pNewAlarmInfo->time).GetTime(), 0xFFFFFFFF,
  848. pNewAlarmInfo->type, posi == eSuoBiPosi::SB_FIX ? eDaoChaPosi::DCP_FIX : eDaoChaPosi::DCP_INVERT, pNewAlarmInfo->level, loworhigh,
  849. TIEDA_ACQ_VALUE(alarm_value, TIEDA_VAL_STATE::TVS_MOVEING), TIEDA_ACQ_VALUE(alarm_refer, TIEDA_VAL_STATE::TVS_MOVEING), 0);
  850. }
  851. //save
  852. //if (bNew)
  853. {
  854. CString sql;
  855. sql.Format("INSERT INTO [rm_alarm]([ID],[mo],[mp],[no],[type],[occur_time],[level],[desc],[suggest],[val],[event_id],[rel_id],posi,loworhigh,referval,[sunroof]) "\
  856. "VALUES(%d, '%s', '%s', %d, %d, '%s', %d, '%s', '%s', %d, '%s', '%s',%d,%d,%d,%d);",
  857. pNewAlarmInfo->id, pNewAlarmInfo->mo.c_str(), pNewAlarmInfo->mp.c_str(), pNewAlarmInfo->no, pNewAlarmInfo->type,
  858. ctAlarmTime.Format("%Y-%m-%d %H:%M:%S"), pNewAlarmInfo->level, pNewAlarmInfo->desc.c_str(), pNewAlarmInfo->suggest.c_str(), pNewAlarmInfo->val,
  859. pNewAlarmInfo->event_id.c_str(), pNewAlarmInfo->rel_id.c_str(), uint8_t(pNewAlarmInfo->posi), uint8_t(pNewAlarmInfo->loworhigh), pNewAlarmInfo->refer_val,
  860. pNewAlarmInfo->sunroof);
  861. if (false == CDBConnectPool::Instance()->DBExecuteSQL(sql))
  862. CSimpleLog::Error("执行语句失败" + sql);
  863. else
  864. {
  865. uint8_t* pack = nullptr; int len = 0;
  866. auto packno = CResistAlarmMng::GeneralNewAlarmData(pNewAlarmInfo, &pack, &len);
  867. if (packno) CSuperManager::Instance()->SendPack(pack, len, packno, E_ZL_PROTOCAL::ZL_ALARM_DATA, true);
  868. //g_p315ClientManager->GetTcpClient()->SendAlarmData(pNewAlarmInfo->no, 1, ctAlarmTime.GetTime(), 0xFFFFFFFF, (WORD)pNewAlarmInfo->type, pNewAlarmInfo->val, 0, 0);
  869. delete[] pack;
  870. pack = nullptr;
  871. }
  872. }
  873. }
  874. }
  875. //判断转换阻力是否超限
  876. void CResistAlarmMng::JudgeAlarm(CResistAlarmMng* pService, CONVERT_RESIST_OVER_LIMIT* pConvertOverLimit,
  877. time_t show_time, int show_val, eDaoChaPosi posi, eUpOrDownInfo eUpOrDown, string mo, string mp, uint32_t zzjno)
  878. {
  879. if (pConvertOverLimit == nullptr || pConvertOverLimit->enable == false) return;
  880. bool bNew = false;
  881. ALARM_INFO* pAlarmInfo = nullptr;
  882. bool bAlarm = false, bWarn = false; int alarm_value = show_val, alarm_refer; time_t alarm_time = show_time;
  883. CTime ctAlarmTime = CTime(alarm_time / 1000);
  884. {
  885. if (eUpOrDown == eUpOrDownInfo::UOD_UP)//缩进
  886. {
  887. if (show_val > pConvertOverLimit->up_alarm_high_limit)
  888. {
  889. bAlarm = true;
  890. alarm_refer = pConvertOverLimit->up_alarm_high_limit;
  891. }
  892. else if (show_val > pConvertOverLimit->up_warn_high_limit)
  893. {
  894. bWarn = true;
  895. alarm_refer = pConvertOverLimit->up_warn_high_limit;
  896. }
  897. }
  898. else if (eUpOrDown == eUpOrDownInfo::UOD_DOWN)
  899. {
  900. if (show_val > pConvertOverLimit->dw_alarm_low_limit)
  901. {
  902. bAlarm = true;
  903. alarm_refer = pConvertOverLimit->dw_alarm_low_limit;
  904. }
  905. else if (show_val > pConvertOverLimit->dw_warn_low_limit)
  906. {
  907. bWarn = true;
  908. alarm_refer = pConvertOverLimit->dw_warn_low_limit;
  909. }
  910. }
  911. if (bAlarm == false && bWarn == false)
  912. {
  913. {
  914. string strPosi= "方向:";
  915. strPosi += (posi == eDaoChaPosi::DCP_FIX2INVERT ? "定扳反" : "反扳定");
  916. std::lock_guard<mutex> lock(pService->m_mtxAlarm);
  917. for (auto it = pService->m_lstUnConfirmAlarm.begin(); it != pService->m_lstUnConfirmAlarm.end();)
  918. {
  919. auto alarm = *it;
  920. if (alarm->no == 2 && alarm->type == eZL_ALARMTYPE::CONVERT_LIMIT //预警和报警单独计算 //
  921. && alarm->posi == posi && alarm->recoveryTime.wYear < 2000
  922. && alarm->mo.compare(mo) == 0 && alarm->mp.compare(mp) == 0) //跟上次报警时间超过1小时,则算新报警
  923. {
  924. pAlarmInfo = alarm;
  925. if (pAlarmInfo)
  926. {
  927. //恢复
  928. //1. 更新数据库
  929. //2. 更新内存数据
  930. //3. 上送恢复消息
  931. if (ctAlarmTime.GetTime() == 0) ctAlarmTime = CTime::GetCurrentTime();
  932. string recovery_time = ctAlarmTime.Format("%Y-%m-%d %H:%M:%S");
  933. CString sql;
  934. sql.Format("UPDATE [rm_alarm] SET [recovery_time]='%s' WHERE ID = %d;", recovery_time.c_str(), pAlarmInfo->id);
  935. CDBConnectPool::Instance()->DBExecuteSQL(sql);
  936. if (pAlarmInfo->ack_result == 1)
  937. {
  938. it = pService->m_lstUnConfirmAlarm.erase(it);
  939. continue;
  940. }
  941. else
  942. {
  943. ctAlarmTime.GetAsSystemTime(alarm->recoveryTime);
  944. }
  945. g_p315ClientManager->GetTcpClient()->SendAlarmData(alarm->zzjno, 2, CTime(pAlarmInfo->time).GetTime(), alarm_time / 1000,
  946. pAlarmInfo->type, posi, pAlarmInfo->level, eLowHigh::LH_HIGH,
  947. TIEDA_ACQ_VALUE(alarm->val, TIEDA_VAL_STATE::TVS_MOVEING), TIEDA_ACQ_VALUE(alarm->refer_val, TIEDA_VAL_STATE::TVS_MOVEING), 0);
  948. }
  949. }
  950. ++it;
  951. }
  952. }
  953. return;
  954. }
  955. //查找是否原有的报警已存在
  956. uint8_t level = 0;
  957. if (bAlarm) level = 1;
  958. {
  959. std::lock_guard<mutex> lock(pService->m_mtxAlarm);
  960. for (const auto& alarm : pService->m_lstUnConfirmAlarm)
  961. {
  962. if (alarm->no == 2 && alarm->type == eZL_ALARMTYPE::CONVERT_LIMIT && level == alarm->level && alarm->recoveryTime.wYear < 2000 //预警和报警单独计算
  963. && alarm->mo.compare(mo) == 0 && alarm->mp.compare(mp) == 0)
  964. {
  965. pAlarmInfo = alarm;
  966. break;
  967. }
  968. }
  969. }
  970. ALARM_INFO* pNewAlarmInfo = nullptr;
  971. {
  972. bNew = true;
  973. pNewAlarmInfo = new ALARM_INFO;
  974. pNewAlarmInfo->event_id = hjfunc_GetGUID();
  975. pNewAlarmInfo->rel_id = pAlarmInfo ? pAlarmInfo->event_id : "";
  976. pNewAlarmInfo->id = ++pService->m_nAlarmID;
  977. pNewAlarmInfo->level = (bAlarm == 1);
  978. pNewAlarmInfo->mo = mo;
  979. pNewAlarmInfo->mp = mp;
  980. pNewAlarmInfo->no = 2;
  981. ctAlarmTime.GetAsSystemTime(pNewAlarmInfo->time);
  982. pNewAlarmInfo->time.wMilliseconds = alarm_time % 1000;
  983. pNewAlarmInfo->type = eZL_ALARMTYPE::CONVERT_LIMIT;
  984. pNewAlarmInfo->val = alarm_value;
  985. pNewAlarmInfo->refer_val = alarm_refer;
  986. pNewAlarmInfo->posi = posi;
  987. pNewAlarmInfo->zzjno = zzjno;
  988. pNewAlarmInfo->sunroof = CSkylightMng::GetInstance()->IsSkylight(&pNewAlarmInfo->time);
  989. char szInfo[200] = { 0 };
  990. sprintf_s(szInfo, sizeof(szInfo), "转换阻力超限,方向:%s, 报警值为%dN, 参考值为%dN",
  991. posi == eDaoChaPosi::DCP_FIX2INVERT ? "定扳反" : "反扳定", alarm_value, alarm_refer);
  992. pNewAlarmInfo->desc = szInfo;
  993. lock_guard<mutex> lock(pService->m_mtxAlarm);
  994. pService->m_lstUnConfirmAlarm.push_back(pNewAlarmInfo);
  995. }
  996. //send
  997. //if (bNew) //不再推送
  998. {
  999. rapidjson::StringBuffer buffer;
  1000. auto ret = AlarmInfo2Pack(pNewAlarmInfo, buffer);
  1001. const char* output = buffer.GetString();
  1002. //CAppService::Instance()->GetLwsServer()->SendPackToALLClient_with_noEncode((uint8_t*)output, buffer.GetLength());
  1003. CAppService::Instance()->GetMgServer()->SendToAllClient(output, buffer.GetLength());
  1004. }
  1005. //主动推送315
  1006. //if (bNew)
  1007. {
  1008. g_p315ClientManager->GetTcpClient()->SendAlarmData(pNewAlarmInfo->zzjno, 1, CTime(pNewAlarmInfo->time).GetTime(), 0xFFFFFFFF,
  1009. pNewAlarmInfo->type, posi, pNewAlarmInfo->level, eLowHigh::LH_HIGH,
  1010. TIEDA_ACQ_VALUE(alarm_value, TIEDA_VAL_STATE::TVS_MOVEING), TIEDA_ACQ_VALUE(alarm_refer, TIEDA_VAL_STATE::TVS_MOVEING), 0);
  1011. }
  1012. //save
  1013. {
  1014. CString sql;
  1015. sql.Format("INSERT INTO [rm_alarm]([ID],[mo],[mp],[no],[type],[occur_time],[level],[desc],[suggest],[val],[event_id],[rel_id],posi,loworhigh,referval,[sunroof]) "\
  1016. "VALUES(%d, '%s', '%s', %d, %d, '%s', %d, '%s', '%s', %d, '%s', '%s',%d,%d,%d,%d);",
  1017. pNewAlarmInfo->id, pNewAlarmInfo->mo.c_str(), pNewAlarmInfo->mp.c_str(), pNewAlarmInfo->no, pNewAlarmInfo->type,
  1018. ctAlarmTime.Format("%Y-%m-%d %H:%M:%S"), pNewAlarmInfo->level, pNewAlarmInfo->desc.c_str(), pNewAlarmInfo->suggest.c_str(), pNewAlarmInfo->val,
  1019. pNewAlarmInfo->event_id.c_str(), pNewAlarmInfo->rel_id.c_str(), uint8_t(pNewAlarmInfo->posi), uint8_t(pNewAlarmInfo->loworhigh), pNewAlarmInfo->refer_val,
  1020. pNewAlarmInfo->sunroof);
  1021. if (false == CDBConnectPool::Instance()->DBExecuteSQL(sql))
  1022. CSimpleLog::Error("执行语句失败" + sql);
  1023. else
  1024. {
  1025. uint8_t* pack = nullptr; int len = 0;
  1026. auto packno = CResistAlarmMng::GeneralNewAlarmData(pNewAlarmInfo, &pack, &len);
  1027. if (packno) CSuperManager::Instance()->SendPack(pack, len, packno, E_ZL_PROTOCAL::ZL_ALARM_DATA, true);
  1028. //g_p315ClientManager->GetTcpClient()->SendAlarmData(pNewAlarmInfo->no, 1, ctAlarmTime.GetTime(), 0xFFFFFFFF, (WORD)pNewAlarmInfo->type, pNewAlarmInfo->val, 0, 0);
  1029. delete[] pack;
  1030. pack = nullptr;
  1031. }
  1032. }
  1033. }
  1034. }
  1035. void CResistAlarmMng::JudgeAlarm(CResistAlarmMng* pService, RETENSION_FORCE_DROP* pRetensionForceDropInfo, time_t show_time, int show_val, eSuoBiPosi posi, string mo, string mp, uint32_t zzjno)
  1036. {
  1037. if (pRetensionForceDropInfo == nullptr || pRetensionForceDropInfo->enable == false) return;
  1038. bool bNew = false;
  1039. ALARM_INFO* pAlarmInfo = nullptr;
  1040. bool bAlarm = false, bWarn = false; int alarm_value = 0, alarm_refer; time_t alarm_time = show_time;
  1041. CTime ctAlarmTime = CTime(alarm_time / 1000);
  1042. eLowHigh loworhigh = eLowHigh::LH_HIGH; //0:最小值 1:最大值
  1043. {
  1044. if (show_val < pRetensionForceDropInfo->dw_alarm_low_drop)
  1045. {
  1046. bWarn = true;
  1047. alarm_refer = pRetensionForceDropInfo->dw_alarm_low_drop;
  1048. alarm_value = show_val;
  1049. //loworhigh = eLowHigh::LH_LOW;
  1050. }
  1051. if (bAlarm == false && bWarn == false)
  1052. {
  1053. if (g_strMoMp.compare(mo + "." + mp) == 0)
  1054. SPDLOG_INFO("[保持力下降]{}.{} 未检测到下降. {} ", mo, mp, show_val);
  1055. {
  1056. std::lock_guard<mutex> lock(pService->m_mtxAlarm);
  1057. for (auto it = pService->m_lstUnConfirmAlarm.begin(); it != pService->m_lstUnConfirmAlarm.end();)
  1058. {
  1059. auto pAlarm = *it;
  1060. if (pAlarm->no == (uint8_t)posi && pAlarm->type == eZL_ALARMTYPE::RETENSION_FORCE && pAlarm->recoveryTime.wYear < 2000 //预警和报警统一恢复
  1061. && pAlarm->mo.compare(mo) == 0 && pAlarm->mp.compare(mp) == 0)
  1062. {
  1063. pAlarmInfo = pAlarm;
  1064. if (pAlarmInfo)
  1065. {
  1066. //恢复
  1067. //1. 更新数据库
  1068. //2. 更新内存数据
  1069. //3. 上送恢复消息
  1070. if (ctAlarmTime.GetTime() == 0) ctAlarmTime = CTime::GetCurrentTime();
  1071. string recovery_time = ctAlarmTime.Format("%Y-%m-%d %H:%M:%S");
  1072. CString sql;
  1073. sql.Format("UPDATE [rm_alarm] SET [recovery_time]='%s' WHERE ID = %d;", recovery_time.c_str(), pAlarmInfo->id);
  1074. CDBConnectPool::Instance()->DBExecuteSQL(sql);
  1075. if (pAlarmInfo->ack_result == 1)
  1076. {
  1077. it = pService->m_lstUnConfirmAlarm.erase(it);
  1078. continue;
  1079. }
  1080. else
  1081. {
  1082. ctAlarmTime.GetAsSystemTime(pAlarm->recoveryTime);
  1083. }
  1084. g_p315ClientManager->GetTcpClient()->SendAlarmData(zzjno, 2, CTime(pAlarmInfo->time).GetTime(), ctAlarmTime.GetTime(),
  1085. pAlarmInfo->type, posi == eSuoBiPosi::SB_FIX ? eDaoChaPosi::DCP_FIX : eDaoChaPosi::DCP_INVERT, pAlarmInfo->level, pAlarm->loworhigh,
  1086. TIEDA_ACQ_VALUE(pAlarm->val, TIEDA_VAL_STATE::TVS_AFTER_MOVE), TIEDA_ACQ_VALUE(pAlarm->refer_val, TIEDA_VAL_STATE::TVS_AFTER_MOVE), 0);
  1087. }
  1088. }
  1089. it++;
  1090. }
  1091. }
  1092. return;
  1093. }
  1094. SPDLOG_INFO("[保持力下降]{}.{} 检测到下降. {} ", mo, mp, show_val);
  1095. //查找是否原有的报警已存在
  1096. uint8_t level = 0;
  1097. if (bAlarm) level = 1;
  1098. {
  1099. std::lock_guard<mutex> lock(pService->m_mtxAlarm);
  1100. for (const auto& alarm : pService->m_lstUnConfirmAlarm)
  1101. {
  1102. if (alarm->no == (uint8_t)posi && alarm->type == eZL_ALARMTYPE::RETENSION_FORCE && level == alarm->level && alarm->recoveryTime.wYear < 2000 //预警和报警单独计算
  1103. && alarm->mo.compare(mo) == 0 && alarm->mp.compare(mp) == 0)
  1104. {
  1105. pAlarmInfo = alarm;
  1106. break;
  1107. }
  1108. }
  1109. }
  1110. if (pAlarmInfo == nullptr)
  1111. {
  1112. bNew = true;
  1113. pAlarmInfo = new ALARM_INFO;
  1114. pAlarmInfo->event_id = hjfunc_GetGUID();
  1115. pAlarmInfo->id = ++pService->m_nAlarmID;
  1116. pAlarmInfo->level = (bAlarm == 1);
  1117. pAlarmInfo->mo = mo;
  1118. pAlarmInfo->mp = mp;
  1119. pAlarmInfo->no = (uint8_t)posi;
  1120. ctAlarmTime.GetAsSystemTime(pAlarmInfo->time);
  1121. pAlarmInfo->time.wMilliseconds = alarm_time % 1000;
  1122. pAlarmInfo->type = eZL_ALARMTYPE::RETENSION_FORCE;
  1123. pAlarmInfo->val = alarm_value;
  1124. pAlarmInfo->refer_val = alarm_refer;
  1125. pAlarmInfo->posi = (posi == eSuoBiPosi::SB_FIX ? eDaoChaPosi::DCP_FIX : eDaoChaPosi::DCP_INVERT);
  1126. pAlarmInfo->loworhigh = loworhigh;
  1127. pAlarmInfo->zzjno = zzjno;
  1128. pAlarmInfo->sunroof = CSkylightMng::GetInstance()->IsSkylight(&pAlarmInfo->time);
  1129. char szInfo[200] = { 0 };
  1130. sprintf_s(szInfo, sizeof(szInfo), "保持力下降,位置:%s, 报警值为%dN, 参考值为%dN",
  1131. posi == eSuoBiPosi::SB_FIX ? "定位" : "反位", alarm_value, alarm_refer);
  1132. pAlarmInfo->desc = szInfo;
  1133. lock_guard<mutex> lock(pService->m_mtxAlarm);
  1134. pService->m_lstUnConfirmAlarm.push_back(pAlarmInfo);
  1135. }
  1136. //send
  1137. if (bNew) //不再推送
  1138. {
  1139. rapidjson::StringBuffer buffer;
  1140. auto ret = AlarmInfo2Pack(pAlarmInfo, buffer);
  1141. const char* output = buffer.GetString();
  1142. //CAppService::Instance()->GetLwsServer()->SendPackToALLClient_with_noEncode((uint8_t*)output, buffer.GetLength());
  1143. CAppService::Instance()->GetMgServer()->SendToAllClient(output, buffer.GetLength());
  1144. }
  1145. //主动推送315
  1146. if (bNew)
  1147. {
  1148. g_p315ClientManager->GetTcpClient()->SendAlarmData(zzjno, 1, CTime(pAlarmInfo->time).GetTime(), 0xFFFFFFFF,
  1149. pAlarmInfo->type, posi == eSuoBiPosi::SB_FIX ? eDaoChaPosi::DCP_FIX : eDaoChaPosi::DCP_INVERT, pAlarmInfo->level, loworhigh,
  1150. TIEDA_ACQ_VALUE(alarm_value, TIEDA_VAL_STATE::TVS_AFTER_MOVE), TIEDA_ACQ_VALUE(alarm_refer, TIEDA_VAL_STATE::TVS_AFTER_MOVE), 0);
  1151. }
  1152. //save
  1153. if (bNew)
  1154. {
  1155. CString sql;
  1156. sql.Format("INSERT INTO [rm_alarm]([ID],[mo],[mp],[no],[type],[occur_time],[level],[desc],[suggest],[val],[event_id],[rel_id],posi,loworhigh,referval,[sunroof]) "\
  1157. "VALUES(%d, '%s', '%s', %d, %d, '%s', %d, '%s', '%s', %d, '%s', '%s',%d,%d,%d,%d);",
  1158. pAlarmInfo->id, pAlarmInfo->mo.c_str(), pAlarmInfo->mp.c_str(), pAlarmInfo->no, pAlarmInfo->type,
  1159. ctAlarmTime.Format("%Y-%m-%d %H:%M:%S"), pAlarmInfo->level, pAlarmInfo->desc.c_str(), pAlarmInfo->suggest.c_str(), pAlarmInfo->val,
  1160. pAlarmInfo->event_id.c_str(), pAlarmInfo->rel_id.c_str(), uint8_t(pAlarmInfo->posi), uint8_t(pAlarmInfo->loworhigh), pAlarmInfo->refer_val,
  1161. pAlarmInfo->sunroof);
  1162. if (false == CDBConnectPool::Instance()->DBExecuteSQL(sql))
  1163. CSimpleLog::Error("执行语句失败" + sql);
  1164. else
  1165. {
  1166. uint8_t* pack = nullptr; int len = 0;
  1167. auto packno = CResistAlarmMng::GeneralNewAlarmData(pAlarmInfo, &pack, &len);
  1168. if (packno) CSuperManager::Instance()->SendPack(pack, len, packno, E_ZL_PROTOCAL::ZL_ALARM_DATA, true);
  1169. delete[] pack;
  1170. pack = nullptr;
  1171. }
  1172. }
  1173. }
  1174. }
  1175. void CResistAlarmMng::JudgeAlarm(CResistAlarmMng* pService, RETENSION_FORCE_DROP* pConstRetensionForceWaveInfo,
  1176. time_t show_time, int show_val, eSuoBiPosi posi, string mo, string mp, uint32_t zzjno, void* pInfo)
  1177. {
  1178. ASSERT(pConstRetensionForceWaveInfo);
  1179. ST_MOMP_INFO* pMoMpInfo = (ST_MOMP_INFO*)pInfo;
  1180. CTime ctShowTime(show_time / 1000);
  1181. bool bSkylight = CSkylightMng::GetInstance()->IsSkylight(&ctShowTime);
  1182. int nOldVal = 0;
  1183. if (bSkylight && show_val == 0)
  1184. {
  1185. if (posi == eSuoBiPosi::SB_FIX)
  1186. {
  1187. pMoMpInfo->fix_const_retension_force = INT_MIN;
  1188. }
  1189. else
  1190. {
  1191. pMoMpInfo->invert_const_retension_force = INT_MIN;
  1192. }
  1193. }
  1194. else
  1195. {
  1196. if (posi == eSuoBiPosi::SB_FIX)
  1197. {
  1198. nOldVal = pMoMpInfo->fix_const_retension_force;
  1199. if (nOldVal == INT_MIN)
  1200. {
  1201. pMoMpInfo->fix_const_retension_force = show_val;
  1202. return;
  1203. }
  1204. }
  1205. else
  1206. {
  1207. nOldVal = pMoMpInfo->invert_const_retension_force;
  1208. if (nOldVal == INT_MIN)
  1209. {
  1210. pMoMpInfo->invert_const_retension_force = show_val;
  1211. return;
  1212. }
  1213. }
  1214. }
  1215. ASSERT(pConstRetensionForceWaveInfo);
  1216. if (pConstRetensionForceWaveInfo->enable == false) return;
  1217. if (nOldVal != INT_MIN)
  1218. {
  1219. int nRatio = show_val * 100 / nOldVal;
  1220. eLowHigh loworhigh = eLowHigh::LH_UNKNOWN;
  1221. int alarm_refer = 0;
  1222. if (nOldVal != 0)
  1223. {
  1224. //下降20%或上升40% 进行预警
  1225. if (pConstRetensionForceWaveInfo->dw_alarm_low_drop != INT_MIN && nRatio <= (100 - pConstRetensionForceWaveInfo->dw_alarm_low_drop))
  1226. {
  1227. loworhigh = eLowHigh::LH_LOW;
  1228. alarm_refer = nOldVal * (100 - pConstRetensionForceWaveInfo->dw_alarm_low_drop) / 100;
  1229. }
  1230. else if (nRatio >= (100 + pConstRetensionForceWaveInfo->alarm_high_percent))
  1231. {
  1232. loworhigh = eLowHigh::LH_HIGH;
  1233. alarm_refer = nOldVal * (100 + pConstRetensionForceWaveInfo->alarm_high_percent) / 100;
  1234. }
  1235. }
  1236. ALARM_INFO* pAlarmInfo = nullptr;
  1237. CTime ctAlarmTime(show_time / 1000);
  1238. bool bNew = false;
  1239. //产生预警
  1240. if (loworhigh != eLowHigh::LH_UNKNOWN)
  1241. {
  1242. //查找是否原有的报警已存在
  1243. int level = 0;// 预警
  1244. {
  1245. std::lock_guard<mutex> lock(pService->m_mtxAlarm);
  1246. for (const auto& alarm : pService->m_lstUnConfirmAlarm)
  1247. {
  1248. if (alarm->no == (uint8_t)posi && alarm->type == eZL_ALARMTYPE::RETENSION_FORCE
  1249. && level == alarm->level && alarm->recoveryTime.wYear < 2000 //预警和报警单独计算
  1250. && alarm->mo.compare(mo) == 0 && alarm->mp.compare(mp) == 0)
  1251. {
  1252. pAlarmInfo = alarm;
  1253. break;
  1254. }
  1255. }
  1256. }
  1257. if (pAlarmInfo == nullptr)
  1258. {
  1259. bNew = true;
  1260. pAlarmInfo = new ALARM_INFO;
  1261. pAlarmInfo->event_id = hjfunc_GetGUID();
  1262. pAlarmInfo->id = ++pService->m_nAlarmID;
  1263. pAlarmInfo->level = level;// 预警
  1264. pAlarmInfo->mo = mo;
  1265. pAlarmInfo->mp = mp;
  1266. pAlarmInfo->no = (uint8_t)posi;
  1267. ctAlarmTime.GetAsSystemTime(pAlarmInfo->time);
  1268. pAlarmInfo->time.wMilliseconds = show_time % 1000;
  1269. pAlarmInfo->type = eZL_ALARMTYPE::RETENSION_FORCE;
  1270. pAlarmInfo->val = show_val;
  1271. pAlarmInfo->refer_val = alarm_refer;
  1272. pAlarmInfo->posi = (posi == eSuoBiPosi::SB_FIX ? eDaoChaPosi::DCP_FIX : eDaoChaPosi::DCP_INVERT);
  1273. pAlarmInfo->loworhigh = loworhigh;
  1274. pAlarmInfo->zzjno = zzjno;
  1275. pAlarmInfo->sunroof = CSkylightMng::GetInstance()->IsSkylight(&pAlarmInfo->time);
  1276. char szInfo[200] = { 0 };
  1277. sprintf_s(szInfo, sizeof(szInfo), "保持力异常报警,位置:%s, 报警值为%dN, 上一次值:%dN, 参考值为%dN",
  1278. posi == eSuoBiPosi::SB_FIX ? "定位" : "反位", show_val, nOldVal, alarm_refer);
  1279. pAlarmInfo->desc = szInfo;
  1280. lock_guard<mutex> lock(pService->m_mtxAlarm);
  1281. pService->m_lstUnConfirmAlarm.push_back(pAlarmInfo);
  1282. }
  1283. //主动推送315
  1284. if (bNew)
  1285. {
  1286. g_p315ClientManager->GetTcpClient()->SendAlarmData(zzjno, 1, CTime(pAlarmInfo->time).GetTime(), 0xFFFFFFFF,
  1287. pAlarmInfo->type, posi == eSuoBiPosi::SB_FIX ? eDaoChaPosi::DCP_FIX : eDaoChaPosi::DCP_INVERT, pAlarmInfo->level, loworhigh,
  1288. TIEDA_ACQ_VALUE(pAlarmInfo->val, TIEDA_VAL_STATE::TVS_AFTER_MOVE), TIEDA_ACQ_VALUE(pAlarmInfo->refer_val, TIEDA_VAL_STATE::TVS_AFTER_MOVE), 0);
  1289. }
  1290. if (bNew)
  1291. {
  1292. CString sql;
  1293. sql.Format("INSERT INTO [rm_alarm]([ID],[mo],[mp],[no],[type],[occur_time],[level],[desc],[suggest],[val],[event_id],[rel_id],posi,loworhigh,referval,[sunroof]) "\
  1294. "VALUES(%d, '%s', '%s', %d, %d, '%s', %d, '%s', '%s', %d, '%s', '%s',%d,%d,%d,%d);",
  1295. pAlarmInfo->id, pAlarmInfo->mo.c_str(), pAlarmInfo->mp.c_str(), pAlarmInfo->no, pAlarmInfo->type,
  1296. ctAlarmTime.Format("%Y-%m-%d %H:%M:%S"), pAlarmInfo->level, pAlarmInfo->desc.c_str(), pAlarmInfo->suggest.c_str(), pAlarmInfo->val,
  1297. pAlarmInfo->event_id.c_str(), pAlarmInfo->rel_id.c_str(), uint8_t(pAlarmInfo->posi), uint8_t(pAlarmInfo->loworhigh), pAlarmInfo->refer_val,
  1298. pAlarmInfo->sunroof);
  1299. if (false == CDBConnectPool::Instance()->DBExecuteSQL(sql))
  1300. CSimpleLog::Error("执行语句失败" + sql);
  1301. else
  1302. {
  1303. uint8_t* pack = nullptr; int len = 0;
  1304. auto packno = CResistAlarmMng::GeneralNewAlarmData(pAlarmInfo, &pack, &len);
  1305. if (packno) CSuperManager::Instance()->SendPack(pack, len, packno, E_ZL_PROTOCAL::ZL_ALARM_DATA, true);
  1306. delete[] pack;
  1307. pack = nullptr;
  1308. }
  1309. }
  1310. }
  1311. else
  1312. {
  1313. //恢复
  1314. {
  1315. std::lock_guard<mutex> lock(pService->m_mtxAlarm);
  1316. for (auto it = pService->m_lstUnConfirmAlarm.begin(); it != pService->m_lstUnConfirmAlarm.end();)
  1317. {
  1318. auto pAlarm = *it;
  1319. if (pAlarm->no == (uint8_t)posi && pAlarm->type == eZL_ALARMTYPE::RETENSION_FORCE && pAlarm->recoveryTime.wYear < 2000 //预警和报警单独计算
  1320. && pAlarm->mo.compare(mo) == 0 && pAlarm->mp.compare(mp) == 0)
  1321. {
  1322. pAlarmInfo = pAlarm;
  1323. if (pAlarmInfo)
  1324. {
  1325. //恢复
  1326. //1. 更新数据库
  1327. //2. 更新内存数据
  1328. //3. 上送恢复消息
  1329. if (ctAlarmTime.GetTime() == 0) ctAlarmTime = CTime::GetCurrentTime();
  1330. string recovery_time = ctAlarmTime.Format("%Y-%m-%d %H:%M:%S");
  1331. CString sql;
  1332. sql.Format("UPDATE [rm_alarm] SET [recovery_time]='%s' WHERE ID = %d;", recovery_time.c_str(), pAlarmInfo->id);
  1333. CDBConnectPool::Instance()->DBExecuteSQL(sql);
  1334. if (pAlarmInfo->ack_result == 1)
  1335. {
  1336. it = pService->m_lstUnConfirmAlarm.erase(it);
  1337. continue;
  1338. }
  1339. else
  1340. {
  1341. ctAlarmTime.GetAsSystemTime(pAlarm->recoveryTime);
  1342. }
  1343. g_p315ClientManager->GetTcpClient()->SendAlarmData(zzjno, 2, CTime(pAlarmInfo->time).GetTime(), ctAlarmTime.GetTime(),
  1344. pAlarmInfo->type, posi == eSuoBiPosi::SB_FIX ? eDaoChaPosi::DCP_FIX : eDaoChaPosi::DCP_INVERT, pAlarmInfo->level, pAlarm->loworhigh,
  1345. TIEDA_ACQ_VALUE(pAlarm->val, TIEDA_VAL_STATE::TVS_AFTER_MOVE), TIEDA_ACQ_VALUE(pAlarm->refer_val, TIEDA_VAL_STATE::TVS_AFTER_MOVE), 0);
  1346. }
  1347. }
  1348. it++;
  1349. }
  1350. }
  1351. return;
  1352. }
  1353. }
  1354. }
  1355. void CResistAlarmMng::JudgeAlarm1(CResistAlarmMng* pService, RETENSION_FORCE_DROP* pRetensionForceDropInfo, time_t show_time, int show_val,
  1356. eSuoBiPosi posi, string mo, string mp, uint32_t zzjno)
  1357. {
  1358. if (pRetensionForceDropInfo == nullptr || pRetensionForceDropInfo->enable == false) return;
  1359. bool bNew = false;
  1360. ALARM_INFO* pAlarmInfo = nullptr;
  1361. bool bAlarm = false, bWarn = false; int alarm_value = 0, alarm_refer; time_t alarm_time = show_time;
  1362. CTime ctAlarmTime = CTime(alarm_time / 1000);
  1363. eLowHigh loworhigh = eLowHigh::LH_HIGH; //0:最小值 1:最大值
  1364. {
  1365. if (show_val < pRetensionForceDropInfo->dw_alarm_low_drop)
  1366. {
  1367. bAlarm = true;
  1368. alarm_refer = pRetensionForceDropInfo->dw_alarm_low_drop;
  1369. alarm_value = show_val;
  1370. //loworhigh = eLowHigh::LH_LOW;
  1371. }
  1372. if (bAlarm == false && bWarn == false)
  1373. {
  1374. {
  1375. std::lock_guard<mutex> lock(pService->m_mtxAlarm);
  1376. for (auto it = pService->m_lstUnConfirmAlarm.begin(); it != pService->m_lstUnConfirmAlarm.end();)
  1377. {
  1378. auto pAlarm = *it;
  1379. if (pAlarm->no == (uint8_t)posi && pAlarm->type == eZL_ALARMTYPE::RETENSION_FORCE && pAlarm->recoveryTime.wYear < 2000 //预警和报警统一恢复
  1380. && pAlarm->mo.compare(mo) == 0 && pAlarm->mp.compare(mp) == 0)
  1381. {
  1382. pAlarmInfo = pAlarm;
  1383. if (pAlarmInfo)
  1384. {
  1385. //恢复
  1386. //1. 更新数据库
  1387. //2. 更新内存数据
  1388. //3. 上送恢复消息
  1389. if (ctAlarmTime.GetTime() == 0) ctAlarmTime = CTime::GetCurrentTime();
  1390. string recovery_time = ctAlarmTime.Format("%Y-%m-%d %H:%M:%S");
  1391. CString sql;
  1392. sql.Format("UPDATE [rm_alarm] SET [recovery_time]='%s' WHERE ID = %d;", recovery_time.c_str(), pAlarmInfo->id);
  1393. CDBConnectPool::Instance()->DBExecuteSQL(sql);
  1394. if (pAlarmInfo->ack_result == 1)
  1395. {
  1396. it = pService->m_lstUnConfirmAlarm.erase(it);
  1397. continue;
  1398. }
  1399. else
  1400. {
  1401. ctAlarmTime.GetAsSystemTime(pAlarm->recoveryTime);
  1402. }
  1403. g_p315ClientManager->GetTcpClient()->SendAlarmData(zzjno, 2, CTime(pAlarmInfo->time).GetTime(), alarm_time / 1000,
  1404. pAlarmInfo->type, posi == eSuoBiPosi::SB_FIX ? eDaoChaPosi::DCP_FIX : eDaoChaPosi::DCP_INVERT, pAlarmInfo->level, pAlarm->loworhigh,
  1405. TIEDA_ACQ_VALUE(pAlarm->val, TIEDA_VAL_STATE::TVS_AFTER_MOVE), TIEDA_ACQ_VALUE(pAlarm->refer_val, TIEDA_VAL_STATE::TVS_AFTER_MOVE), 0);
  1406. }
  1407. }
  1408. it++;
  1409. }
  1410. }
  1411. return;
  1412. }
  1413. //查找是否原有的报警已存在
  1414. uint8_t level = 0;
  1415. if (bAlarm) level = 1;
  1416. {
  1417. std::lock_guard<mutex> lock(pService->m_mtxAlarm);
  1418. for (const auto& alarm : pService->m_lstUnConfirmAlarm)
  1419. {
  1420. if (alarm->no == (uint8_t)posi && alarm->type == eZL_ALARMTYPE::RETENSION_FORCE && level == alarm->level && alarm->recoveryTime.wYear < 2000 //预警和报警单独计算
  1421. && alarm->mo.compare(mo) == 0 && alarm->mp.compare(mp) == 0)
  1422. {
  1423. pAlarmInfo = alarm;
  1424. break;
  1425. }
  1426. }
  1427. }
  1428. ALARM_INFO* pNewAlarmInfo = nullptr;
  1429. {
  1430. bNew = true;
  1431. pNewAlarmInfo = new ALARM_INFO;
  1432. pNewAlarmInfo->event_id = hjfunc_GetGUID();
  1433. pNewAlarmInfo->rel_id = pAlarmInfo ? pAlarmInfo->event_id : "";
  1434. pNewAlarmInfo->id = ++pService->m_nAlarmID;
  1435. pNewAlarmInfo->level = (bAlarm == 1);
  1436. pNewAlarmInfo->mo = mo;
  1437. pNewAlarmInfo->mp = mp;
  1438. pNewAlarmInfo->no = (uint8_t)posi;
  1439. ctAlarmTime.GetAsSystemTime(pNewAlarmInfo->time);
  1440. pNewAlarmInfo->time.wMilliseconds = alarm_time % 1000;
  1441. pNewAlarmInfo->type = eZL_ALARMTYPE::RETENSION_FORCE;
  1442. pNewAlarmInfo->val = alarm_value;
  1443. pNewAlarmInfo->refer_val = alarm_refer;
  1444. pNewAlarmInfo->posi = (posi == eSuoBiPosi::SB_FIX ? eDaoChaPosi::DCP_FIX : eDaoChaPosi::DCP_INVERT);
  1445. pNewAlarmInfo->loworhigh = loworhigh;
  1446. pNewAlarmInfo->zzjno = zzjno;
  1447. pNewAlarmInfo->sunroof = CSkylightMng::GetInstance()->IsSkylight(&pNewAlarmInfo->time);
  1448. char szInfo[200] = { 0 };
  1449. sprintf_s(szInfo, sizeof(szInfo), "保持力下降,位置:%s, 报警值为%dN, 参考值为%dN",
  1450. posi == eSuoBiPosi::SB_FIX ? "定位" : "反位", alarm_value, alarm_refer);
  1451. pNewAlarmInfo->desc = szInfo;
  1452. lock_guard<mutex> lock(pService->m_mtxAlarm);
  1453. pService->m_lstUnConfirmAlarm.push_back(pNewAlarmInfo);
  1454. }
  1455. //send
  1456. //if (bNew) //不再推送
  1457. {
  1458. rapidjson::StringBuffer buffer;
  1459. auto ret = AlarmInfo2Pack(pNewAlarmInfo, buffer);
  1460. const char* output = buffer.GetString();
  1461. //CAppService::Instance()->GetLwsServer()->SendPackToALLClient_with_noEncode((uint8_t*)output, buffer.GetLength());
  1462. CAppService::Instance()->GetMgServer()->SendToAllClient(output, buffer.GetLength());
  1463. }
  1464. //主动推送315
  1465. //if (bNew)
  1466. {
  1467. g_p315ClientManager->GetTcpClient()->SendAlarmData(zzjno, 1, CTime(pNewAlarmInfo->time).GetTime(), 0xFFFFFFFF,
  1468. pNewAlarmInfo->type, posi == eSuoBiPosi::SB_FIX ? eDaoChaPosi::DCP_FIX : eDaoChaPosi::DCP_INVERT, pNewAlarmInfo->level, loworhigh,
  1469. TIEDA_ACQ_VALUE(alarm_value, TIEDA_VAL_STATE::TVS_AFTER_MOVE), TIEDA_ACQ_VALUE(alarm_refer, TIEDA_VAL_STATE::TVS_AFTER_MOVE), 0);
  1470. }
  1471. //save
  1472. //if (bNew)
  1473. {
  1474. CString sql;
  1475. sql.Format("INSERT INTO [rm_alarm]([ID],[mo],[mp],[no],[type],[occur_time],[level],[desc],[suggest],[val],[event_id],[rel_id],posi,loworhigh,referval,[sunroof]) "\
  1476. "VALUES(%d, '%s', '%s', %d, %d, '%s', %d, '%s', '%s', %d, '%s', '%s',%d,%d,%d,%d);",
  1477. pNewAlarmInfo->id, pNewAlarmInfo->mo.c_str(), pNewAlarmInfo->mp.c_str(), pNewAlarmInfo->no, pNewAlarmInfo->type,
  1478. ctAlarmTime.Format("%Y-%m-%d %H:%M:%S"), pNewAlarmInfo->level, pNewAlarmInfo->desc.c_str(), pNewAlarmInfo->suggest.c_str(), pNewAlarmInfo->val,
  1479. pNewAlarmInfo->event_id.c_str(), pNewAlarmInfo->rel_id.c_str(), uint8_t(pNewAlarmInfo->posi), uint8_t(pNewAlarmInfo->loworhigh), pNewAlarmInfo->refer_val,
  1480. pNewAlarmInfo->sunroof);
  1481. if (false == CDBConnectPool::Instance()->DBExecuteSQL(sql))
  1482. CSimpleLog::Error("执行语句失败" + sql);
  1483. else
  1484. {
  1485. uint8_t* pack = nullptr; int len = 0;
  1486. auto packno = CResistAlarmMng::GeneralNewAlarmData(pNewAlarmInfo, &pack, &len);
  1487. if (packno) CSuperManager::Instance()->SendPack(pack, len, packno, E_ZL_PROTOCAL::ZL_ALARM_DATA, true);
  1488. //g_p315ClientManager->GetTcpClient()->SendAlarmData(pNewAlarmInfo->no, 1, ctAlarmTime.GetTime(), 0xFFFFFFFF, (WORD)pNewAlarmInfo->type, pNewAlarmInfo->val, 0, 0);
  1489. delete[] pack;
  1490. pack = nullptr;
  1491. }
  1492. }
  1493. }
  1494. }
  1495. BOOL ConmpareValue(std::map<time_t, int>::const_iterator i, std::map<time_t, int>::const_iterator j, int refer_value, int& hight_num, int& low_num)
  1496. {
  1497. int start_value = 0;
  1498. for (; i != j; i++)
  1499. {
  1500. if (i->second > refer_value)
  1501. {
  1502. if (start_value == 1)
  1503. {
  1504. //原来大于,现在也是大于
  1505. continue;
  1506. }
  1507. else
  1508. {
  1509. //原来小于或者无效,现在大于
  1510. start_value = 1;
  1511. hight_num++;
  1512. }
  1513. }
  1514. else
  1515. {
  1516. if (start_value == -1)
  1517. {
  1518. //原来小于,现在小于等于
  1519. continue;
  1520. }
  1521. else
  1522. {
  1523. //原来大于或者无效,现在小于
  1524. start_value = -1;
  1525. low_num++;
  1526. }
  1527. }
  1528. }
  1529. return TRUE;
  1530. }
  1531. void CResistAlarmMng::ThreadProcAlarmSet(DWORD_PTR param)
  1532. {
  1533. auto pService = (CResistAlarmMng*)param;
  1534. Sleep(2000);
  1535. time_t tmNow;
  1536. bool bLock = false;
  1537. do
  1538. {
  1539. if (bLock)
  1540. {
  1541. g_lockSync.Unlock();
  1542. bLock = false;
  1543. }
  1544. Sleep(800);
  1545. time(&tmNow);
  1546. if (tmNow == pService->m_lastDetectTime)
  1547. continue;
  1548. pService->m_lastDetectTime = tmNow;
  1549. bLock = true;
  1550. g_lockSync.ReadLock();
  1551. //1秒判断一次
  1552. for (const auto& item : pService->m_alarm_set)
  1553. {
  1554. if (item.second == nullptr) continue;
  1555. if (item.second->enable == false) continue;
  1556. if (item.second->type != eZL_ALARMTYPE::MAX_OVER_LIMIT && item.second->type != eZL_ALARMTYPE::FRICTION_OVER_LIMIT) //这个逻辑里面只支持这两种
  1557. continue;
  1558. if (item.second->no < 0 || item.second->no > 2) continue;
  1559. int nPos = item.first.find('.');
  1560. if (nPos == -1) continue;
  1561. int nPos2 = item.first.find('.', nPos + 1);
  1562. if (nPos2 == -1) continue;
  1563. string mo = item.first.substr(0, nPos);
  1564. string mp = item.first.substr(nPos + 1, nPos2 - nPos - 1);
  1565. string momp = item.first.substr(0, nPos2);
  1566. bool bNew = false;
  1567. ALARM_INFO *pAlarmInfo = nullptr;
  1568. CTime ctAlarmTime;
  1569. string imei; int idx;
  1570. auto ret = CMonitorObjectMng::Instance()->MOMP2IMEI(momp, imei, idx);
  1571. if (ret == false) continue;
  1572. #ifdef _DEBUG
  1573. if (strcmp("860588048955283", imei.c_str()) != 0) continue;
  1574. #endif
  1575. auto pDevice = CDeviceMng::Instance()->Find(imei);
  1576. if (pDevice == nullptr)
  1577. {
  1578. CSimpleLog::Error(("找不到imei(" + imei + ")" + std::to_string(__LINE__)).c_str());
  1579. continue;
  1580. }
  1581. int desc_type = 0;
  1582. bool bAlarm = false, bWarn = false; int alarm_value = 0; time_t alarm_time = 0;
  1583. eDaoChaPosi posi = eDaoChaPosi::DCP_UNKNOWN;
  1584. int high_alarm_limit = INT_MAX, high_warn_limit = INT_MAX, low_alarm_limit = INT_MIN, low_warn_limit = INT_MIN, alarm_refer = INT_MAX;
  1585. time_t tmNow; time(&tmNow);
  1586. BASE_INFO* pAlarmSet = item.second;
  1587. do{
  1588. Sleep(0);
  1589. lock_guard<mutex> lock(pDevice->m_mtx);
  1590. std::map<time_t, int>* pData = pDevice->GetMapData(idx, item.second->no);
  1591. assert(pData);
  1592. if (pData == nullptr) continue;
  1593. if (pData->size() == 0) continue;
  1594. if (item.second == 0) continue;
  1595. if (item.second->type == eZL_ALARMTYPE::MAX_OVER_LIMIT)
  1596. {
  1597. auto pInfo = (MAX_OVER_LIMIT_INFO*)pAlarmSet;
  1598. #ifdef _DEBUG
  1599. if (pInfo->tmLastCheckTime > pData->rbegin()->first)
  1600. pInfo->tmLastCheckTime = pData->rbegin()->first;
  1601. #endif
  1602. auto it = pData->cbegin();
  1603. if (pInfo->tmLastCheckTime == 0) pInfo->tmLastCheckTime = pData->rbegin()->first;
  1604. else if (pInfo->tmLastCheckTime != 0) it = pData->find(pInfo->tmLastCheckTime);
  1605. if (it == pData->cend())
  1606. {
  1607. it = pData->find(pInfo->tmLastCheckTime / 1000 * 1000);
  1608. if (it == pData->cend())
  1609. {
  1610. pInfo->tmLastCheckTime = pData->rbegin()->first;
  1611. continue;
  1612. }
  1613. }
  1614. if (pInfo->tmLastCheckTime == pData->rbegin()->first) //没有新的数据
  1615. continue;
  1616. #ifdef _DEBUG
  1617. TRACE("%s.%s.%d.%d lastcheck:%s now:%s\r\n", mo.c_str(), mp.c_str(), pInfo->no, (int)eZL_ALARMTYPE::MAX_OVER_LIMIT,
  1618. CTime(pInfo->tmLastCheckTime / 1000).Format("%Y-%m-%d %H:%M:%S"), CTime(pData->rbegin()->first / 1000).Format("%Y-%m-%d %H:%M:%S"));
  1619. #endif // _DEBUG
  1620. time_t tmStartSteady = pInfo->tmLastCheckTime / 1000 - 3; //稳态报警检测开始, 从上次3秒前到当前秒
  1621. if (item.second->no == 2)
  1622. {
  1623. if (pInfo->alarm_high_limit > pInfo->f_alarm_high_limit)
  1624. {//定板反
  1625. high_alarm_limit = pInfo->alarm_high_limit;
  1626. high_warn_limit = pInfo->warn_high_limit;
  1627. low_alarm_limit = pInfo->f_alarm_high_limit;
  1628. low_warn_limit = pInfo->f_warn_high_limit;
  1629. posi = eDaoChaPosi::DCP_FIX2INVERT;
  1630. }
  1631. else
  1632. {//反板定
  1633. low_alarm_limit = pInfo->alarm_high_limit;
  1634. low_warn_limit = pInfo->warn_high_limit;
  1635. high_alarm_limit = pInfo->f_alarm_high_limit;
  1636. high_warn_limit = pInfo->f_warn_high_limit;
  1637. posi = eDaoChaPosi::DCP_FIX2INVERT;
  1638. }
  1639. }
  1640. for (++it; it != pData->cend(); ++it)
  1641. {
  1642. //CString str = CTime(it->first / 1000).Format("%Y-%m-%d %H:%M:%S");
  1643. if (it->second == INVLID_VAL)
  1644. {
  1645. pInfo->tmLastCheckTime = it->first;
  1646. continue;
  1647. }
  1648. if (item.second->no < 2)
  1649. {
  1650. //1,2 锁闭力报警
  1651. if (it->second > pInfo->alarm_high_limit)
  1652. {
  1653. bAlarm = true;
  1654. alarm_refer = pInfo->alarm_high_limit;
  1655. alarm_value = it->second;
  1656. alarm_time = it->first;
  1657. pInfo->tmLastCheckTime = pData->rbegin()->first;
  1658. desc_type = 1;
  1659. break;
  1660. }
  1661. //else if (it->second > pInfo->warn_high_limit) //稳态报警单独判断
  1662. //{
  1663. // bWarn = true;
  1664. // alarm_refer = pInfo->warn_high_limit;
  1665. // alarm_value = it->second;
  1666. // alarm_time = it->first;
  1667. // pInfo->tmLastCheckTime = pData->rbegin()->first;
  1668. //}
  1669. }
  1670. else
  1671. {
  1672. assert(item.second->no == 2);
  1673. if (pInfo->alarm_high_limit == INT_MAX || pInfo->f_alarm_high_limit == INT_MAX || pInfo->alarm_high_limit == pInfo->f_alarm_high_limit)
  1674. break;
  1675. desc_type = 3;
  1676. if (it->second > high_alarm_limit)
  1677. {
  1678. bAlarm = true;
  1679. alarm_refer = high_alarm_limit;
  1680. alarm_value = it->second;
  1681. alarm_time = it->first;
  1682. pInfo->tmLastCheckTime = pData->rbegin()->first;
  1683. break;
  1684. }
  1685. else if (it->second < low_alarm_limit)
  1686. {
  1687. bAlarm = true;
  1688. alarm_refer = low_alarm_limit;
  1689. alarm_value = it->second;
  1690. alarm_time = it->first;
  1691. pInfo->tmLastCheckTime = pData->rbegin()->first;
  1692. break;
  1693. }
  1694. else if (low_warn_limit == INT_MIN || high_warn_limit == INT_MIN)
  1695. {
  1696. //未设置预警值
  1697. }
  1698. else if (it->second > high_warn_limit)
  1699. {
  1700. bWarn = true;
  1701. alarm_refer = high_warn_limit;
  1702. alarm_value = it->second;
  1703. alarm_time = it->first;
  1704. pInfo->tmLastCheckTime = pData->rbegin()->first;
  1705. }
  1706. else if (it->second < low_warn_limit)
  1707. {
  1708. bWarn = true;
  1709. alarm_refer = low_warn_limit;
  1710. alarm_value = it->second;
  1711. alarm_time = it->first;
  1712. pInfo->tmLastCheckTime = pData->rbegin()->first;
  1713. }
  1714. }
  1715. pInfo->tmLastCheckTime = it->first;
  1716. }
  1717. //稳态报警判断
  1718. if (item.second->no < 2 && bAlarm == false)
  1719. {
  1720. time_t tmEnd = pInfo->tmLastCheckTime / 1000;
  1721. int dif = tmEnd - tmStartSteady;
  1722. for (time_t i = tmStartSteady; i < tmEnd - 3; i++)
  1723. {
  1724. #ifdef _DEBUG
  1725. COleDateTime t(i);
  1726. TRACE("%s\r\n", t.Format("%Y-%m-%d %H:%M:%S"));
  1727. #endif // _DEBUG
  1728. auto j = pData->find(i * 1000);
  1729. if (j == pData->end()) continue;
  1730. time_t t2 = (i + 1) * 1000;
  1731. auto k = pData->find(t2);
  1732. if (k == pData->end()) continue;
  1733. t2 = (i + 2) * 1000;
  1734. k = pData->find(t2);
  1735. if (k == pData->end()) continue;
  1736. //连续3秒在线,继续找下一个结束点
  1737. t2 = (i + 3) * 1000;
  1738. auto g = pData->find(t2);
  1739. if (g == pData->end())
  1740. {
  1741. for (++k; k != pData->end(); k++)
  1742. {
  1743. if (k->first >= t2)
  1744. {
  1745. g = k;
  1746. break;
  1747. }
  1748. }
  1749. }
  1750. if (g == pData->end())
  1751. continue;
  1752. int high_num = 0, low_num = 0;
  1753. ConmpareValue(j, g, pInfo->warn_high_limit, high_num, low_num);
  1754. if (high_num > 0 && high_num <= 2)
  1755. {
  1756. bWarn = true;
  1757. desc_type = 2;
  1758. alarm_refer = pInfo->warn_high_limit;
  1759. alarm_value = high_num;
  1760. alarm_time = i * 1000; //报警时间
  1761. pInfo->tmLastCheckTime = pData->rbegin()->first;
  1762. break;
  1763. }
  1764. }
  1765. }
  1766. }
  1767. else if (item.second->type == eZL_ALARMTYPE::FRICTION_OVER_LIMIT)
  1768. {
  1769. auto pInfo = (FRICTION_OVER_LIMIT_INFO*)item.second;
  1770. #ifdef _DEBUG
  1771. if (pInfo->tmLastCheckTime > pData->rbegin()->first)
  1772. pInfo->tmLastCheckTime = pData->rbegin()->first;
  1773. #endif
  1774. auto it = pData->cbegin();
  1775. if (pInfo->tmLastCheckTime == 0) pInfo->tmLastCheckTime = pData->rbegin()->first;
  1776. else if (pInfo->tmLastCheckTime != 0) it = pData->find(pInfo->tmLastCheckTime);
  1777. if (it == pData->cend())
  1778. {
  1779. it = pData->find(pInfo->tmLastCheckTime / 1000 * 1000);
  1780. if (it == pData->cend())
  1781. {
  1782. pInfo->tmLastCheckTime = pData->rbegin()->first;
  1783. continue;
  1784. }
  1785. }
  1786. if (pInfo->tmLastCheckTime == pData->rbegin()->first) //没有新的数据
  1787. continue;
  1788. if (tmNow * 1000 - pInfo->tmLastCheckTime < 5000) //5秒检测一次
  1789. continue;
  1790. #ifdef _DEBUG
  1791. TRACE("%s.%s.%d.%d lastcheck:%s now:%s\r\n", mo.c_str(), mp.c_str(), pInfo->no, (int)eZL_ALARMTYPE::FRICTION_OVER_LIMIT,
  1792. CTime(pInfo->tmLastCheckTime / 1000).Format("%Y-%m-%d %H:%M:%S"), CTime(pData->rbegin()->first / 1000).Format("%Y-%m-%d %H:%M:%S"));
  1793. #endif // _DEBUG
  1794. if (item.second->no != 2) continue;
  1795. //从检测时间往前取30秒的数据
  1796. static const int dif_resist = 200; //稳定期判定差值
  1797. int up_max_resist = INT_MIN; //上最大值
  1798. time_t up_max_time = 0;
  1799. int up_avg_resist = INT_MIN; //上稳定期平均值
  1800. uint64_t up_stable_sum = 0; //上稳定期的和
  1801. int up_stable_time = 0; //上稳定期的数值个数
  1802. int up_start_resist_stabel = INT_MIN; //上稳定期开始值
  1803. time_t up_start_stable_time = 0; //规定是最大值之后的2秒
  1804. time_t up_end_stable_time = 0;//稳定结束的时间
  1805. int up_dif_time_stable = 0; //上稳定期的时长
  1806. int down_max_resist = INT_MAX; //上最大值
  1807. time_t down_max_time = 0;
  1808. int down_avg_resist = INT_MAX; //上稳定期平均值
  1809. uint64_t down_stable_sum = 0; //上稳定期的和
  1810. int down_stable_time = 0; //上稳定期的数值个数
  1811. int down_start_resist_stabel = INT_MAX; //上稳定期开始值
  1812. time_t down_start_stable_time = 0; //规定是最大值之后的2秒
  1813. int down_dif_time_stable = 0; //上稳定期的时长
  1814. time_t tmEnd = pData->rbegin()->first / 1000 + 1; //结束时间
  1815. time_t tmStart = tmEnd - 36;//开始时间是从上次35秒到当前秒
  1816. std::map<time_t, int>::const_iterator it_start, it_up_max, it_down_min, it_up_stable_start, it_down_stable_start, it_up_stable_end, it_down_stable_end;
  1817. //寻找起始时间
  1818. for (time_t i = tmStart; i < tmEnd; i++)
  1819. {
  1820. #ifdef _DEBUG
  1821. COleDateTime t(i);
  1822. TRACE("%s\r\n", t.Format("%Y-%m-%d %H:%M:%S"));
  1823. #endif // _DEBUG
  1824. it_start = pData->find(i * 1000);
  1825. if (it_start != pData->cend()) break;
  1826. }
  1827. pInfo->tmLastCheckTime = pData->rbegin()->first; //赋值最后的时间
  1828. if (it_start == pData->cend()) break;
  1829. bool bUp = true, bDown = true;
  1830. //先找最大值
  1831. for (auto i = it_start; i != pData->cend(); i++)
  1832. {
  1833. //寻找最大的点
  1834. if (i->second > up_max_resist)
  1835. {
  1836. up_max_resist = i->second;
  1837. it_up_max = i;
  1838. }
  1839. //寻找最小的点
  1840. if (i->second < down_max_resist)
  1841. {
  1842. down_max_resist = i->second;
  1843. it_down_min = i;
  1844. }
  1845. }
  1846. if (pInfo->tmLastCheckTime - it_up_max->first < 17000) bUp = false; //小于17秒
  1847. if (pInfo->tmLastCheckTime - it_down_min->first < 1700) bDown = false; //小于17秒
  1848. if(bUp == false && bDown == false) break;
  1849. //再找稳定开始
  1850. if (bUp)
  1851. {
  1852. it_up_stable_start = it_up_max;
  1853. for (auto i = it_up_max; i != pData->cend(); i++)
  1854. {
  1855. if (i->first - it_up_max->first < 2000) continue;
  1856. else
  1857. {
  1858. it_up_stable_start = i;
  1859. break;
  1860. }
  1861. }
  1862. if (it_up_stable_start == it_up_max) bUp = false;
  1863. }
  1864. if (bDown)
  1865. {
  1866. it_down_stable_start = it_down_min;
  1867. for (auto i = it_down_min; i != pData->cend(); i++)
  1868. {
  1869. if (i->first - it_down_min->first < 2000) continue;
  1870. else
  1871. {
  1872. it_down_stable_start = i;
  1873. break;
  1874. }
  1875. }
  1876. if (it_down_stable_start == it_down_min) bDown = false;
  1877. }
  1878. if (bUp == false && bDown == false) break;
  1879. if (bUp)
  1880. {
  1881. up_start_resist_stabel = it_up_stable_start->second;//稳定期开始的值
  1882. it_up_stable_end = it_up_stable_start;
  1883. for (auto i = it_up_stable_start; i != pData->cend(); i++)
  1884. {
  1885. if (abs(up_start_resist_stabel - i->second) < dif_resist)
  1886. {
  1887. up_stable_sum += i->second;
  1888. up_start_stable_time++;
  1889. it_up_stable_end = i;
  1890. continue;
  1891. }
  1892. else
  1893. {
  1894. break;
  1895. }
  1896. }
  1897. //时间条件限定在 15s-30s 之间
  1898. auto dif = it_up_stable_end->first - it_up_stable_start->first;
  1899. if (dif > 15000 && dif < 30000)
  1900. {
  1901. //平均值
  1902. up_avg_resist = up_stable_sum / up_start_stable_time;
  1903. if (up_avg_resist < pInfo->up_alarm_low_limit)
  1904. {
  1905. bAlarm = true;
  1906. alarm_value = up_avg_resist;
  1907. alarm_time = it_up_stable_start->first;
  1908. alarm_refer = pInfo->up_alarm_low_limit;
  1909. desc_type = 4;
  1910. }
  1911. else if (up_avg_resist < pInfo->up_warn_low_limit)
  1912. {
  1913. bWarn = true;
  1914. alarm_value = up_avg_resist;
  1915. alarm_time = it_up_stable_start->first;
  1916. alarm_refer = pInfo->up_warn_low_limit;
  1917. desc_type = 4;
  1918. }
  1919. }
  1920. }
  1921. if (bDown)
  1922. {
  1923. down_start_resist_stabel = it_down_stable_start->second;//稳定期开始的值
  1924. it_down_stable_end = it_down_stable_start;
  1925. for (auto i = it_down_stable_start; i != pData->cend(); i++)
  1926. {
  1927. if (abs(down_start_resist_stabel - i->second) < dif_resist)
  1928. {
  1929. down_stable_sum += i->second;
  1930. down_start_stable_time++;
  1931. it_down_stable_end = i;
  1932. continue;
  1933. }
  1934. else
  1935. {
  1936. break;
  1937. }
  1938. }
  1939. //时间条件限定在 15s-30s 之间
  1940. auto dif = it_down_stable_end->first - it_down_stable_start->first;
  1941. if (dif > 15000 && dif < 30000)
  1942. {
  1943. //平均值
  1944. down_avg_resist = down_stable_sum / down_start_stable_time;
  1945. if (down_avg_resist > pInfo->dw_alarm_high_limit)
  1946. {
  1947. bAlarm = true;
  1948. alarm_value = down_avg_resist;
  1949. alarm_time = it_down_stable_start->first;
  1950. alarm_refer = pInfo->dw_alarm_high_limit;
  1951. desc_type = 5;
  1952. }
  1953. else if (down_avg_resist > pInfo->dw_warn_high_limit)
  1954. {
  1955. bWarn = true;
  1956. alarm_value = down_avg_resist;
  1957. alarm_time = it_down_stable_start->first;
  1958. alarm_refer = pInfo->dw_warn_high_limit;
  1959. desc_type = 5;
  1960. }
  1961. }
  1962. }
  1963. }
  1964. else
  1965. {
  1966. //其他未实现
  1967. //TODO
  1968. assert(0);
  1969. continue;
  1970. }
  1971. }while (false);
  1972. //update
  1973. CString sql;
  1974. sql.Format("update rm_alarm_set SET time = %I64u WHERE mo = '%s' and mp = '%s' and no = %d and type = %d",
  1975. pAlarmSet->tmLastCheckTime, mo.c_str(), mp.c_str(), pAlarmSet->no, pAlarmSet->type);
  1976. if (CDBConnectPool::Instance()->DBExecuteSQL(sql) == FALSE)
  1977. {
  1978. assert(0);
  1979. CSimpleLog::Error("语句执行失败" + sql);
  1980. }
  1981. string name1, name2, name3, out_name, in_name;
  1982. CMonitorObjectMng::Instance()->GetNameByMoMp(momp, name1, name2, name3, out_name, in_name);
  1983. auto pMoMpInfo = CMonitorObjectMng::Instance()->GetMoMpInfo(momp);
  1984. if (item.second->no == 0 || item.second->no == 1)
  1985. {
  1986. if (name1.find("定位") != -1)
  1987. {
  1988. posi = item.second->no == 0 ? eDaoChaPosi::DCP_FIX : eDaoChaPosi::DCP_INVERT;
  1989. }
  1990. else if (name1.find("反位") != -1)
  1991. {
  1992. posi = item.second->no == 0 ? eDaoChaPosi::DCP_INVERT : eDaoChaPosi::DCP_FIX;
  1993. }
  1994. else if (pMoMpInfo)
  1995. SPDLOG_WARN("{} 未配置定位位关系 {}, {}, {}, {}", momp, name1, name2, pMoMpInfo->name1, pMoMpInfo->name2);
  1996. }
  1997. else if (item.second->no == 2)
  1998. {
  1999. //TODO
  2000. if (posi == eDaoChaPosi::DCP_UNKNOWN)
  2001. posi = eDaoChaPosi::DCP_FIX2INVERT;
  2002. }
  2003. else
  2004. ASSERT(FALSE);
  2005. if (bAlarm == false && bWarn == false) //没有报警
  2006. {
  2007. //恢复报警
  2008. std::lock_guard<mutex> lock(pService->m_mtxAlarm);
  2009. for (auto it = pService->m_lstUnConfirmAlarm.begin(); it != pService->m_lstUnConfirmAlarm.end();)
  2010. {
  2011. auto& alarm = *it;
  2012. if (alarm->no == item.second->no && alarm->type == pAlarmSet->type && alarm->recoveryTime.wYear < 2000
  2013. && alarm->mo.compare(mo) == 0 && alarm->mp.compare(mp) == 0)
  2014. {
  2015. auto pAlarmInfo = alarm;
  2016. //1. 更新数据库
  2017. //2. 更新内存数据
  2018. //3. 上送恢复消息
  2019. if (ctAlarmTime.GetTime() == 0)
  2020. ctAlarmTime = CTime::GetCurrentTime();
  2021. string recovery_time = ctAlarmTime.Format("%Y-%m-%d %H:%M:%S");
  2022. CString sql;
  2023. sql.Format("UPDATE [rm_alarm] SET [recovery_time]='%s' WHERE ID = %d;", recovery_time.c_str(), pAlarmInfo->id);
  2024. CDBConnectPool::Instance()->DBExecuteSQL(sql);
  2025. if (pAlarmInfo->ack_result == 1)
  2026. {
  2027. it = pService->m_lstUnConfirmAlarm.erase(it);
  2028. continue;
  2029. }
  2030. else
  2031. {
  2032. ctAlarmTime.GetAsSystemTime(alarm->recoveryTime);
  2033. }
  2034. g_p315ClientManager->GetTcpClient()->SendAlarmData(CMonitorObjectMng::Instance()->GetZZJNO(mo + '.' + mp), 2, CTime(pAlarmInfo->time).GetTime(), ctAlarmTime.GetTime(),
  2035. pAlarmInfo->type, posi, pAlarmInfo->level, eLowHigh::LH_HIGH,
  2036. TIEDA_ACQ_VALUE(alarm->val, TIEDA_VAL_STATE::TVS_MOVEING), TIEDA_ACQ_VALUE(alarm->refer_val, TIEDA_VAL_STATE::TVS_MOVEING), 0);
  2037. }
  2038. ++it;
  2039. }
  2040. continue; //误报警继续下一个循环
  2041. }
  2042. //查找是否原有的报警已存在
  2043. uint8_t level = 0;
  2044. if (bAlarm) level = 1;
  2045. ctAlarmTime = CTime(alarm_time / 1000);
  2046. {
  2047. std::lock_guard<mutex> lock(pService->m_mtxAlarm);
  2048. for (const auto& alarm : pService->m_lstUnConfirmAlarm)
  2049. {
  2050. if (alarm->no == item.second->no && alarm->type == pAlarmSet->type && level == alarm->level && alarm->recoveryTime.wYear < 2000 //预警和报警单独计算
  2051. && alarm->mo.compare(mo) == 0 && alarm->mp.compare(mp) == 0)
  2052. {
  2053. pAlarmInfo = alarm;
  2054. break;
  2055. }
  2056. }
  2057. }
  2058. if (pAlarmInfo == nullptr)
  2059. {
  2060. bNew = true;
  2061. pAlarmInfo = new ALARM_INFO;
  2062. pAlarmInfo->event_id = hjfunc_GetGUID();
  2063. pAlarmInfo->id = ++pService->m_nAlarmID;
  2064. pAlarmInfo->level = (bAlarm == 1);
  2065. pAlarmInfo->mo = mo;
  2066. pAlarmInfo->mp = mp;
  2067. pAlarmInfo->no = item.second->no;
  2068. ctAlarmTime.GetAsSystemTime(pAlarmInfo->time);
  2069. pAlarmInfo->time.wMilliseconds = alarm_time % 1000;
  2070. pAlarmInfo->type = item.second->type;
  2071. pAlarmInfo->val = alarm_value;
  2072. pAlarmInfo->refer_val = alarm_refer;
  2073. pAlarmInfo->posi = posi;
  2074. pAlarmInfo->zzjno = CMonitorObjectMng::Instance()->GetZZJNO(momp);
  2075. pAlarmInfo->sunroof = CSkylightMng::GetInstance()->IsSkylight(&pAlarmInfo->time);
  2076. char szInfo[200];
  2077. if (desc_type == 1)
  2078. {
  2079. sprintf_s(szInfo, sizeof(szInfo), "瞬时冲击力超限报警,报警值为%dN, 参考值为%dN", alarm_value, alarm_refer);//101 102
  2080. if (pAlarmInfo->type == eZL_ALARMTYPE::MAX_OVER_LIMIT)
  2081. {
  2082. g_p315ClientManager->GetTcpClient()->SendAlarmData(pAlarmInfo->zzjno, 1, CTime(pAlarmInfo->time).GetTime(), 0xFFFFFFFF,
  2083. pAlarmInfo->type, pAlarmInfo->posi, pAlarmInfo->level, eLowHigh::LH_HIGH,
  2084. TIEDA_ACQ_VALUE(alarm_value, TIEDA_VAL_STATE::TVS_POLL), TIEDA_ACQ_VALUE(alarm_refer, TIEDA_VAL_STATE::TVS_POLL), 0);
  2085. }
  2086. }
  2087. else if (desc_type == 2)
  2088. sprintf_s(szInfo, sizeof(szInfo), "稳态值超限预警, 超限次数为%d次, 参考值为%dN", alarm_value, alarm_refer);
  2089. else if (desc_type == 3)
  2090. sprintf_s(szInfo, sizeof(szInfo), "转换阻力值超限报警, 报警值为%dN, 参考值为%dN", alarm_value, alarm_refer);
  2091. else if (desc_type == 4)
  2092. {
  2093. sprintf_s(szInfo, sizeof(szInfo), "摩擦力超限%s, 缩进位报警值为%dN, 参考值为:%dN", bAlarm == 1 ? "告警" : "预警", alarm_value, alarm_refer);
  2094. string direct1, direct2;
  2095. CMonitorObjectMng::Instance()->GetDirectByMoMp(mo + '.' + mp, direct1, direct2);
  2096. }
  2097. else if (desc_type == 5)
  2098. {
  2099. sprintf_s(szInfo, sizeof(szInfo), "摩擦力超限%s, 伸出位报警值为%dN, 参考值为:%dN", bAlarm == 1 ? "告警" : "预警", alarm_value, alarm_refer);
  2100. string direct1, direct2;
  2101. CMonitorObjectMng::Instance()->GetDirectByMoMp(mo + '.' + mp, direct1, direct2);
  2102. }
  2103. else
  2104. assert(0);
  2105. pAlarmInfo->desc = szInfo;
  2106. lock_guard<mutex> lock(pService->m_mtxAlarm);
  2107. pService->m_lstUnConfirmAlarm.push_back(pAlarmInfo);
  2108. }
  2109. //send
  2110. if (bNew) //不再推送
  2111. {
  2112. rapidjson::StringBuffer buffer;
  2113. auto ret = AlarmInfo2Pack(pAlarmInfo, buffer);
  2114. const char* output = buffer.GetString();
  2115. //CAppService::Instance()->GetLwsServer()->SendPackToALLClient_with_noEncode((uint8_t*)output, buffer.GetLength());
  2116. CAppService::Instance()->GetMgServer()->SendToAllClient(output, buffer.GetLength());
  2117. }
  2118. //save
  2119. if (bNew)
  2120. {
  2121. CString sql;
  2122. sql.Format("INSERT INTO [rm_alarm]([ID],[mo],[mp],[no],[type],[occur_time],[level],[desc],[suggest],[val],[event_id],[rel_id],posi,loworhigh,referval,[sunroof]) "\
  2123. "VALUES(%d, '%s', '%s', %d, %d, '%s', %d, '%s', '%s', %d, '%s', '%s',%d,%d,%d,%d);",
  2124. pAlarmInfo->id, pAlarmInfo->mo.c_str(), pAlarmInfo->mp.c_str(), pAlarmInfo->no, pAlarmInfo->type,
  2125. ctAlarmTime.Format("%Y-%m-%d %H:%M:%S"), pAlarmInfo->level, pAlarmInfo->desc.c_str(), pAlarmInfo->suggest.c_str(), pAlarmInfo->val,
  2126. pAlarmInfo->event_id.c_str(), pAlarmInfo->rel_id.c_str(), uint8_t(pAlarmInfo->posi), uint8_t(pAlarmInfo->loworhigh), pAlarmInfo->refer_val, pAlarmInfo->sunroof);
  2127. if (false == CDBConnectPool::Instance()->DBExecuteSQL(sql))
  2128. CSimpleLog::Error("执行语句失败" + sql);
  2129. else
  2130. {
  2131. uint8_t* pack = nullptr; int len = 0;
  2132. auto packno = CResistAlarmMng::GeneralNewAlarmData(pAlarmInfo, &pack, &len);
  2133. if (packno) CSuperManager::Instance()->SendPack(pack, len, packno, E_ZL_PROTOCAL::ZL_ALARM_DATA, true);
  2134. //g_p315ClientManager->GetTcpClient()->SendAlarmData(pAlarmInfo->no, 1, ctAlarmTime.GetTime(), 0xFFFFFFFF, (WORD)pAlarmInfo->type, pAlarmInfo->val, 0, 0);
  2135. delete[] pack;
  2136. pack = nullptr;
  2137. }
  2138. }
  2139. }
  2140. //保持力下降检测
  2141. if (tmNow - pService->m_lastDetectDrop >= 30)
  2142. {
  2143. pService->m_lastDetectDrop = tmNow;
  2144. time_t tt;
  2145. time(&tt);
  2146. for (const auto& item : pService->m_alarm_set)
  2147. {
  2148. if (item.second == nullptr) continue;
  2149. if (item.second->enable == false) continue;
  2150. if (item.second->type != eZL_ALARMTYPE::RETENSION_FORCE) //这个逻辑里面只支持保持力
  2151. continue;
  2152. if ((eSuoBiPosi)item.second->no != eSuoBiPosi::SB_FIX && (eSuoBiPosi)item.second->no != eSuoBiPosi::SB_INVERT)
  2153. continue;
  2154. int nPos = item.first.find('.');
  2155. if (nPos == -1) continue;
  2156. int nPos2 = item.first.find('.', nPos + 1);
  2157. if (nPos2 == -1) continue;
  2158. string mo = item.first.substr(0, nPos);
  2159. string mp = item.first.substr(nPos + 1, nPos2 - nPos - 1);
  2160. string momp = item.first.substr(0, nPos2);
  2161. auto pInfo = CMonitorObjectMng::Instance()->GetMoMpInfo(momp);
  2162. //SPDLOG_INFO("[保持力下降]{} {}", momp, (bool)pInfo);
  2163. //ASSERT(pInfo && pInfo->pDeivce);
  2164. if (pInfo == nullptr) continue;
  2165. if (pInfo->pDeivce == nullptr) continue;
  2166. auto pAlarmSet = (RETENSION_FORCE_DROP*)item.second;
  2167. if (pAlarmSet->tmLastCheckTime == pInfo->pDeivce->m_ctUpdateTime.GetTime()) continue;//不重复检测
  2168. pAlarmSet->tmLastCheckTime = pInfo->pDeivce->m_ctUpdateTime.GetTime();
  2169. auto pMap0 = pInfo->pDeivce->GetStatInfo(pInfo->idx, 0);
  2170. auto pMap1 = pInfo->pDeivce->GetStatInfo(pInfo->idx, 1);
  2171. //auto pMap2 = pInfo->pDeivce->GetStatInfo(pInfo->idx, 0);
  2172. std::map<time_t, tagSecondStatInfo>* pMap = nullptr;
  2173. if (((eSuoBiPosi)pAlarmSet->no == eSuoBiPosi::SB_FIX && pInfo->name1.find("定位") != -1) ||
  2174. ((eSuoBiPosi)pAlarmSet->no == eSuoBiPosi::SB_INVERT && pInfo->name1.find("反位") != -1))
  2175. {
  2176. if (pMap0 == nullptr || pMap0->size() < 3) {
  2177. //SPDLOG_INFO("[保持力下降]{}.{} {} 未满足检测条件. {} ", mo, mp,
  2178. // (eSuoBiPosi)pAlarmSet->no == eSuoBiPosi::SB_FIX ? "定位" : "反位", pMap0->size());
  2179. continue;
  2180. }
  2181. pMap = pMap0;
  2182. }
  2183. else if(((eSuoBiPosi)pAlarmSet->no == eSuoBiPosi::SB_FIX && pInfo->name2.find("定位") != -1) ||
  2184. ((eSuoBiPosi)pAlarmSet->no == eSuoBiPosi::SB_INVERT && pInfo->name2.find("反位") != -1))
  2185. {
  2186. if (pMap1 == nullptr || pMap1->size() < 3) {
  2187. //SPDLOG_INFO("[保持力下降]{}.{} {} 未满足检测条件. {} ", mo, mp,
  2188. // (eSuoBiPosi)pAlarmSet->no == eSuoBiPosi::SB_FIX ? "定位" : "反位", pMap1->size());
  2189. continue;
  2190. }
  2191. pMap = pMap1;
  2192. }
  2193. else
  2194. continue;
  2195. //zzj位置是否变化
  2196. bool bEPOSChanged = CMonitorObjectMng::Instance()->IsZZJEPOSChanged(momp,
  2197. (eSuoBiPosi)pAlarmSet->no == eSuoBiPosi::SB_FIX ? DAOCHA_POSITION::MP_FIX : DAOCHA_POSITION::MP_INVERT, tt);
  2198. //位置变化后不能使用
  2199. if (bEPOSChanged) continue;
  2200. int nMinVal = INT_MAX;
  2201. time_t tMinTime = 0;
  2202. {
  2203. lock_guard<mutex> lock(pInfo->pDeivce->m_mtx);
  2204. auto it = pMap->end();
  2205. auto it0 = --it;
  2206. auto it1 = --it;
  2207. auto it2 = --it;
  2208. if (momp.compare(g_strMoMp) == 0)
  2209. SPDLOG_INFO("[保持力下降]{}.{} 波动值. {} {} {}", mo, mp, it2->second.dif_val, it1->second.dif_val, it0->second.dif_val);
  2210. if (it0->second.dif_val < 100 && it1->second.dif_val < 100 && it2->second.dif_val < 100)
  2211. {
  2212. nMinVal = it0->second.max_val;
  2213. tMinTime = it0->second.max_time;
  2214. if (it1->second.max_val > nMinVal)
  2215. {
  2216. nMinVal = it1->second.max_val;
  2217. tMinTime = it1->second.max_time;
  2218. }
  2219. else if (it2->second.max_val < nMinVal)
  2220. {
  2221. nMinVal = it2->second.max_val;
  2222. tMinTime = it2->second.max_time;
  2223. }
  2224. }
  2225. }
  2226. if (nMinVal != INT_MAX)
  2227. JudgeAlarm(pService, pAlarmSet, tMinTime, nMinVal, (eSuoBiPosi)pAlarmSet->no, pInfo->mo, pInfo->mp, pInfo->zzjno);
  2228. }
  2229. //清理无效位置
  2230. CMonitorObjectMng::Instance()->ClearZZJHistroyEPOS(tt);
  2231. }
  2232. } while (pService->m_bWork);
  2233. if (bLock)
  2234. {
  2235. g_lockSync.Unlock();
  2236. bLock = false;
  2237. }
  2238. }
  2239. typedef struct tagMoMp
  2240. {
  2241. string mo;
  2242. string mp;
  2243. uint8_t no; //传感器的序号
  2244. tagMoMp(string a, string b)
  2245. {
  2246. mo = a;
  2247. mp = b;
  2248. }
  2249. tagMoMp(uint8_t a)
  2250. {
  2251. no = a;
  2252. }
  2253. };
  2254. typedef struct offline_info
  2255. {
  2256. string imei;
  2257. CTime time;
  2258. uint8_t idx = 0;
  2259. std::vector<tagMoMp> vct;
  2260. offline_info(string a, CTime b)
  2261. {
  2262. imei = a;
  2263. time = b;
  2264. }
  2265. offline_info(string a, CTime b, uint8_t c, uint8_t d)
  2266. {
  2267. imei = a;
  2268. time = b;
  2269. idx = c;
  2270. vct.push_back(tagMoMp(d));
  2271. }
  2272. };
  2273. void CResistAlarmMng::ThreadProcDevice(DWORD_PTR param)
  2274. {
  2275. auto pService = (CResistAlarmMng*)param;
  2276. this_thread::sleep_for(chrono::seconds(2));
  2277. static CTimeSpan offline_time(0, 0, 3, 0); //设备离线时间暂定5分钟
  2278. static CTimeSpan offline_limit(30, 0, 0, 0);
  2279. bool bLock = false;
  2280. do
  2281. {
  2282. if (bLock)
  2283. {
  2284. g_lockSync.Unlock();
  2285. bLock = false;
  2286. }
  2287. for (int i = 0; i < 30; i++)
  2288. {
  2289. if (false == pService->m_bWork) break;
  2290. Sleep(1000);
  2291. }
  2292. g_lockSync.ReadLock();
  2293. bLock = true;
  2294. CTime ctNow = CTime::GetCurrentTime();
  2295. //离线设备
  2296. {
  2297. auto& pMap = CMonitorObjectMng::Instance()->m_mapMoMpInfo;
  2298. for (auto& it : pMap)
  2299. {
  2300. auto pInfo = it.second;
  2301. if (pInfo == nullptr) continue;
  2302. bool bAlarm = false;
  2303. if (!pInfo->pDeivce) continue;
  2304. const auto pDevice = pInfo->pDeivce;
  2305. if (ctNow - pDevice->m_ctUpdateTime > offline_time)
  2306. bAlarm = true;
  2307. else
  2308. int ii = 0;
  2309. ALARM_INFO* pAlarmInfo = nullptr;
  2310. bool bNew = false;
  2311. auto ctAlarmTime = ctNow;
  2312. {
  2313. std::lock_guard<mutex> lock(pService->m_mtxAlarm);
  2314. for (auto it = pService->m_lstUnConfirmAlarm.begin(); it != pService->m_lstUnConfirmAlarm.end();)
  2315. {
  2316. auto pAlarm = *it;
  2317. if (pAlarm->type == eZL_ALARMTYPE::EQUIP_OFFLINE && pAlarm->recoveryTime.wYear < 2000 &&
  2318. pAlarm->mo.compare(pInfo->mo) == 0 && pAlarm->mp.compare(pInfo->mp) == 0)
  2319. {
  2320. pAlarmInfo = pAlarm;
  2321. if (bAlarm) //连续报警
  2322. {
  2323. break;
  2324. }
  2325. else
  2326. {
  2327. //恢复报警
  2328. //更新数据库
  2329. string recovery_time = ctAlarmTime.Format("%Y-%m-%d %H:%M:%S");
  2330. CString sql;
  2331. sql.Format("UPDATE [rm_alarm] SET [recovery_time]='%s' WHERE ID = %d;", recovery_time.c_str(), pAlarmInfo->id);
  2332. CDBConnectPool::Instance()->DBExecuteSQL(sql);
  2333. //315上送
  2334. g_p315ClientManager->GetTcpClient()->SendAlarmData(pInfo->zzjno, 2, CTime(pAlarm->time).GetTime(), ctAlarmTime.GetTime(), pAlarm->type,
  2335. eDaoChaPosi::DCP_UNKNOWN, pAlarm->level, eLowHigh::LH_UNKNOWN, TIEDA_ACQ_VALUE(INT_MIN), TIEDA_ACQ_VALUE(INT_MIN), 0);
  2336. if (pAlarm->ack_result == 1)
  2337. {
  2338. it = pService->m_lstUnConfirmAlarm.erase(it);
  2339. continue;
  2340. }
  2341. else
  2342. {
  2343. ctAlarmTime.GetAsSystemTime(pAlarm->recoveryTime);
  2344. }
  2345. }
  2346. }
  2347. ++it;
  2348. }
  2349. }
  2350. if (bAlarm == false) continue;
  2351. if (pAlarmInfo == nullptr)
  2352. {
  2353. bNew = true;
  2354. pAlarmInfo = new ALARM_INFO;
  2355. pAlarmInfo->event_id = hjfunc_GetGUID();
  2356. pAlarmInfo->id = ++pService->m_nAlarmID;
  2357. pAlarmInfo->level = 1;
  2358. pAlarmInfo->mo = pInfo->mo;
  2359. pAlarmInfo->mp = pInfo->mp;
  2360. pAlarmInfo->no = -1;
  2361. ctAlarmTime.GetAsSystemTime(pAlarmInfo->time);
  2362. //pAlarmInfo->time.wMilliseconds = alarm_time % 1000;
  2363. pAlarmInfo->type = eZL_ALARMTYPE::EQUIP_OFFLINE;
  2364. pAlarmInfo->val = 0;
  2365. pAlarmInfo->zzjno = pInfo->zzjno;
  2366. char szInfo[200];
  2367. sprintf_s(szInfo, sizeof(szInfo), "设备离线时间:%s", (LPCSTR)(pDevice->m_ctUpdateTime.Format("%Y-%m-%d %H:%M:%S")));
  2368. pAlarmInfo->desc = szInfo;
  2369. pAlarmInfo->sunroof = CSkylightMng::GetInstance()->IsSkylight(&pAlarmInfo->time);
  2370. lock_guard<mutex> lock(pService->m_mtxAlarm);
  2371. pService->m_lstUnConfirmAlarm.push_back(pAlarmInfo);
  2372. }
  2373. //send
  2374. if (bNew) //不再推送
  2375. {
  2376. rapidjson::StringBuffer buffer;
  2377. auto ret = AlarmInfo2Pack(pAlarmInfo, buffer);
  2378. const char* output = buffer.GetString();
  2379. //CAppService::Instance()->GetLwsServer()->SendPackToALLClient_with_noEncode((uint8_t*)output, buffer.GetLength());
  2380. CAppService::Instance()->GetMgServer()->SendToAllClient(output, buffer.GetLength());
  2381. }
  2382. //save
  2383. if (bNew)
  2384. {
  2385. CString sql;
  2386. sql.Format("INSERT INTO [rm_alarm]([ID],[mo],[mp],[no],[type],[occur_time],[level],[desc],[suggest],[val],[event_id],[rel_id],posi,loworhigh,referval,[sunroof]) "\
  2387. "VALUES(%d, '%s', '%s', %d, %d, '%s', %d, '%s', '%s', %d, '%s', '%s',%d,%d,%d,%d);",
  2388. pAlarmInfo->id, pAlarmInfo->mo.c_str(), pAlarmInfo->mp.c_str(), pAlarmInfo->no, pAlarmInfo->type,
  2389. (LPCSTR)(ctAlarmTime.Format("%Y-%m-%d %H:%M:%S")), pAlarmInfo->level, pAlarmInfo->desc.c_str(), pAlarmInfo->suggest.c_str(), pAlarmInfo->val,
  2390. pAlarmInfo->event_id.c_str(), pAlarmInfo->rel_id.c_str(), uint8_t(pAlarmInfo->posi), uint8_t(pAlarmInfo->loworhigh), pAlarmInfo->refer_val, pAlarmInfo->sunroof);
  2391. if (false == CDBConnectPool::Instance()->DBExecuteSQL(sql))
  2392. CSimpleLog::Error("执行语句失败" + sql);
  2393. else
  2394. {
  2395. uint8_t* pack = nullptr; int len = 0;
  2396. auto packno = CResistAlarmMng::GeneralNewAlarmData(pAlarmInfo, &pack, &len);
  2397. if (packno) CSuperManager::Instance()->SendPack(pack, len, packno, E_ZL_PROTOCAL::ZL_ALARM_DATA, true);
  2398. g_p315ClientManager->GetTcpClient()->SendAlarmData(pAlarmInfo->zzjno, 1, ctAlarmTime.GetTime(), 0xFFFFFFFF, pAlarmInfo->type, eDaoChaPosi::DCP_UNKNOWN, 1, eLowHigh::LH_UNKNOWN,
  2399. TIEDA_ACQ_VALUE(INT_MIN), TIEDA_ACQ_VALUE(INT_MIN), 0);
  2400. delete[] pack;
  2401. pack = nullptr;
  2402. }
  2403. }
  2404. }
  2405. }
  2406. } while (pService->m_bWork);
  2407. if (bLock)
  2408. {
  2409. g_lockSync.Unlock();
  2410. bLock = false;
  2411. }
  2412. }
  2413. void CResistAlarmMng::ThreadProcMove(DWORD_PTR param)
  2414. {
  2415. auto pService = (CResistAlarmMng*)param;
  2416. Sleep(2000);
  2417. time_t tmNow;
  2418. auto strIniPath = CSimpleLog::GetAppDir() + "DataServices.ini";
  2419. bool bLock = false;
  2420. do
  2421. {
  2422. if (bLock)
  2423. {
  2424. g_lockSync.Unlock();
  2425. bLock = false;
  2426. }
  2427. Sleep(800);
  2428. time(&tmNow);
  2429. if (pService->m_lastDetectMove > tmNow)
  2430. {
  2431. SPDLOG_INFO("时间有跳变");
  2432. pService->m_lastDetectMove = tmNow;
  2433. }
  2434. //10秒归档一次 扳动判断
  2435. if (tmNow - pService->m_lastDetectMove < 10)
  2436. continue;
  2437. g_lockSync.ReadLock();
  2438. bLock = true;
  2439. pService->m_lastDetectMove = tmNow;
  2440. char szTemp[100] = { 0 };
  2441. ::GetPrivateProfileString("SET", "momp", "", szTemp, sizeof(szTemp), strIniPath);
  2442. g_strMoMp = szTemp;
  2443. //SPDLOG_INFO("扳动判断:{}", g_strMoMp);
  2444. auto& mapDevice = CDeviceMng::Instance()->m_map_devices;
  2445. for (auto& it : mapDevice)
  2446. {
  2447. auto pDevice = it.second;
  2448. for (auto i = 0; i < 3; i++)
  2449. {
  2450. string mo, mp;
  2451. if (FALSE == CMonitorObjectMng::Instance()->IMEI2MOMP(pDevice->imei, i, mo, mp))
  2452. {
  2453. //CSimpleLog::Error(fmt::format("IMEI2MOMP fail. imei:{}, idx:{}, {}:{}", pDevice->imei, i, __FILE__, __LINE__).c_str());
  2454. continue; //没有绑定返回
  2455. }
  2456. std::map<time_t, tagSecondStatInfo> map0, map1, map2;
  2457. std::map<time_t, int>* mapData[3] = { 0 };
  2458. if (i == 0)
  2459. {
  2460. if (tmNow - pDevice->GetUpdateTime(0) < 3) continue; //3秒钟 无数据上送 开始判断
  2461. else if (tmNow - pDevice->m_tmMoveDetectTime0 < 10) continue; //距离上次应大于10秒
  2462. else if (pDevice->m_mapSecondStatInfo00.size() == 0) continue;
  2463. const auto tCheck = pDevice->m_tmMoveDetectTime0;
  2464. lock_guard<mutex> lock(pDevice->m_mtx);
  2465. for (auto it = pDevice->m_mapSecondStatInfo00.rbegin(); it != pDevice->m_mapSecondStatInfo00.rend(); it++)
  2466. {
  2467. if (it->first < tCheck) break;
  2468. map0.insert(*it);
  2469. }
  2470. for (auto it = pDevice->m_mapSecondStatInfo01.rbegin(); it != pDevice->m_mapSecondStatInfo01.rend(); it++)
  2471. {
  2472. if (it->first < tCheck) break;
  2473. map1.insert(*it);
  2474. }
  2475. for (auto it = pDevice->m_mapSecondStatInfo02.rbegin(); it != pDevice->m_mapSecondStatInfo02.rend(); it++)
  2476. {
  2477. if (it->first < tCheck) break;
  2478. map2.insert(*it);
  2479. }
  2480. if (map2.size() > 2)
  2481. { //拷贝原始数据
  2482. time_t tStart = 0;
  2483. std::map<time_t, int>::iterator itStart2;
  2484. if (tCheck)
  2485. {
  2486. for (auto ii = tCheck; ii < tmNow; ii++)
  2487. {
  2488. itStart2 = pDevice->map_resist_idx02.find(ii*1000);
  2489. if (itStart2 != pDevice->map_resist_idx02.end())
  2490. {
  2491. tStart = ii*1000;
  2492. break;
  2493. }
  2494. }
  2495. }
  2496. else
  2497. {
  2498. itStart2 = pDevice->map_resist_idx02.begin();
  2499. tStart = itStart2->first;
  2500. }
  2501. if (tStart)
  2502. {
  2503. mapData[0] = new std::map<time_t, int>(pDevice->map_resist_idx00.find(tStart), pDevice->map_resist_idx00.end());
  2504. mapData[1] = new std::map<time_t, int>(pDevice->map_resist_idx01.find(tStart), pDevice->map_resist_idx01.end());
  2505. mapData[2] = new std::map<time_t, int>(itStart2, pDevice->map_resist_idx02.end());
  2506. }
  2507. else
  2508. ASSERT(0);
  2509. }
  2510. pDevice->m_tmMoveDetectTime0 = tmNow;
  2511. }
  2512. else if (i == 1)
  2513. {
  2514. if (tmNow - pDevice->GetUpdateTime(1) < 3) continue; //3秒钟 无数据上送 开始判断
  2515. else if (tmNow - pDevice->m_tmMoveDetectTime1 < 10) continue; //距离上次应大于10秒
  2516. else if (pDevice->m_mapSecondStatInfo10.size() == 0) continue;
  2517. const auto tCheck = pDevice->m_tmMoveDetectTime1;
  2518. lock_guard<mutex> lock(pDevice->m_mtx);
  2519. for (auto it = pDevice->m_mapSecondStatInfo10.rbegin(); it != pDevice->m_mapSecondStatInfo10.rend(); it++)
  2520. {
  2521. if (it->first < tCheck) break;
  2522. map0.insert(*it);
  2523. }
  2524. for (auto it = pDevice->m_mapSecondStatInfo11.rbegin(); it != pDevice->m_mapSecondStatInfo11.rend(); it++)
  2525. {
  2526. if (it->first < tCheck) break;
  2527. map1.insert(*it);
  2528. }
  2529. for (auto it = pDevice->m_mapSecondStatInfo12.rbegin(); it != pDevice->m_mapSecondStatInfo12.rend(); it++)
  2530. {
  2531. if (it->first < tCheck) break;
  2532. map2.insert(*it);
  2533. }
  2534. if (map2.size() > 2)
  2535. { //拷贝原始数据
  2536. time_t tStart = 0;
  2537. std::map<time_t, int>::iterator itStart2;
  2538. if (tCheck)
  2539. {
  2540. for (auto ii = tCheck; ii < tmNow; ii++)
  2541. {
  2542. itStart2 = pDevice->map_resist_idx12.find(ii*1000);
  2543. if (itStart2 != pDevice->map_resist_idx12.end())
  2544. {
  2545. tStart = ii*1000;
  2546. break;
  2547. }
  2548. }
  2549. }
  2550. else
  2551. {
  2552. itStart2 = pDevice->map_resist_idx12.begin();
  2553. tStart = itStart2->first;
  2554. }
  2555. if (tStart)
  2556. {
  2557. mapData[0] = new std::map<time_t, int>(pDevice->map_resist_idx10.find(tStart), pDevice->map_resist_idx10.end());
  2558. mapData[1] = new std::map<time_t, int>(pDevice->map_resist_idx11.find(tStart), pDevice->map_resist_idx11.end());
  2559. mapData[2] = new std::map<time_t, int>(itStart2, pDevice->map_resist_idx12.end());
  2560. }
  2561. else
  2562. ASSERT(0);
  2563. }
  2564. pDevice->m_tmMoveDetectTime1 = tmNow;
  2565. }
  2566. else
  2567. {
  2568. if (tmNow - pDevice->GetUpdateTime(2) < 3) continue; //3秒钟 无数据上送 开始判断
  2569. else if (tmNow - pDevice->m_tmMoveDetectTime2 < 10) continue; //距离上次应大于10秒
  2570. else if (pDevice->m_mapSecondStatInfo20.size() == 0) continue;
  2571. //else if (pDevice->m_mapSecondStatInfo20.rbegin()->second.dif_val > 100) continue; //最后值需要稳定 2024年5月31日移除
  2572. //else if (pDevice->m_mapSecondStatInfo21.rbegin()->second.dif_val > 100) continue;
  2573. //else if (pDevice->m_mapSecondStatInfo22.rbegin()->second.dif_val > 100) continue;
  2574. const auto tCheck = pDevice->m_tmMoveDetectTime2;
  2575. lock_guard<mutex> lock(pDevice->m_mtx);
  2576. for (auto it = pDevice->m_mapSecondStatInfo20.rbegin(); it != pDevice->m_mapSecondStatInfo20.rend(); it++)
  2577. {
  2578. //RACE("20: first:%s check:%s\r\n", CTime(it->first).Format("%Y-%m-%d %H:%M:%S"), CTime(tCheck).Format("%Y-%m-%d %H:%M:%S"));
  2579. if (it->first < tCheck) break;
  2580. map0.insert(*it);
  2581. }
  2582. for (auto it = pDevice->m_mapSecondStatInfo21.rbegin(); it != pDevice->m_mapSecondStatInfo21.rend(); it++)
  2583. {
  2584. //TRACE("21: first:%s check:%s\r\n", CTime(it->first).Format("%Y-%m-%d %H:%M:%S"), CTime(tCheck).Format("%Y-%m-%d %H:%M:%S"));
  2585. if (it->first < tCheck) break;
  2586. map1.insert(*it);
  2587. }
  2588. for (auto it = pDevice->m_mapSecondStatInfo22.rbegin(); it != pDevice->m_mapSecondStatInfo22.rend(); it++)
  2589. {
  2590. //TRACE("22: first:%s check:%s\r\n", CTime(it->first).Format("%Y-%m-%d %H:%M:%S"), CTime(tCheck).Format("%Y-%m-%d %H:%M:%S"));
  2591. if (it->first < tCheck) break;
  2592. map2.insert(*it);
  2593. }
  2594. if (map2.size() > 2)
  2595. { //拷贝原始数据
  2596. time_t tStart = 0;
  2597. std::map<time_t, int>::iterator itStart2;
  2598. if (tCheck)
  2599. {
  2600. for (auto ii = tCheck; ii < tmNow; ii++)
  2601. {
  2602. itStart2 = pDevice->map_resist_idx22.find(ii*1000);
  2603. if (itStart2 != pDevice->map_resist_idx22.end())
  2604. {
  2605. tStart = ii*1000;
  2606. break;
  2607. }
  2608. }
  2609. }
  2610. else
  2611. {
  2612. itStart2 = pDevice->map_resist_idx22.begin();
  2613. tStart = itStart2->first;
  2614. }
  2615. if (tStart)
  2616. {
  2617. mapData[0] = new std::map<time_t, int>(pDevice->map_resist_idx20.find(tStart), pDevice->map_resist_idx20.end());
  2618. mapData[1] = new std::map<time_t, int>(pDevice->map_resist_idx21.find(tStart), pDevice->map_resist_idx21.end());
  2619. mapData[2] = new std::map<time_t, int>(itStart2, pDevice->map_resist_idx22.end());
  2620. }
  2621. else
  2622. assert(0);
  2623. }
  2624. TRACE("update: m_tmMoveDetectTime2:%s tmNow:%s size:%d\r\n", CTime(pDevice->m_tmMoveDetectTime2).Format("%Y-%m-%d %H:%M:%S"), CTime(tmNow).Format("%Y-%m-%d %H:%M:%S"), map2.size());
  2625. pDevice->m_tmMoveDetectTime2 = tmNow;
  2626. }
  2627. if (map2.size() <= 2) continue;
  2628. if (!mapData[0] && !mapData[1] && !mapData[2]) continue;
  2629. if (mapData[0]->size() != mapData[1]->size() || mapData[1]->size() != mapData[2]->size()) continue;
  2630. SPDLOG_INFO("扳动判断数据{}.{} map:{} 0:{} 1:{} 2:{}", mo, mp, map2.size(), mapData[0]->size(), mapData[1]->size(), mapData[2]->size());
  2631. list<CONVERT_RESIST> lstResit2; //转换力数据
  2632. list<PASS_RESIST> lstPass; //过车数据
  2633. std::map<time_t, int64_t> maxlock0, maxlock1;//锁闭力 前面时间戳, 后面 高4位显示值, 低4位报警值
  2634. std::map<time_t, int64_t> retentionforce0, retentionforce1; //保持力 时间戳, 高4位显示值, 低4位报警值
  2635. mg_per_session_data::GetMaxResistNew(map2, lstResit2, mo, mp);
  2636. if (g_bDataCompression)
  2637. {
  2638. SPDLOG_INFO("[过车判断]{}.{} 开始过车判断 size:{}", mo, mp, map0.size());
  2639. mg_per_session_data::GetPassNew(map0, mapData[0], map1, mapData[1], mo, mp, &lstPass);
  2640. SPDLOG_INFO("[过车判断]{}.{} 结束过车判断 size:{} pass:{}", mo, mp, map0.size(), lstPass.size());
  2641. //过车过滤
  2642. for (auto& item : lstPass)
  2643. {
  2644. for (auto it = lstResit2.begin(); it != lstResit2.end();)
  2645. {
  2646. if (it->tmStart * 1000 <= item.show_time && item.show_time <= (it->tmEnd + 1) * 1000)
  2647. {
  2648. it = lstResit2.erase(it);
  2649. break;
  2650. }
  2651. ++it;
  2652. }
  2653. }
  2654. }
  2655. if (lstResit2.size())
  2656. {
  2657. mg_per_session_data::GetMaxLockNew(map0, lstResit2, maxlock0, retentionforce0);
  2658. mg_per_session_data::GetMaxLockNew(map1, lstResit2, maxlock1, retentionforce1);
  2659. SPDLOG_INFO("[转换阻力][{}:{}] lock0:{} lock1:{}, resist2:{}", mo, mp, maxlock0.size(), maxlock1.size(), lstResit2.size());
  2660. }
  2661. auto pMompInfo = CMonitorObjectMng::Instance()->GetMoMpInfo(mo + "." + mp);
  2662. ASSERT(pMompInfo);
  2663. //扳动处理逻辑
  2664. if (lstResit2.size())
  2665. {
  2666. //获取报警设置
  2667. //转换阻力报警设置项
  2668. auto pConvertResistOverLimitInfo =
  2669. (CONVERT_RESIST_OVER_LIMIT*)pService->Find(mo, mp, 2, eZL_ALARMTYPE::CONVERT_LIMIT);
  2670. //定位锁闭力报警设置项
  2671. auto pFixSuobiOverInfo = (SUOBI_OVER_LIMIT_INFO*)pService->Find(mo, mp, (uint8_t)eSuoBiPosi::SB_FIX, eZL_ALARMTYPE::SUOBI_LOCK_LIMIT);
  2672. //反位
  2673. auto pInvertSuobiOverInfo = (SUOBI_OVER_LIMIT_INFO*)pService->Find(mo, mp, (uint8_t)eSuoBiPosi::SB_INVERT, eZL_ALARMTYPE::SUOBI_LOCK_LIMIT);
  2674. //定位保持力报警设置项
  2675. auto pFixConstRetensionForceWaveInfo = (RETENSION_FORCE_DROP*)pService->Find(mo, mp, (uint8_t)eSuoBiPosi::SB_FIX, eZL_ALARMTYPE::RETENSION_FORCE);
  2676. if (pFixConstRetensionForceWaveInfo == nullptr)//默认开启
  2677. {
  2678. pFixConstRetensionForceWaveInfo = new RETENSION_FORCE_DROP;
  2679. pService->Insert(mo, mp, (uint8_t)eSuoBiPosi::SB_FIX, (uint8_t)eZL_ALARMTYPE::RETENSION_FORCE, pFixConstRetensionForceWaveInfo);
  2680. }
  2681. //反位
  2682. auto pInvertConstRetensionForceWaveInfo = (RETENSION_FORCE_DROP*)pService->Find(mo, mp, (uint8_t)eSuoBiPosi::SB_INVERT, eZL_ALARMTYPE::RETENSION_FORCE);
  2683. if (pInvertConstRetensionForceWaveInfo == nullptr)//默认开启
  2684. {
  2685. pInvertConstRetensionForceWaveInfo = new RETENSION_FORCE_DROP;
  2686. pService->Insert(mo, mp, (uint8_t)eSuoBiPosi::SB_INVERT, (uint8_t)eZL_ALARMTYPE::RETENSION_FORCE, pInvertConstRetensionForceWaveInfo);
  2687. }
  2688. //获取报警设置 end
  2689. eDaoChaPosi posiLock0;
  2690. if (pMompInfo->name1.find("定位") != -1)
  2691. {
  2692. posiLock0 = eDaoChaPosi::DCP_FIX;
  2693. for (auto& it : maxlock0)
  2694. {
  2695. int show_val = (it.second >> 32);
  2696. CResistAlarmMng::InsertToDBByMove(mo, mp, it.first, 0, 0,
  2697. (int)it.second, show_val, i, (uint8_t)eDaoChaPosi::DCP_FIX, (uint8_t)ePowerName::PN_RESIST, fmt::format("锁闭力:{}", show_val));
  2698. JudgeAlarm(pService, pFixSuobiOverInfo, it.first, show_val, eSuoBiPosi::SB_FIX, mo, mp, pMompInfo->zzjno);
  2699. }
  2700. for (auto& it : retentionforce0)
  2701. {
  2702. int show_val = (int)it.second;
  2703. CResistAlarmMng::InsertToDBByMove(mo, mp, it.first, 0, 0,
  2704. (int)it.second, show_val, i, (uint8_t)eDaoChaPosi::DCP_FIX, (uint8_t)ePowerName::PN_RETENTION, fmt::format("保持力:{}", show_val));
  2705. JudgeAlarm(pService, pFixConstRetensionForceWaveInfo, it.first, show_val, eSuoBiPosi::SB_FIX, mo, mp, pMompInfo->zzjno, pMompInfo);
  2706. }
  2707. for (auto& it : maxlock1)
  2708. {
  2709. int show_val = (it.second >> 32);
  2710. CResistAlarmMng::InsertToDBByMove(mo, mp, it.first, 0, 0,
  2711. (int)it.second, show_val, i, (uint8_t)eDaoChaPosi::DCP_INVERT, (uint8_t)ePowerName::PN_RESIST, fmt::format("锁闭力:{}", show_val));
  2712. JudgeAlarm(pService, pInvertSuobiOverInfo, it.first, show_val, eSuoBiPosi::SB_INVERT, mo, mp, pMompInfo->zzjno);
  2713. }
  2714. for (auto& it : retentionforce1)
  2715. {
  2716. int show_val = (int)it.second;
  2717. CResistAlarmMng::InsertToDBByMove(mo, mp, it.first, 0, 0,
  2718. (int)it.second, show_val, i, (uint8_t)eDaoChaPosi::DCP_INVERT, (uint8_t)ePowerName::PN_RETENTION, fmt::format("保持力:{}", show_val));
  2719. JudgeAlarm(pService, pInvertConstRetensionForceWaveInfo, it.first, show_val, eSuoBiPosi::SB_FIX, mo, mp, pMompInfo->zzjno, pMompInfo);
  2720. }
  2721. }
  2722. else
  2723. {
  2724. posiLock0 = eDaoChaPosi::DCP_INVERT;
  2725. for (auto& it : maxlock0)
  2726. {
  2727. int show_val = (it.second >> 32);
  2728. CResistAlarmMng::InsertToDBByMove(mo, mp, it.first, 0, 0, (int)it.second, show_val, i,
  2729. (uint8_t)eDaoChaPosi::DCP_INVERT, (uint8_t)ePowerName::PN_RESIST, fmt::format("锁闭力:{}", show_val));
  2730. JudgeAlarm(pService, pInvertSuobiOverInfo, it.first, show_val, eSuoBiPosi::SB_INVERT, mo, mp, pMompInfo->zzjno);
  2731. }
  2732. for (auto& it : retentionforce0)
  2733. {
  2734. int show_val = (int)it.second;
  2735. CResistAlarmMng::InsertToDBByMove(mo, mp, it.first, 0, 0, (int)it.second, show_val, i,
  2736. (uint8_t)eDaoChaPosi::DCP_INVERT, (uint8_t)ePowerName::PN_RETENTION, fmt::format("保持力:{}", show_val));
  2737. JudgeAlarm(pService, pInvertConstRetensionForceWaveInfo, it.first, show_val, eSuoBiPosi::SB_INVERT, mo, mp, pMompInfo->zzjno);
  2738. }
  2739. for (auto& it : maxlock1)
  2740. {
  2741. int show_val = (it.second >> 32);
  2742. CResistAlarmMng::InsertToDBByMove(mo, mp, it.first, 0, 0, (int)it.second, show_val, i,
  2743. (uint8_t)eDaoChaPosi::DCP_FIX, (uint8_t)ePowerName::PN_RESIST, fmt::format("锁闭力:{}", show_val));
  2744. JudgeAlarm(pService, pFixSuobiOverInfo, it.first, show_val, eSuoBiPosi::SB_FIX, mo, mp, pMompInfo->zzjno);
  2745. }
  2746. for (auto& it : retentionforce1)
  2747. {
  2748. int show_val = (int)it.second;
  2749. CResistAlarmMng::InsertToDBByMove(mo, mp, it.first, 0, 0, (int)it.second, show_val, i,
  2750. (uint8_t)eDaoChaPosi::DCP_FIX, (uint8_t)ePowerName::PN_RETENTION, fmt::format("保持力:{}", show_val));
  2751. JudgeAlarm(pService, pFixConstRetensionForceWaveInfo, it.first, show_val, eSuoBiPosi::SB_FIX, mo, mp, pMompInfo->zzjno, pMompInfo);
  2752. }
  2753. }
  2754. auto mg_315_server = g_p315ClientManager->GetTcpClient();
  2755. //SPDLOG_INFO("扳动判断数据{}.{} 转换力", mo, mp);
  2756. auto eqpno = CMonitorObjectMng::Instance()->GetZZJNO(mo + "." + mp);
  2757. for (auto& it : lstResit2)
  2758. {
  2759. //auto it_start0 = mapData[0]->find(it.tmStart * 1000);
  2760. //auto it_end0 = mapData[0]->find(it.tmEnd * 1000);
  2761. //auto it_start1 = mapData[1]->find(it.tmStart * 1000);
  2762. //auto it_end1 = mapData[1]->find(it.tmEnd * 1000);
  2763. //auto it_start2 = mapData[2]->find(it.tmStart * 1000);
  2764. //auto it_end2 = mapData[2]->find(it.tmEnd * 1000);
  2765. eDaoChaPosi posi = eDaoChaPosi::DCP_UNKNOWN;
  2766. if (it.bUpOrDown == eUpOrDownInfo::UOD_UP)
  2767. {
  2768. if (pMompInfo->in_name.find("定扳反") != -1)
  2769. {
  2770. CResistAlarmMng::InsertToDBByMove(mo, mp, it.time, it.tmStart, it.tmEnd, it.val, it.fluctuation_val, i,
  2771. (uint8_t)eDaoChaPosi::DCP_FIX2INVERT, (uint8_t)ePowerName::PN_MOVE, fmt::format("{}:{}", pMompInfo->in_name, it.fluctuation_val));
  2772. posi = eDaoChaPosi::DCP_FIX2INVERT;
  2773. CMonitorObjectMng::Instance()->UpdateZZJEPOS(mo + "." + mp, DAOCHA_POSITION::MP_INVERT);
  2774. JudgeAlarm(pService, pConvertResistOverLimitInfo, it.time, it.fluctuation_val, posi, it.bUpOrDown, mo, mp, pMompInfo->zzjno);
  2775. }
  2776. else
  2777. {
  2778. CResistAlarmMng::InsertToDBByMove(mo, mp, it.time, it.tmStart, it.tmEnd, it.val, it.fluctuation_val, i,
  2779. (uint8_t)eDaoChaPosi::DCP_INVERT2FIX, (uint8_t)ePowerName::PN_MOVE, fmt::format("{}:{}", pMompInfo->in_name, it.fluctuation_val));
  2780. posi = eDaoChaPosi::DCP_INVERT2FIX;
  2781. CMonitorObjectMng::Instance()->UpdateZZJEPOS(mo + "." + mp, DAOCHA_POSITION::MP_FIX);
  2782. JudgeAlarm(pService, pConvertResistOverLimitInfo, it.time, it.fluctuation_val, posi, it.bUpOrDown, mo, mp, pMompInfo->zzjno);
  2783. }
  2784. }
  2785. else
  2786. {
  2787. if (pMompInfo->out_name.find("定扳反") != -1)
  2788. {
  2789. CResistAlarmMng::InsertToDBByMove(mo, mp, it.time, it.tmStart, it.tmEnd, it.val, it.fluctuation_val, i,
  2790. (uint8_t)eDaoChaPosi::DCP_FIX2INVERT, (uint8_t)ePowerName::PN_MOVE, fmt::format("{}:{}", pMompInfo->out_name, it.fluctuation_val));
  2791. posi = eDaoChaPosi::DCP_FIX2INVERT;
  2792. CMonitorObjectMng::Instance()->UpdateZZJEPOS(mo + "." + mp, DAOCHA_POSITION::MP_INVERT);
  2793. JudgeAlarm(pService, pConvertResistOverLimitInfo, it.time, it.fluctuation_val, posi, it.bUpOrDown, mo, mp, pMompInfo->zzjno);
  2794. }
  2795. else
  2796. {
  2797. CResistAlarmMng::InsertToDBByMove(mo, mp, it.time, it.tmStart, it.tmEnd, it.val, it.fluctuation_val, i,
  2798. (uint8_t)eDaoChaPosi::DCP_INVERT2FIX, (uint8_t)ePowerName::PN_MOVE, fmt::format("{}:{}", pMompInfo->out_name, it.fluctuation_val));
  2799. posi = eDaoChaPosi::DCP_INVERT2FIX;
  2800. CMonitorObjectMng::Instance()->UpdateZZJEPOS(mo + "." + mp, DAOCHA_POSITION::MP_FIX);
  2801. JudgeAlarm(pService, pConvertResistOverLimitInfo, it.time, it.fluctuation_val, posi, it.bUpOrDown, mo, mp, pMompInfo->zzjno);
  2802. }
  2803. }
  2804. SPDLOG_INFO("扳动判断数据{}.{} 315 上送", mo, mp);
  2805. if(mapData[0])
  2806. mg_315_server->SendCurveData(pMompInfo->zzjno, posi, 1, true, mapData, it.tmStart, it.tmEnd);
  2807. SPDLOG_INFO("扳动判断数据结束{}.{} 315 上送", mo, mp);
  2808. }
  2809. }
  2810. //过车处理逻辑
  2811. if (lstPass.size())
  2812. {
  2813. //TODO
  2814. //SPDLOG_INFO("扳动判断数据{}.{} 315 上送", mo, mp);
  2815. //mg_315_server->SendCurveData(pMompInfo->zzjno, posi, 1, true, mapData, it.tmStart, it.tmEnd);
  2816. //SPDLOG_INFO("扳动判断数据结束{}.{} 315 上送", mo, mp);
  2817. }
  2818. delete mapData[0];
  2819. delete mapData[1];
  2820. delete mapData[2];
  2821. }
  2822. }
  2823. } while (pService->m_bWork);
  2824. if (bLock)
  2825. {
  2826. g_lockSync.Unlock();
  2827. bLock = false;
  2828. }
  2829. }
  2830. BOOL CResistAlarmMng::LoadAlarmSet()
  2831. {
  2832. CString sql = "SELECT [mo],[mp],[no],[type],[conf],[time] FROM [rm_alarm_set]";
  2833. COdbcStatement stmt;
  2834. if (!CDBConnectPool::Instance()->DBQuery(stmt, sql))
  2835. {
  2836. CSimpleLog::Error("执行语句失败" + sql);
  2837. return FALSE;
  2838. }
  2839. for (auto& it : m_alarm_set)
  2840. {
  2841. //delete it.second; //暂时不释放
  2842. it.second = nullptr;
  2843. }
  2844. m_alarm_set.clear();
  2845. char mo[51], mp[51], conf[501];
  2846. uint8_t no, type;
  2847. __time64_t time;
  2848. int nCol = 1;
  2849. stmt.BindCharCol(nCol++, mo, sizeof(mo));
  2850. stmt.BindCharCol(nCol++, mp, sizeof(mp));
  2851. stmt.BindTinyIntCol(nCol++, &no);
  2852. stmt.BindTinyIntCol(nCol++, &type);
  2853. stmt.BindCharCol(nCol++, conf, sizeof(conf));
  2854. stmt.BindBigIntCol(nCol++, &time);
  2855. do
  2856. {
  2857. if (stmt.FetchNext() != 0) break;
  2858. using namespace rapidjson;
  2859. int len = strlen(conf);
  2860. if (len == 0) continue;
  2861. Document doc;
  2862. if (doc.Parse(conf, len).HasParseError())
  2863. {
  2864. CSimpleLog::Error(CString("解析数据出错") + conf);
  2865. continue;
  2866. }
  2867. const auto eType = (eZL_ALARMTYPE)type;
  2868. if (eType == eZL_ALARMTYPE::MAX_OVER_LIMIT)
  2869. {
  2870. if (doc.IsArray() == false)
  2871. {
  2872. CSimpleLog::Error(CString("conf是非数组") + conf);
  2873. continue;
  2874. }
  2875. SizeType n = doc.Size();
  2876. bool enable = false;
  2877. int alarm_high_limit = INT_MAX;
  2878. int warn_high_limit = INT_MAX;
  2879. int f_alarm_high_limit = INT_MAX;
  2880. int f_warn_high_limit = INT_MAX;
  2881. for (SizeType i = 0; i < n; i++)
  2882. {
  2883. if (doc[i].HasMember("name") == false || doc[i].HasMember("val") == false || doc[i]["name"].IsString() == false || doc[i]["val"].IsString() == false)
  2884. {
  2885. assert(false);
  2886. continue;
  2887. }
  2888. string key = doc[i]["name"].GetString();
  2889. string value = doc[i]["val"].GetString();
  2890. if (key.compare("enable") == 0)
  2891. {
  2892. enable = (value.compare("true") == 0);
  2893. }
  2894. else if (key.compare("lock_alarm_high_limit") == 0)
  2895. {
  2896. alarm_high_limit = atoi(value.c_str());
  2897. }
  2898. else if (key.compare("lock_warn_high_limit") == 0)
  2899. {
  2900. warn_high_limit = atoi(value.c_str());
  2901. }
  2902. else if (key.compare("d_alarm_high_limit") == 0)
  2903. {
  2904. alarm_high_limit = atoi(value.c_str());
  2905. }
  2906. else if (key.compare("d_warn_high_limit") == 0)
  2907. {
  2908. warn_high_limit = atoi(value.c_str());
  2909. }
  2910. else if (key.compare("f_alarm_high_limit") == 0)
  2911. {
  2912. f_alarm_high_limit = atoi(value.c_str());
  2913. }
  2914. else if (key.compare("f_warn_high_limit") == 0)
  2915. {
  2916. f_warn_high_limit = atoi(value.c_str());
  2917. }
  2918. else if (key.compare("keep_alarm_high_limit") == 0)
  2919. {
  2920. f_alarm_high_limit = atoi(value.c_str());
  2921. }
  2922. else if (key.compare("keep_warn_high_limit") == 0)
  2923. {
  2924. f_warn_high_limit = atoi(value.c_str());
  2925. }
  2926. else
  2927. {
  2928. assert(0);
  2929. }
  2930. }
  2931. if (alarm_high_limit == INT_MAX || warn_high_limit == INT_MAX) continue;
  2932. auto pInfo = new MAX_OVER_LIMIT_INFO;
  2933. pInfo->enable = enable;
  2934. pInfo->no = no;
  2935. pInfo->type = eType;
  2936. pInfo->alarm_high_limit = alarm_high_limit;
  2937. pInfo->warn_high_limit = warn_high_limit;
  2938. pInfo->f_alarm_high_limit = f_alarm_high_limit;
  2939. pInfo->f_warn_high_limit = f_warn_high_limit;
  2940. pInfo->tmLastCheckTime = time;
  2941. stringstream ss;
  2942. ss << mo << '.' << mp << '.' << to_string(no) << '.' << to_string(type);
  2943. m_alarm_set[ss.str()] = pInfo;
  2944. }
  2945. else if (eType == eZL_ALARMTYPE::FRICTION_OVER_LIMIT)
  2946. {
  2947. if (doc.IsArray() == false)
  2948. {
  2949. CSimpleLog::Error(CString("conf是非数组") + conf);
  2950. continue;
  2951. }
  2952. SizeType n = doc.Size();
  2953. bool enable = false;
  2954. int up_alarm_low_limit = INT_MAX;
  2955. int up_warn_low_limit = INT_MAX;
  2956. int dw_alarm_high_limit = INT_MIN;
  2957. int dw_warn_high_limit = INT_MIN;
  2958. for (SizeType i = 0; i < n; i++)
  2959. {
  2960. if (doc[i].HasMember("name") == false || doc[i].HasMember("val") == false || doc[i]["name"].IsString() == false || doc[i]["val"].IsString() == false)
  2961. {
  2962. assert(false);
  2963. continue;
  2964. }
  2965. string key = doc[i]["name"].GetString();
  2966. string value = doc[i]["val"].GetString();
  2967. if (key.compare("enable") == 0)
  2968. {
  2969. enable = (value.compare("true") == 0);
  2970. }
  2971. else if (key.compare("up_alarm_low_limit") == 0)
  2972. {
  2973. up_alarm_low_limit = atoi(value.c_str());
  2974. }
  2975. else if (key.compare("up_warn_low_limit") == 0)
  2976. {
  2977. up_warn_low_limit = atoi(value.c_str());
  2978. }
  2979. else if (key.compare("dw_alarm_high_limit") == 0)
  2980. {
  2981. dw_alarm_high_limit = atoi(value.c_str());
  2982. }
  2983. else if (key.compare("dw_warn_high_limit") == 0)
  2984. {
  2985. dw_warn_high_limit = atoi(value.c_str());
  2986. }
  2987. else
  2988. {
  2989. assert(0);
  2990. }
  2991. }
  2992. auto pInfo = new FRICTION_OVER_LIMIT_INFO;
  2993. pInfo->enable = enable;
  2994. pInfo->no = no;
  2995. pInfo->type = eType;
  2996. pInfo->up_alarm_low_limit = up_alarm_low_limit;
  2997. pInfo->up_warn_low_limit = up_warn_low_limit;
  2998. pInfo->dw_alarm_high_limit = dw_alarm_high_limit;
  2999. pInfo->dw_warn_high_limit = dw_warn_high_limit;
  3000. pInfo->tmLastCheckTime = time;
  3001. stringstream ss;
  3002. ss << mo << '.' << mp << '.' << to_string(no) << '.' << to_string(type);
  3003. m_alarm_set[ss.str()] = pInfo;
  3004. }
  3005. else if (eType == eZL_ALARMTYPE::SUOBI_LOCK_LIMIT)
  3006. {
  3007. auto doc = yyjson_read(conf, len, 0);
  3008. if (doc == nullptr) continue;
  3009. auto root = yyjson_doc_get_root(doc);
  3010. auto pInfo = new SUOBI_OVER_LIMIT_INFO;
  3011. pInfo->type = eZL_ALARMTYPE::SUOBI_LOCK_LIMIT;
  3012. pInfo->no = no;
  3013. pInfo->enable = yyjson_get_bool(yyjson_obj_get(root, "enable"));
  3014. pInfo->alarm_low_limit = yyjson_get_int(yyjson_obj_get(root, "alarm_low_limit"));
  3015. pInfo->warn_low_limit = yyjson_get_int(yyjson_obj_get(root, "warn_low_limit"));
  3016. pInfo->alarm_high_limit = yyjson_get_int(yyjson_obj_get(root, "alarm_high_limit"));
  3017. pInfo->warn_high_limit = yyjson_get_int(yyjson_obj_get(root, "warn_high_limit"));
  3018. stringstream ss;
  3019. ss << mo << '.' << mp << '.' << to_string(no) << '.' << to_string(type);
  3020. m_alarm_set[ss.str()] = pInfo;
  3021. yyjson_doc_free(doc);
  3022. }
  3023. else if (eType == eZL_ALARMTYPE::CONVERT_LIMIT)
  3024. {
  3025. auto doc = yyjson_read(conf, len, 0);
  3026. if (doc == nullptr) continue;
  3027. auto root = yyjson_doc_get_root(doc);
  3028. auto pInfo = new CONVERT_RESIST_OVER_LIMIT;
  3029. pInfo->type = eZL_ALARMTYPE::CONVERT_LIMIT;
  3030. pInfo->no = 2;
  3031. pInfo->enable = yyjson_get_bool(yyjson_obj_get(root, "enable"));
  3032. pInfo->dw_alarm_low_limit = yyjson_get_int(yyjson_obj_get(root, "dw_alarm_low_limit"));
  3033. pInfo->dw_warn_low_limit = yyjson_get_int(yyjson_obj_get(root, "dw_warn_low_limit"));
  3034. pInfo->up_alarm_high_limit = yyjson_get_int(yyjson_obj_get(root, "up_alarm_high_limit"));
  3035. pInfo->up_warn_high_limit = yyjson_get_int(yyjson_obj_get(root, "up_warn_high_limit"));
  3036. stringstream ss;
  3037. ss << mo << '.' << mp << '.' << to_string(no) << '.' << to_string(type);
  3038. m_alarm_set[ss.str()] = pInfo;
  3039. yyjson_doc_free(doc);
  3040. }
  3041. else if (eType == eZL_ALARMTYPE::RETENSION_FORCE)
  3042. {
  3043. auto doc = yyjson_read(conf, len, 0);
  3044. if (doc == nullptr) continue;
  3045. auto root = yyjson_doc_get_root(doc);
  3046. auto pInfo = new RETENSION_FORCE_DROP;
  3047. pInfo->type = eZL_ALARMTYPE::RETENSION_FORCE;
  3048. pInfo->no = no;
  3049. pInfo->enable = yyjson_get_bool(yyjson_obj_get(root, "enable"));
  3050. pInfo->dw_alarm_low_drop = yyjson_get_int(yyjson_obj_get(root, "dw_alarm_low_drop"));
  3051. stringstream ss;
  3052. ss << mo << '.' << mp << '.' << to_string(no) << '.' << to_string(type);
  3053. m_alarm_set[ss.str()] = pInfo;
  3054. yyjson_doc_free(doc);
  3055. }
  3056. else
  3057. {
  3058. ASSERT(0);
  3059. }
  3060. } while (true);
  3061. CSimpleLog::Info(fmt::format("一共加载到{}条报警规则:{}", m_alarm_set.size(), sql).c_str());
  3062. return TRUE;
  3063. }
  3064. BOOL CResistAlarmMng::LoadAlarmID()
  3065. {
  3066. CString sql = "SELECT TOP 1 ID FROM RM_ALARM ORDER BY ID DESC";
  3067. COdbcStatement stmt;
  3068. if (!CDBConnectPool::Instance()->DBQuery(stmt, sql))
  3069. {
  3070. CSimpleLog::Error("执行语句失败" + sql);
  3071. return FALSE;
  3072. }
  3073. int alarm_id;
  3074. int nCol = 1;
  3075. stmt.BindIntCol(nCol++, &alarm_id);
  3076. do
  3077. {
  3078. if (stmt.FetchNext() != 0) break;
  3079. m_nAlarmID = alarm_id;
  3080. } while (false);
  3081. stmt.Close();
  3082. return TRUE;
  3083. }
  3084. //加载未受理和未恢复的报警
  3085. BOOL CResistAlarmMng::LoadUnAck()
  3086. {
  3087. //只加载31天内的报警
  3088. CString sql = " SELECT [ID],a.[mo],a.[mp],[no],[type],[occur_time],[level],[desc],[suggest],[val],[event_id],[rel_id],[posi],[loworhigh],[referval],[recovery_time],[sunroof],zzjno FROM [rm_alarm] as A \
  3089. LEFT JOIN(SELECT mo, mp, zzjno FROM rm_map) AS B ON a.mo = b.mo and a.mp = b.mp \
  3090. WHERE DATEADD(dd, -31, getdate()) < occur_time and (ack_result = 0 OR [recovery_time] < '2000') \
  3091. ORDER BY ID; ";
  3092. COdbcStatement stmt;
  3093. if (!CDBConnectPool::Instance()->DBQuery(stmt, sql))
  3094. {
  3095. ASSERT(FALSE);
  3096. CSimpleLog::Error("执行语句失败" + sql);
  3097. return FALSE;
  3098. }
  3099. char mo[51], mp[51], desc[200], suggest[200], event_id[37] = { 0 }, rel_id[37] = { 0 };
  3100. uint8_t no, type, level, posi, loworhigh, sunroof;
  3101. int val;
  3102. int id;
  3103. int zzjno;
  3104. TIMESTAMP_STRUCT ts,tsRecovery;
  3105. int referval;
  3106. int nCol = 1;
  3107. stmt.BindIntCol(nCol++, &id);
  3108. stmt.BindCharCol(nCol++, mo, sizeof(mo));
  3109. stmt.BindCharCol(nCol++, mp, sizeof(mp));
  3110. stmt.BindTinyIntCol(nCol++, &no);
  3111. stmt.BindTinyIntCol(nCol++, &type);
  3112. stmt.BindTimeStampCol(nCol++, &ts);
  3113. stmt.BindTinyIntCol(nCol++, &level);
  3114. stmt.BindCharCol(nCol++, desc, sizeof(desc));
  3115. stmt.BindCharCol(nCol++, suggest, sizeof(suggest));
  3116. stmt.BindIntCol(nCol++, &val);
  3117. stmt.BindCharCol(nCol++, event_id, 36);
  3118. stmt.BindCharCol(nCol++, rel_id, 36);
  3119. stmt.BindTinyIntCol(nCol++, &posi);
  3120. stmt.BindTinyIntCol(nCol++, &loworhigh);
  3121. stmt.BindIntCol(nCol++, &referval);
  3122. stmt.BindTimeStampCol(nCol++, &tsRecovery);
  3123. stmt.BindTinyIntCol(nCol++, &sunroof);
  3124. stmt.BindIntCol(nCol++, &zzjno);
  3125. do
  3126. {
  3127. if (stmt.FetchNext() != 0) break;
  3128. ALARM_INFO *pAlarm = new ALARM_INFO;
  3129. pAlarm->event_id = event_id;
  3130. if (rel_id[0] != 0x20)
  3131. pAlarm->rel_id = rel_id;
  3132. pAlarm->id = id;
  3133. pAlarm->mo = mo;
  3134. pAlarm->mp = mp;
  3135. pAlarm->no = no;
  3136. pAlarm->type = (eZL_ALARMTYPE)type;
  3137. ConvertData(ts, pAlarm->time);
  3138. pAlarm->val = val;
  3139. pAlarm->level = level;
  3140. pAlarm->desc = desc;
  3141. pAlarm->suggest = suggest;
  3142. pAlarm->posi = (eDaoChaPosi)posi;
  3143. pAlarm->loworhigh = (eLowHigh)loworhigh;
  3144. pAlarm->refer_val = referval;
  3145. ConvertData(tsRecovery, pAlarm->recoveryTime);
  3146. pAlarm->sunroof = sunroof;
  3147. pAlarm->zzjno = zzjno;
  3148. m_lstUnConfirmAlarm.push_back(pAlarm);
  3149. } while (true);
  3150. stmt.Close();
  3151. CSimpleLog::Info(("一共加载(" + to_string(m_lstUnConfirmAlarm.size()) + ")条未受理和未恢复的报警").c_str());
  3152. return TRUE;
  3153. }