module_call.h 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523
  1. #ifndef MODULE_CALL_H
  2. #define MODULE_CALL_H
  3. /**
  4. * @brief 呼叫模块
  5. * 用户通过WEB呼叫指定人员或者全部人员
  6. * @author 戴月腾
  7. * @date 2018-08-22
  8. */
  9. #include <map>
  10. #include <mutex>
  11. #include <string>
  12. #include <chrono>
  13. #include <boost/thread.hpp>
  14. #include <boost/enable_shared_from_this.hpp>
  15. #include <atomic>
  16. #include "rapidjson/prettywriter.h"
  17. #include"module_const.h"
  18. #include"module_web.h"
  19. #include"ant.h"
  20. /**
  21. * @brief 呼叫类型 全员 定员
  22. */
  23. enum CALL_CARD_TYPE
  24. {
  25. /// 全员呼叫
  26. CCT_CALL_ALL = 0,
  27. /// 定员
  28. CCT_CALL_APOINT=1,
  29. };
  30. /**
  31. * @brief 呼叫等级 一般呼叫 紧急呼叫
  32. */
  33. enum CALL_CARD_LEVEL
  34. {
  35. /// 呼叫等级: 1 一般呼叫
  36. CALL_LEVEL_NORMAL=1,
  37. /// 呼叫等级: 2 紧急呼叫
  38. CALL_LEVEL_CRITICAL=2
  39. };
  40. /**
  41. * @brief 呼叫状态
  42. */
  43. enum CALL_STATE{
  44. ///无呼叫信息
  45. CALL_NONE = 0,
  46. ///呼叫成功
  47. CALL_SUCCESSED = 1,
  48. ///呼叫中
  49. CALL_ING = 2,
  50. ///呼叫失败
  51. CALL_FAILED = 3,
  52. ///呼叫结束
  53. CALL_END=100
  54. };
  55. /**
  56. * @brief 呼叫模块
  57. */
  58. class module_call : public i_thread, public singleton_base<module_call>
  59. {
  60. private:
  61. friend class singleton_base<module_call>;
  62. module_call()
  63. {
  64. }
  65. private:
  66. /**
  67. * @brief 呼叫卡的信息
  68. */
  69. struct call_card
  70. {
  71. /// 呼叫卡号 0为全员
  72. uint32_t cardid;
  73. int32_t cardtype;
  74. /// 分站号 0为全员
  75. int stationid;
  76. ///呼叫类型 全员0, 定员1 CALL_CARD_TYPE
  77. int call_type_id;
  78. /// 呼叫类型:一般1,紧急2 CALL_CARD_LEVEL
  79. int call_level_id;
  80. /// 呼叫时长,单位分钟
  81. uint32_t call_time_out;
  82. int call_time_interval;
  83. /// 呼叫开始时间
  84. std::chrono::system_clock::time_point call_time;
  85. /// 呼叫状态,正在呼叫、呼叫成功
  86. int call_state;
  87. ///呼叫人
  88. std::string user_name;
  89. bool is_display = true;
  90. bool is_call_all()
  91. {
  92. return CCT_CALL_ALL == call_type_id;
  93. }
  94. uint64_t to_id64()
  95. {
  96. return tool_other::type_id_to_u64(cardtype, cardid);
  97. }
  98. std::string to_id64_str()
  99. {
  100. return tool_other::type_id_to_str(cardtype, cardid);
  101. }
  102. bool is_timeout()
  103. {
  104. uint32_t seconds = tool_time::elapse_seconds(call_time);
  105. return (seconds >= call_time_out*60);
  106. }
  107. };
  108. typedef std::shared_ptr<call_card> call_card_ptr;
  109. typedef std::map<int64_t,call_card_ptr> call_card_map;
  110. /**
  111. * @brief 呼叫用户信息
  112. */
  113. struct call_user
  114. {
  115. ///呼叫人
  116. std::string user_name;
  117. ///呼叫类型 全员0, 定员1 CALL_CARD_TYPE
  118. int call_type_id;
  119. /// 呼叫等级:一般1,紧急2 CALL_CARD_LEVEL
  120. int call_level_id;
  121. /// 呼叫时长,单位分钟
  122. uint32_t call_time_out;
  123. int call_time_interval;
  124. /// 呼叫开始时间
  125. std::chrono::system_clock::time_point call_time;
  126. ///呼叫卡列表
  127. call_card_map cards;
  128. bool is_timeout()
  129. {
  130. uint32_t seconds = tool_time::elapse_seconds(call_time);
  131. return (seconds >= call_time_out*60);
  132. }
  133. bool is_call_all()
  134. {
  135. return CCT_CALL_ALL == call_type_id;
  136. }
  137. };
  138. typedef std::shared_ptr<call_user> call_user_ptr;
  139. typedef std::map<std::string, call_user_ptr> call_user_map;
  140. struct call_site
  141. {
  142. int32_t site_id;
  143. ///呼叫类型 全员0, 定员1 CALL_CARD_TYPE
  144. int call_type_id;
  145. /// 呼叫等级:一般1,紧急2 CALL_CARD_LEVEL
  146. int call_level_id;
  147. call_card_map cards;
  148. bool is_call_all()
  149. {
  150. return CCT_CALL_ALL == call_type_id;
  151. }
  152. };
  153. typedef std::shared_ptr<call_site> call_site_ptr;
  154. typedef std::map<int32_t, call_site_ptr> call_site_map;
  155. public:
  156. /**
  157. * @brief web发给采集:发起呼叫
  158. * @param node_data
  159. */
  160. void accept_call(sio::message::ptr const& data);
  161. /**
  162. * @brief web发给采集:取消呼叫
  163. * @param node_data
  164. */
  165. void accept_cancel(sio::message::ptr const& data);
  166. /**
  167. * @brief 登陆时,采集发送web:请求井下所有呼叫
  168. */
  169. std::string accept_login();
  170. void rev_from_card_resp(std::shared_ptr<card_location_base> card_ptr);
  171. /**
  172. * @brief 系统自动发送呼叫给指定的卡
  173. */
  174. void system_call_apoint(int card_id,int card_type);
  175. /**
  176. * @brief 系统自动发送呼叫给指定的卡
  177. */
  178. void system_cancel_call_apoint(int card_id,int card_type);
  179. /**int card_id,int card_type
  180. * @brief
  181. */
  182. void init(config_file& config)
  183. {
  184. sleep_ms = std::stoi(config.get("service.send_call_interval","5000"));
  185. }
  186. private:
  187. /**
  188. * @brief 采集回复web发起呼叫
  189. * @param arr
  190. */
  191. void response_accept_call(const std::vector<call_card_ptr> arr);
  192. /**
  193. * @brief 采集回复web:取消呼叫
  194. * @param node_data
  195. */
  196. void response_accept_cancel(const call_user_ptr user_ptr);
  197. void add_cards_to_user(const std::vector<sio::message::ptr>& card_vec,
  198. call_user_ptr user_ptr);
  199. /**
  200. * @brief 线程函数
  201. */
  202. void run();
  203. void copy(const call_user_ptr src, call_user_ptr des)
  204. {
  205. des->call_level_id = src->call_level_id;
  206. des->call_time = src->call_time;
  207. des->call_time_out = src->call_time_out;
  208. des->call_type_id = src->call_type_id;
  209. des->user_name = src->user_name;
  210. }
  211. void copy(const call_card_ptr src, call_card_ptr des)
  212. {
  213. des->call_level_id = src->call_level_id;
  214. des->call_state = src->call_state;
  215. des->call_time = src->call_time;
  216. des->call_time_out = src->call_time_out;
  217. des->cardid = src->cardid;
  218. des->cardtype = src->cardtype;
  219. des->user_name = src->user_name;
  220. des->stationid = src->stationid;
  221. des->call_type_id = src->call_type_id;
  222. }
  223. void copy(const call_user_ptr src, call_card_ptr des)
  224. {
  225. des->call_level_id = src->call_level_id;
  226. //des->call_state = src->call_state;
  227. des->call_time = src->call_time;
  228. des->call_time_out = src->call_time_out;
  229. des->call_time_interval = src->call_time_interval;
  230. //des->cardid = src->cardid;
  231. des->user_name = src->user_name;
  232. des->call_type_id = src->call_type_id;
  233. }
  234. /**
  235. * @brief 获取用户下的所有呼叫卡
  236. * @param user_ptr
  237. * @param cardlist
  238. * @param out_data
  239. */
  240. void get_user_all_call_cards(call_user_ptr& user_ptr,
  241. const std::unordered_map<uint64_t,std::shared_ptr<card_location_base>>& cardlist,
  242. std::vector<call_card_ptr>& out_data);
  243. /**
  244. * @brief 获取所有的呼叫卡
  245. * @param user_map
  246. * @param cardlist
  247. * @param out_data
  248. */
  249. void get_all_call_cards(call_user_map& user_map,
  250. const std::unordered_map<uint64_t,std::shared_ptr<card_location_base>>& cardlist,
  251. std::vector<call_card_ptr>& out_data)
  252. {
  253. auto iter_m_map=user_map.begin();
  254. for(;iter_m_map!=user_map.end();++iter_m_map)
  255. {
  256. get_user_all_call_cards(iter_m_map->second, cardlist, out_data);
  257. if(iter_m_map->second->cards.empty())//没有卡了 删除呼叫用户
  258. {
  259. user_map.erase(iter_m_map--);
  260. }
  261. }
  262. }
  263. /**
  264. * @brief 获取所有的呼叫卡
  265. * @param out_data
  266. */
  267. void get_all_call_cards(std::vector<call_card_ptr>& out_data)
  268. {
  269. auto cardlist = card_list::instance()->m_map;
  270. {
  271. std::lock_guard<std::mutex> lock(_mutex);
  272. get_all_call_cards(_map, cardlist, out_data);
  273. }
  274. }
  275. /*
  276. * @brief
  277. * ["fjb", // 发起呼叫人
  278. * "0010000000001", // 呼叫的卡号
  279. * 103, // 分站号
  280. * 23431, // 发起呼叫时间
  281. * 0, //
  282. * 0 // 正在呼叫:2/结束呼叫/
  283. * ]
  284. */
  285. static void to_node_element(rapidjson::Value& out_elemet,
  286. const std::shared_ptr<call_card> card_ptr,
  287. rapidjson::Document::AllocatorType& allocator)
  288. {
  289. tool_json::push_back(out_elemet, card_ptr->user_name, allocator);
  290. tool_json::push_back(out_elemet, card_ptr->to_id64_str(), allocator);
  291. out_elemet.PushBack(card_ptr->stationid, allocator);
  292. out_elemet.PushBack(card_ptr->call_time_interval, allocator);
  293. out_elemet.PushBack(card_ptr->call_type_id, allocator);
  294. out_elemet.PushBack(card_ptr->call_state, allocator);
  295. }
  296. static void to_node_array(rapidjson::Value& out_array,
  297. const std::vector<std::shared_ptr<call_card>> cards,
  298. rapidjson::Document::AllocatorType& allocator)
  299. {
  300. std::vector<std::shared_ptr<call_card>>::size_type i;
  301. for(i=0; i<cards.size(); i++)
  302. {
  303. if(!cards[i]->is_display)
  304. {
  305. continue;
  306. }
  307. rapidjson::Value node_card(rapidjson::kArrayType);
  308. to_node_element(node_card, cards[i], allocator);
  309. out_array.PushBack(node_card, allocator);
  310. }
  311. }
  312. /**
  313. * @brief 生成所有呼叫卡的json
  314. * @param cards
  315. * @return
  316. */
  317. std::string to_call_card_list_json(std::vector<call_card_ptr> cards);
  318. void get_site_map(const std::vector<call_card_ptr>& arr, call_site_map& out_site_map)
  319. {
  320. auto card_ptr = arr.begin();
  321. for(;card_ptr!=arr.end();++card_ptr)
  322. {
  323. if((*card_ptr)->stationid<1)
  324. {
  325. continue;
  326. }
  327. auto iter = out_site_map.find((*card_ptr)->stationid);
  328. if(out_site_map.end()==iter)
  329. {
  330. call_site_ptr site_ptr(new call_site());
  331. site_ptr->site_id = (*card_ptr)->stationid;
  332. site_ptr->call_type_id = (*card_ptr)->call_type_id;
  333. site_ptr->call_level_id = (*card_ptr)->call_level_id;
  334. out_site_map.insert(std::make_pair(site_ptr->site_id, site_ptr));
  335. iter = out_site_map.find((*card_ptr)->stationid);
  336. }
  337. if(iter->second->is_call_all())
  338. {
  339. continue;
  340. }
  341. if((*card_ptr)->is_call_all())
  342. {
  343. iter->second->call_type_id = (*card_ptr)->call_type_id;
  344. iter->second->cards.clear();
  345. continue;
  346. }
  347. iter->second->cards[(*card_ptr)->cardid]=(*card_ptr);
  348. }
  349. }
  350. ///发呼叫的报文数据给分站
  351. void send_to_sites(call_site_map& site_map)
  352. {
  353. auto iter_site = site_map.begin();
  354. for(; iter_site != site_map.end(); ++iter_site)
  355. {
  356. if(iter_site->second->cards.size() > 254)
  357. {
  358. log_error("呼叫:分站下卡数量过多:分站id=%d, 卡数=%d",
  359. iter_site->first, iter_site->second->cards.size());
  360. continue;
  361. }
  362. std::vector<char> arr;
  363. //分站地址 4字节,支持大小分站呼叫协议;added by zhuyf 2018/06/04
  364. //大小分站的全员呼叫,分站id要求4字节全为FF
  365. uint32_t anchor_id = (iter_site->second->is_call_all()? 0xffffffff : static_cast<uint32_t>(iter_site->first));
  366. memcpy_uint32(arr,anchor_id);
  367. //卡数
  368. arr.push_back(static_cast<char>(iter_site->second->cards.size()));
  369. //呼叫类型
  370. arr.push_back(static_cast<char>(iter_site->second->call_type_id));
  371. if(!iter_site->second->is_call_all())
  372. {
  373. //循环添加卡信息
  374. auto it_card = iter_site->second->cards.begin();
  375. for(; it_card != iter_site->second->cards.end(); ++it_card)
  376. {
  377. //呼叫类型
  378. arr.push_back(static_cast<char>(it_card->second->call_level_id));
  379. //卡类型
  380. arr.push_back(static_cast<char>(it_card->second->cardtype));
  381. //卡ID长度
  382. int id_len = 4;
  383. arr.push_back(static_cast<char>(id_len));
  384. //卡ID
  385. memcpy_uint32(arr, it_card->second->cardid);
  386. }
  387. }
  388. auto sit_ptr = sit_list::instance()->get(iter_site->first);
  389. if(!sit_ptr || !sit_ptr->m_clt)
  390. {
  391. log_error("在全局分站列表中找不到分站 或者sit_ptr->m_clt==null:分站id=%d", iter_site->first);
  392. continue;
  393. }
  394. //std::vector<char> arr2 = arr;
  395. sit_ptr->m_clt->send(std::move(arr));
  396. //print_test(std::move(arr2), iter_site->first);
  397. }
  398. }
  399. static void print_test(std::vector<char>&& arr, int siteid)
  400. {
  401. std::string str("呼叫发送的数据帧,分站号=");
  402. str.append(std::to_string(siteid));
  403. str.append(":");
  404. char a[4]={0};
  405. for(std::vector<char>::size_type i=0; i<arr.size(); i++)
  406. {
  407. sprintf(a, "%02X ", static_cast<unsigned char>(arr[i]));
  408. str.append(std::string(a));
  409. }
  410. log_info("%s", str.c_str());
  411. }
  412. void static memcpy_uint32(std::vector<char>& arr, uint32_t dwSrc)
  413. {
  414. char bt = HIBYTE(HIWORD(dwSrc));
  415. arr.push_back(bt);
  416. bt = LOBYTE(HIWORD(dwSrc));
  417. arr.push_back(bt);
  418. bt = HIBYTE(LOWORD(dwSrc));
  419. arr.push_back(bt);
  420. bt = LOBYTE(LOWORD(dwSrc));
  421. arr.push_back(bt);
  422. }
  423. static char HIBYTE(uint16_t dwSrc)
  424. {
  425. return static_cast<char>(dwSrc>>8);
  426. }
  427. static char LOBYTE(uint16_t dwSrc)
  428. {
  429. return static_cast<char>(dwSrc);
  430. }
  431. static uint16_t HIWORD(uint32_t dwSrc)
  432. {
  433. return static_cast<uint16_t>(dwSrc>>16);
  434. }
  435. static uint16_t LOWORD(uint32_t dwSrc)
  436. {
  437. return static_cast<uint16_t>(dwSrc);
  438. }
  439. private:
  440. /// 呼叫用户列表
  441. call_user_map _map;
  442. };
  443. #endif // MODULE_CALL_H