module_call.cpp 24 KB

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