module_call.cpp 26 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847
  1. #include "module_call.h"
  2. #include <boost/bind.hpp>
  3. #include <boost/algorithm/string.hpp>
  4. #include <vector>
  5. #include <set>
  6. #include <string>
  7. #include <chrono>
  8. #include <algorithm>
  9. #include <area.h>
  10. #include "common_tool.h"
  11. #include "tool_time.h"
  12. #include "ant.h"
  13. #include "websocket/constdef.h"
  14. #include "websocket/wsClientMgr.h"
  15. #include "crc.h"
  16. #include "mine.h"
  17. #include "event.h"
  18. uint64_t call_card::to_id64()
  19. {
  20. return tool_other::type_id_to_u64(cardtype, cardid);
  21. }
  22. std::string call_card::to_id64_str()
  23. {
  24. return tool_other::type_id_to_str(cardtype, cardid);
  25. }
  26. bool call_card::is_timeout()
  27. {
  28. uint32_t seconds = tool_time::elapse_seconds(call_time);
  29. return (seconds >= call_time_out*60);
  30. }
  31. bool module_call::call_user::is_timeout()
  32. {
  33. uint32_t seconds = tool_time::elapse_seconds(call_time);
  34. return (seconds >= call_time_out*60);
  35. }
  36. void module_call::to_node_element(rapidjson::Value& out_elemet,
  37. const std::shared_ptr<call_card> card_ptr,
  38. rapidjson::Document::AllocatorType& allocator,bool f)
  39. {
  40. tool_json::push_back(out_elemet, card_ptr->user_name, allocator);
  41. tool_json::push_back(out_elemet, card_ptr->to_id64_str(), allocator);
  42. uint32_t sid= card_ptr->stationid;
  43. if(f)sid=0;
  44. out_elemet.PushBack(sid, allocator);
  45. out_elemet.PushBack(tool_time::to_ms(card_ptr->call_time), allocator);
  46. out_elemet.PushBack(card_ptr->call_type_id, allocator);
  47. out_elemet.PushBack(card_ptr->call_state, allocator);
  48. }
  49. void module_call::run()
  50. {
  51. std::vector<call_card_ptr> arr;
  52. get_all_call_cards(arr);
  53. if(arr.empty())
  54. {
  55. return ;
  56. }
  57. //转发给web
  58. //swsClientMgr.send(JSON_CMD_VALUE_PUSH, to_call_card_list_json(arr));
  59. std::string ac = get_json_anti_collision();
  60. if(!ac.empty()){
  61. log_info("send2web: %s", ac.c_str());
  62. swsClientMgr.send(JSON_CMD_VALUE_PUSH, ac);
  63. }
  64. //将呼叫命令发送给标识卡终端
  65. call_site_map site_map;
  66. get_site_map(arr, site_map);
  67. if(!site_map.empty())
  68. {
  69. send_to_sites(site_map);
  70. }
  71. }
  72. ///发呼叫的报文数据给分站
  73. void module_call::send_to_sites(call_site_map& site_map)
  74. {
  75. auto iter_site = site_map.begin();
  76. for(; iter_site != site_map.end(); ++iter_site)
  77. {
  78. if(iter_site->second->cards.size() > 254)
  79. {
  80. log_error("呼叫:分站下卡数量过多:分站id=%d, 卡数=%d",
  81. iter_site->first, iter_site->second->cards.size());
  82. continue;
  83. }
  84. std::vector<char> arr;
  85. int16_t cmd = 0x77ab;
  86. arr.push_back(cmd>>8 & 0xff);
  87. arr.push_back(cmd & 0xff);
  88. //分站地址 4字节,支持大小分站呼叫协议;added by zhuyf 2018/06/04
  89. //大小分站的全员呼叫,分站id要求4字节全为FF
  90. uint32_t anchor_id = (iter_site->second->is_call_all()? 0xffffffff : static_cast<uint32_t>(iter_site->first));
  91. memcpy_uint32(arr,anchor_id);
  92. //卡数
  93. arr.push_back(static_cast<char>(iter_site->second->cards.size()));
  94. //呼叫类型
  95. arr.push_back(static_cast<char>(iter_site->second->call_type_id));
  96. if(!iter_site->second->is_call_all())
  97. {
  98. //循环添加卡信息
  99. auto it_card = iter_site->second->cards.begin();
  100. for(; it_card != iter_site->second->cards.end(); ++it_card)
  101. {
  102. //呼叫类型
  103. arr.push_back(static_cast<char>(it_card->second->call_level_id));
  104. //卡类型
  105. arr.push_back(static_cast<char>(it_card->second->cardtype));
  106. //卡ID长度
  107. int id_len = 4;
  108. arr.push_back(static_cast<char>(id_len));
  109. //卡ID
  110. memcpy_uint32(arr, it_card->second->cardid);
  111. }
  112. }
  113. auto sit_ptr = sit_list::instance()->get(iter_site->first);
  114. if(!sit_ptr || !sit_ptr->m_clt)
  115. {
  116. log_error("在全局分站列表中找不到分站 或者sit_ptr->m_clt==null:分站id=%d", iter_site->first);
  117. continue;
  118. }
  119. int16_t len=arr.size()+2;
  120. int16_t crc=do_crc_1(arr,0);
  121. auto it= arr.begin();arr.insert(it,len&0xff);
  122. it= arr.begin();arr.insert(it,len>>8&0xff);
  123. arr.push_back(crc>>8&0xff);arr.push_back(crc&0xff);
  124. std::vector<char> arr2 = arr;
  125. sit_ptr->m_clt->send(std::move(arr));
  126. print_test(std::move(arr2), iter_site->first);
  127. }
  128. }
  129. void module_call::print_test(std::vector<char>&& arr, int siteid)
  130. {
  131. std::string str("呼叫发送的数据帧,分站号=");
  132. str.append(std::to_string(siteid));
  133. str.append(":");
  134. char a[4]={0};
  135. for(std::vector<char>::size_type i=0; i<arr.size(); i++)
  136. {
  137. sprintf(a, "%02X ", static_cast<unsigned char>(arr[i]));
  138. str.append(std::string(a));
  139. }
  140. log_info("%s", str.c_str());
  141. }
  142. void module_call::rev_from_card_resp(std::shared_ptr<card_location_base> card_ptr)
  143. {
  144. card_ptr->get_mine_tool()->m_status_call = CALL_SUCCESSED;
  145. }
  146. /*
  147. web发给采集:发起呼叫
  148. {
  149. cmd: 'call_card_req',
  150. data: {
  151. call_type_id: this.cal_type, // 全员呼叫:0 定员呼叫:1
  152. call_time_out: this.choosedTime, // 呼叫时长
  153. call_level_id: this.call_type_id, // 呼叫类型 一般呼叫:1 紧急呼叫:2
  154. user_name: xdata.userName, // 呼叫人
  155. call_time: time, // 呼叫时间戳
  156. stations: [{ stationid: 0 }], // 分站 0为全员
  157. cards: cards // 人员 0为全员 //旧代码为map格式 [{ stationid: 0 }]
  158. }
  159. }
  160. */
  161. void module_call::accept_call(sio::message::ptr const& data)
  162. {
  163. int call_type = -1, call_level = 1, call_time_interval=-1;
  164. uint32_t call_time_out = 0;
  165. std::string user_name;
  166. if(!tool_map::try_get_value(call_type, JSON_KEY_CALL_CARD_CALL_TYPE, data)
  167. || !tool_map::try_get_value(call_level, JSON_KEY_CALL_CARD_CALL_LEVEL, data)
  168. || !tool_map::try_get_value(call_time_out, JSON_KEY_CALL_CARD_CALL_TIME_OUT, data)
  169. || !tool_map::try_get_value(user_name, JSON_KEY_CALL_CARD_USER_NAME, data)
  170. || !tool_map::try_get_value(call_time_interval, JSON_KEY_CALL_CARD_CALL_TIME, data)
  171. )
  172. {
  173. log_error("发起呼叫: 收到的json不对,解析不出call_type = -1, call_level = -1, \
  174. call_time_out=-1, call_time_interval=-1,user_name");
  175. return;
  176. }
  177. if(user_name.empty())
  178. {
  179. log_error("发起呼叫: user_name用户名为空");
  180. return;
  181. }
  182. std::vector<call_card_ptr> result_arr;//结果集
  183. call_user_ptr user_ptr;
  184. if(CCT_CALL_ALL==call_type)// 全员
  185. {
  186. user_ptr=call_user_ptr(new call_user());
  187. user_ptr->call_time=std::chrono::system_clock::now();
  188. user_ptr->call_time_out=call_time_out;
  189. user_ptr->call_time_interval = call_time_interval;
  190. user_ptr->call_type_id=call_type;
  191. user_ptr->call_level_id=call_level;
  192. user_ptr->user_name=user_name;
  193. call_card_ptr temp(new call_card());
  194. copy(user_ptr, temp);
  195. temp->cardid = CCT_CALL_ALL; //全员卡
  196. temp->call_state = CALL_ING;
  197. result_arr.push_back(temp);//增加到结果集中
  198. //添加全员卡
  199. user_ptr->cards.insert(std::make_pair(temp->cardid, temp));
  200. //更新用户信息
  201. std::lock_guard<std::mutex> lock(_mutex);
  202. _map[user_name]=user_ptr;
  203. }
  204. else// 定员: 如果有全员,定员不生效
  205. {
  206. std::lock_guard<std::mutex> lock(_mutex);
  207. auto user_map_ptr = _map.find(user_name);
  208. if(_map.end() == user_map_ptr)//没有这个用户就新建并加入
  209. {
  210. user_ptr=call_user_ptr(new call_user());
  211. _map[user_name]=user_ptr;
  212. }
  213. else//有这个用户
  214. {
  215. user_ptr=user_map_ptr->second;
  216. // 如果有全员,定员不生效
  217. auto card_temp = user_ptr->cards.find(CCT_CALL_ALL);
  218. if(user_ptr->cards.end() != card_temp)
  219. {
  220. return;
  221. }
  222. }
  223. //更新用户信息
  224. user_ptr->call_time=std::chrono::system_clock::now();
  225. user_ptr->call_time_out=call_time_out;
  226. user_ptr->call_time_interval = call_time_interval;
  227. user_ptr->call_type_id=call_type;
  228. user_ptr->call_level_id=call_level;
  229. user_ptr->user_name=user_name;
  230. std::vector<sio::message::ptr> card_vec;
  231. if(!tool_map::try_get_value(card_vec, JSON_KEY_CALL_CARD_CARDS, data) || card_vec.size() == 0)
  232. {
  233. log_error("发起呼叫,web发来的数据JSON_KEY_CALL_CARD_CARDS字段为空 或者不是数组");
  234. return;
  235. }
  236. add_cards_to_user(card_vec, user_ptr);
  237. for(auto& it:user_ptr->cards)//添加到结果集中
  238. {
  239. result_arr.push_back(it.second);
  240. }
  241. }
  242. //组装json发送给web
  243. if(!result_arr.empty())
  244. {
  245. response_accept_call(result_arr);
  246. }
  247. }
  248. //采集回复web发起呼叫
  249. //{
  250. // 'cmd': 'call_card_resp',
  251. // 'data': [
  252. // ["fjb", // 发起呼叫人
  253. // "0010000000001", // 呼叫的卡号
  254. // 103, // 分站号
  255. // 23431, // 发起呼叫时间
  256. // 0, //
  257. // 0 // 正在呼叫:2/结束呼叫/
  258. // ]
  259. // ]
  260. //}
  261. void module_call::response_accept_call(std::vector<call_card_ptr> cards)
  262. {
  263. rapidjson::Document doc(rapidjson::kObjectType);
  264. auto& allocator = doc.GetAllocator();
  265. rapidjson::Value node_cards(rapidjson::kArrayType);
  266. //组装json发送给web
  267. to_node_array(node_cards, cards, allocator,false);
  268. doc.AddMember(JSON_ROOT_KEY_CMD,JSON_CMD_VALUE_CALL_CARD_RESPONSE, allocator);
  269. doc.AddMember(JSON_ROOT_KEY_VERSION, INTERFACE_VERSION, allocator);
  270. doc.AddMember(JSON_ROOT_KEY_DATA, node_cards, allocator);
  271. swsClientMgr.send(JSON_CMD_VALUE_PUSH, tool_json::doc_to_json(doc));
  272. }
  273. //web发给采集:取消呼叫
  274. //{
  275. // cmd: 'call_card_cancel_req',
  276. // data: {
  277. // call_type_id: type, // 全员/定员
  278. // user_name: xdata.userName, // 取消人
  279. // call_time: callTime, // 时间戳
  280. // stations: this.stationid, // 分站
  281. // cards: this.cards // 取消呼叫卡 //旧代码为map格式 [{ stationid: 0 }]
  282. // }
  283. //}
  284. void module_call::accept_cancel(sio::message::ptr const& node_data)
  285. {
  286. int call_type = -1;//, call_level = -1;
  287. std::string user_name;
  288. int64_t call_time;
  289. if(!tool_map::try_get_value(call_type, JSON_KEY_CALL_CARD_CALL_TYPE, node_data)
  290. || !tool_map::try_get_value(call_time, JSON_KEY_CALL_CARD_CALL_TIME, node_data)
  291. || !tool_map::try_get_value(user_name, JSON_KEY_CALL_CARD_USER_NAME, node_data)
  292. )
  293. {
  294. log_error("取消呼叫: 收到的json不对,解析不出int call_type = -1, call_level = -1,user_name");
  295. return;
  296. }
  297. if(user_name.empty())
  298. {
  299. log_error("取消呼叫: user_name用户名为空");
  300. return;
  301. }
  302. call_user_ptr result_user_ptr(new call_user());
  303. result_user_ptr->call_type_id=call_type;
  304. // result_user_ptr->call_level_id=call_level;
  305. result_user_ptr->call_time = std::chrono::system_clock::time_point(std::chrono::milliseconds(call_time));
  306. result_user_ptr->user_name=user_name;
  307. //取消呼叫有两种:
  308. //取消全员呼叫
  309. //对指定卡取消呼叫
  310. if(CCT_CALL_ALL==call_type)// 取消全员呼叫
  311. {
  312. //删除用户
  313. std::lock_guard<std::mutex> lock(_mutex);
  314. _map.erase(user_name);
  315. }
  316. else// 取消定员呼叫
  317. {
  318. std::vector<sio::message::ptr> card_vec;
  319. if(!tool_map::try_get_value(card_vec, JSON_KEY_CALL_CARD_CARDS, node_data) || card_vec.size() == 0)
  320. {
  321. log_error("取消呼叫: web发来的数据cards字段为空 或者不是数组");
  322. return;
  323. }
  324. std::lock_guard<std::mutex> lock(_mutex);
  325. call_user_ptr user_ptr(new call_user());
  326. auto it_map = _map.find(user_name);
  327. if(_map.end()!=it_map)//呼叫用户
  328. {
  329. user_ptr=it_map->second;
  330. }
  331. std::vector<sio::message::ptr>::const_iterator it_card = card_vec.begin();
  332. std::string s_card_id;
  333. for(; it_card != card_vec.end(); ++it_card)
  334. {
  335. if(!tool_map::try_get_value(s_card_id, JSON_KEY_CALL_CARD_CARD_ID, (*it_card)))
  336. {
  337. log_error("取消呼叫: web发来的数据 card_id 格式不对");
  338. continue;
  339. }
  340. uint32_t id = tool_other::id64_to_id(s_card_id);
  341. int type= tool_other::id64_to_type(s_card_id);
  342. log_info("取消呼叫: cardid=%d, cardtype=%d", id, type);
  343. call_card_ptr card_ptr(new call_card());
  344. card_ptr->cardid = id;
  345. card_ptr->cardtype = type;
  346. result_user_ptr->cards[card_ptr->cardid]=card_ptr;//增加到结果集中
  347. user_ptr->cards.erase(card_ptr->cardid);//删除这个卡
  348. }
  349. // 没有呼叫信息,删除该用户记录
  350. if(user_ptr->cards.empty())
  351. {
  352. _map.erase(user_name);
  353. }
  354. }
  355. //如果要取消呼叫的用户下没有卡,建一张CCT_CALL_ALL卡返回
  356. if(result_user_ptr->cards.empty())
  357. {
  358. // result_user_ptr->call_level_id=call_level;
  359. //result_user_ptr->call_time=std::chrono::system_clock::now();
  360. result_user_ptr->call_time_out=0;
  361. result_user_ptr->call_type_id=call_type;
  362. result_user_ptr->user_name=user_name;
  363. call_card_ptr card_ptr(new call_card());
  364. card_ptr->cardid=CCT_CALL_ALL;
  365. result_user_ptr->cards[card_ptr->cardid]=card_ptr;
  366. }
  367. //转发给web
  368. response_accept_cancel(result_user_ptr);
  369. }
  370. /*
  371. 采集回复web:取消呼叫
  372. {
  373. "user_name":"fjb", // 取消人
  374. "call_time": 23432, // 发起呼叫时间
  375. "stations":[{"stationid":102}], // 分站号
  376. "cards":[{"cardid":"0","cardtype":"1"}] // 取消的卡
  377. }
  378. */
  379. void module_call::response_accept_cancel(const call_user_ptr user_ptr)
  380. {
  381. rapidjson::Document doc(rapidjson::kObjectType);
  382. auto& allocator = doc.GetAllocator();
  383. rapidjson::Value node_data(rapidjson::kObjectType);
  384. tool_json::add_member(node_data, JSON_KEY_CALL_CARD_USER_NAME, user_ptr->user_name, allocator);
  385. node_data.AddMember(JSON_KEY_CALL_CARD_CALL_TIME, tool_time::to_ms(user_ptr->call_time), allocator);
  386. node_data.AddMember(JSON_KEY_CALL_CARD_CALL_TYPE, user_ptr->call_type_id, allocator);
  387. rapidjson::Value node_cards(rapidjson::kArrayType);
  388. //加入取消呼叫的卡
  389. auto iter = user_ptr->cards.begin();
  390. for(;iter != user_ptr->cards.end();++iter)
  391. {
  392. rapidjson::Value node_card(rapidjson::kObjectType);
  393. tool_json::add_member(node_card, JSON_KEY_CALL_CARD_CARD_ID, iter->second->to_id64_str(), allocator);
  394. node_card.AddMember(JSON_KEY_CALL_CARD_CARD_TYPE_ID,iter->second->cardtype, allocator);
  395. node_cards.PushBack(node_card,allocator);
  396. }
  397. node_data.AddMember(JSON_KEY_CALL_CARD_CARDS, node_cards, allocator);
  398. doc.AddMember(JSON_ROOT_KEY_CMD, JSON_CMD_VALUE_CALL_CARD_CANCEL_RESPONSE, allocator);
  399. doc.AddMember(JSON_ROOT_KEY_VERSION, INTERFACE_VERSION, allocator);
  400. doc.AddMember(JSON_ROOT_KEY_DATA, node_data, allocator);
  401. //转发给web
  402. swsClientMgr.send(JSON_CMD_VALUE_PUSH, tool_json::doc_to_json(doc));
  403. }
  404. /*
  405. 登陆时,采集发送web:井下所有呼叫
  406. {"cmd":"callcardlist","data":[
  407. ["fjb", // 发起呼叫人
  408. "0010000000001", // 呼叫的卡号
  409. 103, // 分站号
  410. 23431, // 发起呼叫时间
  411. 0, //
  412. 0 // 正在呼叫:2/结束呼叫/
  413. ]
  414. ]}
  415. */
  416. std::string module_call::accept_login()
  417. {
  418. std::vector<call_card_ptr> arr;
  419. get_all_call_cards(arr);
  420. if(arr.empty())
  421. {
  422. return "";
  423. }
  424. return to_call_card_list_json(arr);
  425. }
  426. std::string module_call::to_call_card_list_json(std::vector<call_card_ptr> arr)
  427. {
  428. rapidjson::Document doc(rapidjson::kObjectType);
  429. auto& allocator = doc.GetAllocator();
  430. rapidjson::Value node_cards(rapidjson::kArrayType);
  431. to_node_array(node_cards, arr, allocator);
  432. doc.AddMember(JSON_ROOT_KEY_CMD, JSON_CMD_VALUE_CALL_CARD_LIST, allocator);
  433. doc.AddMember(JSON_ROOT_KEY_VERSION, INTERFACE_VERSION, allocator);
  434. doc.AddMember(JSON_ROOT_KEY_DATA, node_cards, allocator);
  435. return tool_json::doc_to_json(doc);
  436. }
  437. void module_call::add_cards_to_user(const std::vector<sio::message::ptr>& card_vec, call_user_ptr user_ptr)
  438. {
  439. std::vector<sio::message::ptr>::const_iterator it_card = card_vec.begin();
  440. for(; it_card != card_vec.end(); ++it_card)
  441. {
  442. std::string s_card_id;
  443. if(!tool_map::try_get_value(s_card_id, JSON_KEY_CALL_CARD_CARD_ID, (*it_card))
  444. ||s_card_id.empty())
  445. {
  446. log_error("发起呼叫,web发来的数据 card_id 格式不对 或为空");
  447. continue;
  448. }
  449. uint32_t id = tool_other::id64_to_id(s_card_id);
  450. int type= tool_other::id64_to_type(s_card_id);
  451. log_info("发起呼叫 cardid=%d, cardtype=%d", id, type);
  452. call_card_ptr card_ptr(new call_card());
  453. card_ptr->cardid = id;
  454. card_ptr->cardtype = type;
  455. copy(user_ptr, card_ptr);
  456. card_ptr->call_state = CALL_ING;
  457. user_ptr->cards[card_ptr->cardid]=card_ptr;
  458. }
  459. }
  460. void module_call::get_user_all_call_cards(call_user_ptr& user_ptr,
  461. const std::unordered_map<uint64_t,std::shared_ptr<card_location_base>>& cardlist,
  462. std::vector<call_card_ptr>& out_data)
  463. {
  464. auto& cards=user_ptr->cards;
  465. int status = CALL_ING;
  466. //如果是全员呼叫,增加所有卡
  467. auto card_it = cards.find(CCT_CALL_ALL);
  468. if(cards.end()!=card_it)
  469. {
  470. if(user_ptr->is_timeout())//呼叫超时, 清空全员卡, 并设置呼叫状态为结束
  471. {
  472. log_info("全员呼叫发送线程:呼叫用户超时,用户名=%s", user_ptr->user_name.c_str());
  473. cards.clear();
  474. status=CALL_END;
  475. }
  476. auto g_it=cardlist.begin();
  477. for(;g_it!=cardlist.end();++g_it)//如果是全员呼叫,增加所有卡
  478. {
  479. auto site_ptr = g_it->second->get_area_tool()->m_site;
  480. if(!site_ptr)
  481. {
  482. continue;
  483. }
  484. call_card_ptr card_ptr(new call_card());
  485. copy(user_ptr, card_ptr);
  486. card_ptr->cardid = g_it->second->m_id;
  487. card_ptr->cardtype = g_it->second->m_type;
  488. card_ptr->stationid=site_ptr->id();
  489. card_ptr->call_state=status;
  490. if(CALL_SUCCESSED == g_it->second->get_mine_tool()->m_status_call)
  491. {
  492. card_ptr->call_state = CALL_SUCCESSED;
  493. }
  494. else
  495. {
  496. g_it->second->get_mine_tool()->m_status_call = CALL_ING;
  497. }
  498. if(card_ptr->is_timeout())
  499. {
  500. g_it->second->get_mine_tool()->m_status_call = 0;
  501. }
  502. card_ptr->is_display = g_it->second->m_display;
  503. out_data.push_back(card_ptr);
  504. }
  505. return;
  506. }
  507. //定员呼叫
  508. auto iter_card = cards.begin();
  509. for(;iter_card!=cards.end();++iter_card)
  510. {
  511. //增加到结果集中
  512. out_data.push_back(iter_card->second);
  513. auto g_card = cardlist.find(iter_card->second->to_id64());
  514. if(cardlist.end()==g_card)//在全局卡列表中没有这张卡,
  515. {
  516. log_error("定员呼叫发送线程:全局卡列表中没有这张卡,卡id=%d,卡类型=%d",
  517. iter_card->second->cardid, iter_card->second->cardtype);
  518. iter_card->second->call_state = CALL_FAILED;
  519. cards.erase(iter_card--); //在用户下删除这张卡
  520. continue;
  521. }
  522. auto site_ptr = g_card->second->get_area_tool()->m_site;
  523. if(!site_ptr)
  524. {
  525. log_error("定员呼叫发送线程:全局卡列表中这张卡已经上井或无效,卡id=%d,卡类型=%d",
  526. iter_card->second->cardid, iter_card->second->cardtype);
  527. iter_card->second->call_state = CALL_FAILED;
  528. cards.erase(iter_card--); //在用户下删除这张卡
  529. continue;
  530. }
  531. //更新卡的分站id
  532. iter_card->second->stationid = site_ptr->id();
  533. if(CALL_SUCCESSED == g_card->second->get_mine_tool()->m_status_call)
  534. {
  535. iter_card->second->call_state = CALL_SUCCESSED;
  536. }
  537. else
  538. {
  539. g_card->second->get_mine_tool()->m_status_call = CALL_ING;
  540. }
  541. if(iter_card->second->is_timeout())//呼叫超时
  542. {
  543. log_info("定员呼叫发送线程:呼叫卡超时,用户名=%s,卡id=%d,卡类型=%d",
  544. iter_card->second->user_name.c_str(),
  545. iter_card->second->cardid, iter_card->second->cardtype);
  546. iter_card->second->call_state = CALL_END;
  547. if(CALL_SUCCESSED == g_card->second->get_mine_tool()->m_status_call)
  548. {
  549. iter_card->second->call_state = CALL_SUCCESSED;
  550. g_card->second->get_mine_tool()->m_status_call = 0;
  551. }
  552. cards.erase(iter_card--);//在用户下删除这张卡
  553. }
  554. }
  555. }
  556. /**
  557. * @brief 系统自动发送呼叫给指定的卡
  558. */
  559. void module_call::system_call_apoint(int card_id,int card_type)
  560. {
  561. int call_type = -1, call_level = 1, call_time_interval=-1;
  562. uint32_t call_time_out = 0;
  563. std::string user_name = "system_apoint";
  564. std::vector<call_card_ptr> result_arr;//结果集
  565. call_user_ptr user_ptr;
  566. std::lock_guard<std::mutex> lock(_mutex);
  567. auto user_map_ptr = _map.find(user_name);
  568. if(_map.end() == user_map_ptr)//没有这个用户就新建并加入
  569. {
  570. user_ptr=call_user_ptr(new call_user());
  571. _map[user_name]=user_ptr;
  572. }
  573. else//有这个用户
  574. {
  575. user_ptr=user_map_ptr->second;
  576. }
  577. //更新用户信息
  578. user_ptr->call_time=std::chrono::system_clock::now();
  579. user_ptr->call_time_out=call_time_out;
  580. user_ptr->call_time_interval = call_time_interval;
  581. user_ptr->call_type_id=call_type;
  582. user_ptr->call_level_id=call_level;
  583. user_ptr->user_name=user_name;
  584. call_card_ptr c_card_ptr(new call_card());
  585. c_card_ptr->cardid = card_id;
  586. c_card_ptr->cardtype = card_type;
  587. copy(user_ptr, c_card_ptr);
  588. c_card_ptr->call_state = CALL_ING;
  589. user_ptr->cards[c_card_ptr->cardid]=c_card_ptr;
  590. for(auto& it:user_ptr->cards)//添加到结果集中
  591. {
  592. result_arr.push_back(it.second);
  593. }
  594. log_info("发起呼叫 cardid=%d, cardtype=%d", card_id,card_type);
  595. //组装json发送给web
  596. response_accept_call(result_arr);
  597. }
  598. /**
  599. * @brief 系统自动发送呼叫给指定的卡
  600. */
  601. void module_call::system_cancel_call_apoint(int card_id,int card_type)
  602. {
  603. int call_type = -1;//, call_level = -1;
  604. std::string user_name = "system_apoint";
  605. int64_t call_time;
  606. call_user_ptr result_user_ptr(new call_user());
  607. result_user_ptr->call_type_id=call_type;
  608. // result_user_ptr->call_level_id=call_level;
  609. result_user_ptr->call_time = std::chrono::system_clock::time_point(std::chrono::milliseconds(call_time));
  610. result_user_ptr->user_name=user_name;
  611. std::lock_guard<std::mutex> lock(_mutex);
  612. call_user_ptr user_ptr(new call_user());
  613. auto it_map = _map.find(user_name);
  614. if(_map.end()!=it_map)//呼叫用户
  615. {
  616. user_ptr=it_map->second;
  617. }
  618. if (user_ptr->cards.find(card_id) == user_ptr->cards.end())
  619. {
  620. return ;
  621. }
  622. call_card_ptr c_card_ptr(new call_card());
  623. c_card_ptr->cardid = card_id;
  624. c_card_ptr->cardtype = card_type;
  625. result_user_ptr->cards[c_card_ptr->cardid]=c_card_ptr;//增加到结果集中
  626. user_ptr->cards.erase(c_card_ptr->cardid);//删除这个卡
  627. log_info("取消呼叫: cardid=%d, cardtype=%d", card_id,card_type);
  628. // 没有呼叫信息,删除该用户记录
  629. if(user_ptr->cards.empty())
  630. {
  631. _map.erase(user_name);
  632. }
  633. response_accept_cancel(result_user_ptr);
  634. }
  635. void module_call::send_anti_collision(const std::map<int, call_card>& cards)
  636. {
  637. std::vector<call_card_ptr> vt_cards;
  638. for(auto c : cards){
  639. vt_cards.push_back(std::make_shared<call_card>(c.second));
  640. }
  641. log_info("[anti_coll] card's size = %d, vt_cards's size = %d", cards.size(), vt_cards.size());
  642. call_site_map site_map;
  643. get_site_map(vt_cards, site_map);
  644. log_info("[anti_coll] site's size = %d", site_map.size());
  645. if(!site_map.empty()){
  646. send_to_sites(site_map);
  647. }
  648. }
  649. std::string module_call::get_json_anti_collision()
  650. {
  651. rapidjson::Document doc(rapidjson::kObjectType);
  652. rapidjson::Document::AllocatorType& allocator = doc.GetAllocator();
  653. doc.SetObject();
  654. rapidjson::Value data(rapidjson::kArrayType);
  655. auto tmp = event_list::instance()->m_map;
  656. for(auto it_ev = tmp.cbegin(); it_ev != tmp.cend(); ++it_ev)
  657. {
  658. if(ET_PERSON_VEHICLE_ANTI_COLLISION != it_ev->second->m_ev_type)
  659. {
  660. continue;
  661. }
  662. rapidjson::Value obj(rapidjson::kObjectType);
  663. rapidjson::Value vid(rapidjson::kObjectType);
  664. vid.SetInt(atoi(it_ev->second->m_obj_id.c_str()));
  665. obj.AddMember("vid", vid, allocator);
  666. rapidjson::Value elems(rapidjson::kArrayType);
  667. std::string msg = it_ev->second->m_desc;
  668. std::vector<std::string> vt_msg;
  669. boost::split(vt_msg, msg, boost::is_any_of(";"));
  670. for(auto it_elem : vt_msg){
  671. std::vector<std::string> vt_cid_d;
  672. boost::split(vt_cid_d, it_elem, boost::is_any_of(","));
  673. if(vt_cid_d.size()<2){
  674. continue;
  675. }
  676. rapidjson::Value e(rapidjson::kArrayType);
  677. rapidjson::Value o(rapidjson::kObjectType);
  678. o.SetInt(atoi(vt_cid_d[0].c_str()));
  679. e.PushBack(obj, allocator);
  680. o.SetDouble(atof(vt_cid_d[1].c_str()));
  681. e.PushBack(obj, allocator);
  682. elems.PushBack(e, allocator);
  683. }
  684. obj.AddMember("pid", elems, allocator);
  685. std::string t = tool_time::to_str(it_ev->second->m_cur_time);
  686. tool_json::add_member(obj, "start_time", t, allocator);
  687. data.PushBack(obj, allocator);
  688. }
  689. if(!data.Size()){
  690. return "";
  691. }
  692. doc.AddMember(JSON_ROOT_KEY_CMD, "pv_anti_coll", allocator);
  693. doc.AddMember(JSON_ROOT_KEY_VERSION, INTERFACE_VERSION, allocator);
  694. doc.AddMember(JSON_ROOT_KEY_DATA, data, allocator);
  695. rapidjson::StringBuffer sb;
  696. rapidjson::PrettyWriter<rapidjson::StringBuffer> writer(sb);
  697. doc.Accept(writer);
  698. return sb.GetString();
  699. }