1
0

module_call.cpp 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533
  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. void module_call::run()
  9. {
  10. std::vector<call_card_ptr> arr;
  11. get_all_call_cards(arr);
  12. if(arr.empty())
  13. {
  14. return ;
  15. }
  16. //转发给web
  17. tool_other::send_json(JSON_CMD_VALUE_PUSH, to_call_card_list_json(arr));
  18. //将呼叫命令发送给标识卡终端
  19. call_site_map site_map;
  20. get_site_map(arr, site_map);
  21. if(!site_map.empty())
  22. {
  23. send_to_sites(site_map);
  24. }
  25. }
  26. void module_call::rev_from_card_resp(std::shared_ptr<card_location_base> card_ptr)
  27. {
  28. card_ptr->get_mine_tool()->m_status_call = CALL_SUCCESSED;
  29. }
  30. /*
  31. web发给采集:发起呼叫
  32. {
  33. cmd: 'call_card_req',
  34. data: {
  35. call_type_id: this.cal_type, // 全员呼叫:0 定员呼叫:1
  36. call_time_out: this.choosedTime, // 呼叫时长
  37. call_level_id: this.call_type_id, // 呼叫类型 一般呼叫:1 紧急呼叫:2
  38. user_name: xdata.userName, // 呼叫人
  39. call_time: time, // 呼叫时间戳
  40. stations: [{ stationid: 0 }], // 分站 0为全员
  41. cards: cards // 人员 0为全员 //旧代码为map格式 [{ stationid: 0 }]
  42. }
  43. }
  44. */
  45. void module_call::accept_call(sio::message::ptr const& data)
  46. {
  47. int call_type = -1, call_level = -1, call_time_interval=-1;
  48. uint32_t call_time_out = 0;
  49. std::string user_name;
  50. if(!tool_map::try_get_value(call_type, JSON_KEY_CALL_CARD_CALL_TYPE, data)
  51. || !tool_map::try_get_value(call_level, JSON_KEY_CALL_CARD_CALL_LEVEL, data)
  52. || !tool_map::try_get_value(call_time_out, JSON_KEY_CALL_CARD_CALL_TIME_OUT, data)
  53. || !tool_map::try_get_value(user_name, JSON_KEY_CALL_CARD_USER_NAME, data)
  54. || !tool_map::try_get_value(call_time_interval, JSON_KEY_CALL_CARD_CALL_TIME, data)
  55. )
  56. {
  57. log_error("发起呼叫: 收到的json不对,解析不出call_type = -1, call_level = -1, \
  58. call_time_out=-1, call_time_interval=-1,user_name");
  59. return;
  60. }
  61. if(user_name.empty())
  62. {
  63. log_error("发起呼叫: user_name用户名为空");
  64. return;
  65. }
  66. std::vector<call_card_ptr> result_arr;//结果集
  67. call_user_ptr user_ptr;
  68. if(CCT_CALL_ALL==call_type)// 全员
  69. {
  70. user_ptr=call_user_ptr(new call_user());
  71. user_ptr->call_time=std::chrono::system_clock::now();
  72. user_ptr->call_time_out=call_time_out;
  73. user_ptr->call_time_interval = call_time_interval;
  74. user_ptr->call_type_id=call_type;
  75. user_ptr->call_level_id=call_level;
  76. user_ptr->user_name=user_name;
  77. call_card_ptr temp(new call_card());
  78. copy(user_ptr, temp);
  79. temp->cardid = CCT_CALL_ALL; //全员卡
  80. temp->call_state = CALL_ING;
  81. result_arr.push_back(temp);//增加到结果集中
  82. //添加全员卡
  83. user_ptr->cards.insert(std::make_pair(temp->cardid, temp));
  84. //更新用户信息
  85. std::lock_guard<std::mutex> lock(_mutex);
  86. _map[user_name]=user_ptr;
  87. }
  88. else// 定员: 如果有全员,定员不生效
  89. {
  90. std::lock_guard<std::mutex> lock(_mutex);
  91. auto user_map_ptr = _map.find(user_name);
  92. if(_map.end() == user_map_ptr)//没有这个用户就新建并加入
  93. {
  94. user_ptr=call_user_ptr(new call_user());
  95. _map[user_name]=user_ptr;
  96. }
  97. else//有这个用户
  98. {
  99. user_ptr=user_map_ptr->second;
  100. // 如果有全员,定员不生效
  101. auto card_temp = user_ptr->cards.find(CCT_CALL_ALL);
  102. if(user_ptr->cards.end() != card_temp)
  103. {
  104. return;
  105. }
  106. }
  107. //更新用户信息
  108. user_ptr->call_time=std::chrono::system_clock::now();
  109. user_ptr->call_time_out=call_time_out;
  110. user_ptr->call_time_interval = call_time_interval;
  111. user_ptr->call_type_id=call_type;
  112. user_ptr->call_level_id=call_level;
  113. user_ptr->user_name=user_name;
  114. std::vector<sio::message::ptr> card_vec;
  115. if(!tool_map::try_get_value(card_vec, JSON_KEY_CALL_CARD_CARDS, data) || card_vec.size() == 0)
  116. {
  117. log_error("发起呼叫,web发来的数据JSON_KEY_CALL_CARD_CARDS字段为空 或者不是数组");
  118. return;
  119. }
  120. add_cards_to_user(card_vec, user_ptr);
  121. for(auto& it:user_ptr->cards)//添加到结果集中
  122. {
  123. result_arr.push_back(it.second);
  124. }
  125. }
  126. //组装json发送给web
  127. if(!result_arr.empty())
  128. {
  129. response_accept_call(result_arr);
  130. }
  131. }
  132. //采集回复web发起呼叫
  133. //{
  134. // 'cmd': 'call_card_resp',
  135. // 'data': [
  136. // ["fjb", // 发起呼叫人
  137. // "0010000000001", // 呼叫的卡号
  138. // 103, // 分站号
  139. // 23431, // 发起呼叫时间
  140. // 0, //
  141. // 0 // 正在呼叫:2/结束呼叫/
  142. // ]
  143. // ]
  144. //}
  145. void module_call::response_accept_call(std::vector<call_card_ptr> cards)
  146. {
  147. rapidjson::Document doc(rapidjson::kObjectType);
  148. auto& allocator = doc.GetAllocator();
  149. rapidjson::Value node_cards(rapidjson::kArrayType);
  150. //组装json发送给web
  151. to_node_array(node_cards, cards, allocator);
  152. doc.AddMember(JSON_ROOT_KEY_CMD,JSON_CMD_VALUE_CALL_CARD_RESPONSE, allocator);
  153. doc.AddMember(JSON_ROOT_KEY_VERSION, INTERFACE_VERSION, allocator);
  154. doc.AddMember(JSON_ROOT_KEY_DATA, node_cards, allocator);
  155. tool_other::send_json(JSON_CMD_VALUE_PUSH, tool_json::doc_to_json(doc));
  156. }
  157. //web发给采集:取消呼叫
  158. //{
  159. // cmd: 'call_card_cancel_req',
  160. // data: {
  161. // call_type_id: type, // 全员/定员
  162. // user_name: xdata.userName, // 取消人
  163. // call_time: callTime, // 时间戳
  164. // stations: this.stationid, // 分站
  165. // cards: this.cards // 取消呼叫卡 //旧代码为map格式 [{ stationid: 0 }]
  166. // }
  167. //}
  168. void module_call::accept_cancel(sio::message::ptr const& node_data)
  169. {
  170. int call_type = -1;//, call_level = -1;
  171. std::string user_name;
  172. int64_t call_time;
  173. if(!tool_map::try_get_value(call_type, JSON_KEY_CALL_CARD_CALL_TYPE, node_data)
  174. || !tool_map::try_get_value(call_time, JSON_KEY_CALL_CARD_CALL_TIME, node_data)
  175. || !tool_map::try_get_value(user_name, JSON_KEY_CALL_CARD_USER_NAME, node_data)
  176. )
  177. {
  178. log_error("取消呼叫: 收到的json不对,解析不出int call_type = -1, call_level = -1,user_name");
  179. return;
  180. }
  181. if(user_name.empty())
  182. {
  183. log_error("取消呼叫: user_name用户名为空");
  184. return;
  185. }
  186. call_user_ptr result_user_ptr(new call_user());
  187. result_user_ptr->call_type_id=call_type;
  188. // result_user_ptr->call_level_id=call_level;
  189. result_user_ptr->call_time = std::chrono::system_clock::time_point(std::chrono::milliseconds(call_time));
  190. result_user_ptr->user_name=user_name;
  191. //取消呼叫有两种:
  192. //取消全员呼叫
  193. //对指定卡取消呼叫
  194. if(CCT_CALL_ALL==call_type)// 取消全员呼叫
  195. {
  196. //删除用户
  197. std::lock_guard<std::mutex> lock(_mutex);
  198. _map.erase(user_name);
  199. }
  200. else// 取消定员呼叫
  201. {
  202. std::vector<sio::message::ptr> card_vec;
  203. if(!tool_map::try_get_value(card_vec, JSON_KEY_CALL_CARD_CARDS, node_data) || card_vec.size() == 0)
  204. {
  205. log_error("取消呼叫: web发来的数据cards字段为空 或者不是数组");
  206. return;
  207. }
  208. std::lock_guard<std::mutex> lock(_mutex);
  209. call_user_ptr user_ptr(new call_user());
  210. auto it_map = _map.find(user_name);
  211. if(_map.end()!=it_map)//呼叫用户
  212. {
  213. user_ptr=it_map->second;
  214. }
  215. std::vector<sio::message::ptr>::const_iterator it_card = card_vec.begin();
  216. std::string s_card_id;
  217. for(; it_card != card_vec.end(); ++it_card)
  218. {
  219. if(!tool_map::try_get_value(s_card_id, JSON_KEY_CALL_CARD_CARD_ID, (*it_card)))
  220. {
  221. log_error("取消呼叫: web发来的数据 card_id 格式不对");
  222. continue;
  223. }
  224. uint32_t id = tool_other::id64_to_id(s_card_id);
  225. int type= tool_other::id64_to_type(s_card_id);
  226. log_info("取消呼叫: cardid=%d, cardtype=%d", id, type);
  227. call_card_ptr card_ptr(new call_card());
  228. card_ptr->cardid = id;
  229. card_ptr->cardtype = type;
  230. result_user_ptr->cards[card_ptr->cardid]=card_ptr;//增加到结果集中
  231. user_ptr->cards.erase(card_ptr->cardid);//删除这个卡
  232. }
  233. // 没有呼叫信息,删除该用户记录
  234. if(user_ptr->cards.empty())
  235. {
  236. _map.erase(user_name);
  237. }
  238. }
  239. //如果要取消呼叫的用户下没有卡,建一张CCT_CALL_ALL卡返回
  240. if(result_user_ptr->cards.empty())
  241. {
  242. // result_user_ptr->call_level_id=call_level;
  243. //result_user_ptr->call_time=std::chrono::system_clock::now();
  244. result_user_ptr->call_time_out=0;
  245. result_user_ptr->call_type_id=call_type;
  246. result_user_ptr->user_name=user_name;
  247. call_card_ptr card_ptr(new call_card());
  248. card_ptr->cardid=CCT_CALL_ALL;
  249. result_user_ptr->cards[card_ptr->cardid]=card_ptr;
  250. }
  251. //转发给web
  252. response_accept_cancel(result_user_ptr);
  253. }
  254. /*
  255. 采集回复web:取消呼叫
  256. {
  257. "user_name":"fjb", // 取消人
  258. "call_time": 23432, // 发起呼叫时间
  259. "stations":[{"stationid":102}], // 分站号
  260. "cards":[{"cardid":"0","cardtype":"1"}] // 取消的卡
  261. }
  262. */
  263. void module_call::response_accept_cancel(const call_user_ptr user_ptr)
  264. {
  265. rapidjson::Document doc(rapidjson::kObjectType);
  266. auto& allocator = doc.GetAllocator();
  267. rapidjson::Value node_data(rapidjson::kObjectType);
  268. tool_json::add_member(node_data, JSON_KEY_CALL_CARD_USER_NAME, user_ptr->user_name, allocator);
  269. node_data.AddMember(JSON_KEY_CALL_CARD_CALL_TIME, tool_time::to_ms(user_ptr->call_time), allocator);
  270. node_data.AddMember(JSON_KEY_CALL_CARD_CALL_TYPE, user_ptr->call_type_id, allocator);
  271. rapidjson::Value node_cards(rapidjson::kArrayType);
  272. //加入取消呼叫的卡
  273. auto iter = user_ptr->cards.begin();
  274. for(;iter != user_ptr->cards.end();++iter)
  275. {
  276. rapidjson::Value node_card(rapidjson::kObjectType);
  277. tool_json::add_member(node_card, JSON_KEY_CALL_CARD_CARD_ID, iter->second->to_id64_str(), allocator);
  278. node_card.AddMember(JSON_KEY_CALL_CARD_CARD_TYPE_ID,iter->second->cardtype, allocator);
  279. node_cards.PushBack(node_card,allocator);
  280. }
  281. node_data.AddMember(JSON_KEY_CALL_CARD_CARDS, node_cards, allocator);
  282. doc.AddMember(JSON_ROOT_KEY_CMD, JSON_CMD_VALUE_CALL_CARD_CANCEL_RESPONSE, allocator);
  283. doc.AddMember(JSON_ROOT_KEY_VERSION, INTERFACE_VERSION, allocator);
  284. doc.AddMember(JSON_ROOT_KEY_DATA, node_data, allocator);
  285. //转发给web
  286. tool_other::send_json(JSON_CMD_VALUE_PUSH, tool_json::doc_to_json(doc));
  287. }
  288. /*
  289. 登陆时,采集发送web:井下所有呼叫
  290. {"cmd":"callcardlist","data":[
  291. ["fjb", // 发起呼叫人
  292. "0010000000001", // 呼叫的卡号
  293. 103, // 分站号
  294. 23431, // 发起呼叫时间
  295. 0, //
  296. 0 // 正在呼叫:2/结束呼叫/
  297. ]
  298. ]}
  299. */
  300. std::string module_call::accept_login()
  301. {
  302. std::vector<call_card_ptr> arr;
  303. get_all_call_cards(arr);
  304. if(arr.empty())
  305. {
  306. return "";
  307. }
  308. return to_call_card_list_json(arr);
  309. }
  310. std::string module_call::to_call_card_list_json(std::vector<call_card_ptr> arr)
  311. {
  312. rapidjson::Document doc(rapidjson::kObjectType);
  313. auto& allocator = doc.GetAllocator();
  314. rapidjson::Value node_cards(rapidjson::kArrayType);
  315. to_node_array(node_cards, arr, allocator);
  316. doc.AddMember(JSON_ROOT_KEY_CMD, JSON_CMD_VALUE_CALL_CARD_LIST, allocator);
  317. doc.AddMember(JSON_ROOT_KEY_VERSION, INTERFACE_VERSION, allocator);
  318. doc.AddMember(JSON_ROOT_KEY_DATA, node_cards, allocator);
  319. return tool_json::doc_to_json(doc);
  320. }
  321. void module_call::add_cards_to_user(const std::vector<sio::message::ptr>& card_vec, call_user_ptr user_ptr)
  322. {
  323. std::vector<sio::message::ptr>::const_iterator it_card = card_vec.begin();
  324. for(; it_card != card_vec.end(); ++it_card)
  325. {
  326. std::string s_card_id;
  327. if(!tool_map::try_get_value(s_card_id, JSON_KEY_CALL_CARD_CARD_ID, (*it_card))
  328. ||s_card_id.empty())
  329. {
  330. log_error("发起呼叫,web发来的数据 card_id 格式不对 或为空");
  331. continue;
  332. }
  333. uint32_t id = tool_other::id64_to_id(s_card_id);
  334. int type= tool_other::id64_to_type(s_card_id);
  335. log_info("发起呼叫 cardid=%d, cardtype=%d", id, type);
  336. call_card_ptr card_ptr(new call_card());
  337. card_ptr->cardid = id;
  338. card_ptr->cardtype = type;
  339. copy(user_ptr, card_ptr);
  340. card_ptr->call_state = CALL_ING;
  341. user_ptr->cards[card_ptr->cardid]=card_ptr;
  342. }
  343. }
  344. void module_call::get_user_all_call_cards(call_user_ptr& user_ptr,
  345. const std::unordered_map<uint64_t,std::shared_ptr<card_location_base>>& cardlist,
  346. std::vector<call_card_ptr>& out_data)
  347. {
  348. auto& cards=user_ptr->cards;
  349. int status = CALL_ING;
  350. //如果是全员呼叫,增加所有卡
  351. auto card_it = cards.find(CCT_CALL_ALL);
  352. if(cards.end()!=card_it)
  353. {
  354. if(user_ptr->is_timeout())//呼叫超时, 清空全员卡, 并设置呼叫状态为结束
  355. {
  356. log_info("全员呼叫发送线程:呼叫用户超时,用户名=%s", user_ptr->user_name.c_str());
  357. cards.clear();
  358. status=CALL_END;
  359. }
  360. auto g_it=cardlist.begin();
  361. for(;g_it!=cardlist.end();++g_it)//如果是全员呼叫,增加所有卡
  362. {
  363. auto site_ptr = g_it->second->get_site_area();
  364. if(!site_ptr || site_ptr->is_invalid())
  365. {
  366. continue;
  367. }
  368. call_card_ptr card_ptr(new call_card());
  369. copy(user_ptr, card_ptr);
  370. card_ptr->cardid = g_it->second->m_id;
  371. card_ptr->cardtype = g_it->second->m_type;
  372. card_ptr->stationid=site_ptr->site_id();
  373. card_ptr->call_state=status;
  374. if(CALL_SUCCESSED == g_it->second->get_mine_tool()->m_status_call)
  375. {
  376. card_ptr->call_state = CALL_SUCCESSED;
  377. }
  378. else
  379. {
  380. g_it->second->get_mine_tool()->m_status_call = CALL_ING;
  381. }
  382. if(card_ptr->is_timeout())
  383. {
  384. g_it->second->get_mine_tool()->m_status_call = 0;
  385. }
  386. card_ptr->is_display = g_it->second->m_display;
  387. out_data.push_back(card_ptr);
  388. }
  389. return;
  390. }
  391. //定员呼叫
  392. auto iter_card = cards.begin();
  393. for(;iter_card!=cards.end();++iter_card)
  394. {
  395. //增加到结果集中
  396. out_data.push_back(iter_card->second);
  397. auto g_card = cardlist.find(iter_card->second->to_id64());
  398. if(cardlist.end()==g_card)//在全局卡列表中没有这张卡,
  399. {
  400. log_error("定员呼叫发送线程:全局卡列表中没有这张卡,卡id=%d,卡类型=%d",
  401. iter_card->second->cardid, iter_card->second->cardtype);
  402. iter_card->second->call_state = CALL_FAILED;
  403. cards.erase(iter_card--); //在用户下删除这张卡
  404. continue;
  405. }
  406. auto site_ptr = g_card->second->get_site_area();
  407. if(!site_ptr || site_ptr->is_invalid())
  408. {
  409. log_error("定员呼叫发送线程:全局卡列表中这张卡已经上井或无效,卡id=%d,卡类型=%d",
  410. iter_card->second->cardid, iter_card->second->cardtype);
  411. iter_card->second->call_state = CALL_FAILED;
  412. cards.erase(iter_card--); //在用户下删除这张卡
  413. continue;
  414. }
  415. //更新卡的分站id
  416. iter_card->second->stationid = site_ptr->site_id();
  417. if(CALL_SUCCESSED == g_card->second->get_mine_tool()->m_status_call)
  418. {
  419. iter_card->second->call_state = CALL_SUCCESSED;
  420. }
  421. else
  422. {
  423. g_card->second->get_mine_tool()->m_status_call = CALL_ING;
  424. }
  425. if(iter_card->second->is_timeout())//呼叫超时
  426. {
  427. log_info("定员呼叫发送线程:呼叫卡超时,用户名=%s,卡id=%d,卡类型=%d",
  428. iter_card->second->user_name.c_str(),
  429. iter_card->second->cardid, iter_card->second->cardtype);
  430. iter_card->second->call_state = CALL_END;
  431. if(CALL_SUCCESSED == g_card->second->get_mine_tool()->m_status_call)
  432. {
  433. iter_card->second->call_state = CALL_SUCCESSED;
  434. g_card->second->get_mine_tool()->m_status_call = 0;
  435. }
  436. cards.erase(iter_card--);//在用户下删除这张卡
  437. }
  438. }
  439. }