module_call.h 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514
  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 card_list::to_id64(cardtype, cardid);
  97. }
  98. std::string to_id64_str()
  99. {
  100. return card_list::to_id64_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 init(config_file& config)
  175. {
  176. sleep_ms = std::stoi(config.get("service.send_call_interval","5000"));
  177. }
  178. private:
  179. /**
  180. * @brief 采集回复web发起呼叫
  181. * @param arr
  182. */
  183. void response_accept_call(const std::vector<call_card_ptr> arr);
  184. /**
  185. * @brief 采集回复web:取消呼叫
  186. * @param node_data
  187. */
  188. void response_accept_cancel(const call_user_ptr user_ptr);
  189. void add_cards_to_user(const std::vector<sio::message::ptr>& card_vec,
  190. call_user_ptr user_ptr);
  191. /**
  192. * @brief 线程函数
  193. */
  194. void run();
  195. void copy(const call_user_ptr src, call_user_ptr des)
  196. {
  197. des->call_level_id = src->call_level_id;
  198. des->call_time = src->call_time;
  199. des->call_time_out = src->call_time_out;
  200. des->call_type_id = src->call_type_id;
  201. des->user_name = src->user_name;
  202. }
  203. void copy(const call_card_ptr src, call_card_ptr des)
  204. {
  205. des->call_level_id = src->call_level_id;
  206. des->call_state = src->call_state;
  207. des->call_time = src->call_time;
  208. des->call_time_out = src->call_time_out;
  209. des->cardid = src->cardid;
  210. des->cardtype = src->cardtype;
  211. des->user_name = src->user_name;
  212. des->stationid = src->stationid;
  213. des->call_type_id = src->call_type_id;
  214. }
  215. void copy(const call_user_ptr src, call_card_ptr des)
  216. {
  217. des->call_level_id = src->call_level_id;
  218. //des->call_state = src->call_state;
  219. des->call_time = src->call_time;
  220. des->call_time_out = src->call_time_out;
  221. des->call_time_interval = src->call_time_interval;
  222. //des->cardid = src->cardid;
  223. des->user_name = src->user_name;
  224. des->call_type_id = src->call_type_id;
  225. }
  226. /**
  227. * @brief 获取用户下的所有呼叫卡
  228. * @param user_ptr
  229. * @param cardlist
  230. * @param out_data
  231. */
  232. void get_user_all_call_cards(call_user_ptr& user_ptr,
  233. const std::unordered_map<uint64_t,std::shared_ptr<card_location_base>>& cardlist,
  234. std::vector<call_card_ptr>& out_data);
  235. /**
  236. * @brief 获取所有的呼叫卡
  237. * @param user_map
  238. * @param cardlist
  239. * @param out_data
  240. */
  241. void get_all_call_cards(call_user_map& user_map,
  242. const std::unordered_map<uint64_t,std::shared_ptr<card_location_base>>& cardlist,
  243. std::vector<call_card_ptr>& out_data)
  244. {
  245. auto iter_m_map=user_map.begin();
  246. for(;iter_m_map!=user_map.end();++iter_m_map)
  247. {
  248. get_user_all_call_cards(iter_m_map->second, cardlist, out_data);
  249. if(iter_m_map->second->cards.empty())//没有卡了 删除呼叫用户
  250. {
  251. user_map.erase(iter_m_map--);
  252. }
  253. }
  254. }
  255. /**
  256. * @brief 获取所有的呼叫卡
  257. * @param out_data
  258. */
  259. void get_all_call_cards(std::vector<call_card_ptr>& out_data)
  260. {
  261. auto cardlist = card_list::instance()->m_map;
  262. {
  263. std::lock_guard<std::mutex> lock(_mutex);
  264. get_all_call_cards(_map, cardlist, out_data);
  265. }
  266. }
  267. /*
  268. * @brief
  269. * ["fjb", // 发起呼叫人
  270. * "0010000000001", // 呼叫的卡号
  271. * 103, // 分站号
  272. * 23431, // 发起呼叫时间
  273. * 0, //
  274. * 0 // 正在呼叫:2/结束呼叫/
  275. * ]
  276. */
  277. static void to_node_element(rapidjson::Value& out_elemet,
  278. const std::shared_ptr<call_card> card_ptr,
  279. rapidjson::Document::AllocatorType& allocator)
  280. {
  281. tool_json::push_back(out_elemet, card_ptr->user_name, allocator);
  282. tool_json::push_back(out_elemet, card_ptr->to_id64_str(), allocator);
  283. out_elemet.PushBack(card_ptr->stationid, allocator);
  284. out_elemet.PushBack(card_ptr->call_time_interval, allocator);
  285. out_elemet.PushBack(card_ptr->call_type_id, allocator);
  286. out_elemet.PushBack(card_ptr->call_state, allocator);
  287. }
  288. static void to_node_array(rapidjson::Value& out_array,
  289. const std::vector<std::shared_ptr<call_card>> cards,
  290. rapidjson::Document::AllocatorType& allocator)
  291. {
  292. std::vector<std::shared_ptr<call_card>>::size_type i;
  293. for(i=0; i<cards.size(); i++)
  294. {
  295. if(!cards[i]->is_display)
  296. {
  297. continue;
  298. }
  299. rapidjson::Value node_card(rapidjson::kArrayType);
  300. to_node_element(node_card, cards[i], allocator);
  301. out_array.PushBack(node_card, allocator);
  302. }
  303. }
  304. /**
  305. * @brief 生成所有呼叫卡的json
  306. * @param cards
  307. * @return
  308. */
  309. std::string to_call_card_list_json(std::vector<call_card_ptr> cards);
  310. void get_site_map(const std::vector<call_card_ptr>& arr, call_site_map& out_site_map)
  311. {
  312. auto card_ptr = arr.begin();
  313. for(;card_ptr!=arr.end();++card_ptr)
  314. {
  315. if((*card_ptr)->stationid<1)
  316. {
  317. continue;
  318. }
  319. auto iter = out_site_map.find((*card_ptr)->stationid);
  320. if(out_site_map.end()==iter)
  321. {
  322. call_site_ptr site_ptr(new call_site());
  323. site_ptr->site_id = (*card_ptr)->stationid;
  324. site_ptr->call_type_id = (*card_ptr)->call_type_id;
  325. site_ptr->call_level_id = (*card_ptr)->call_level_id;
  326. out_site_map.insert(std::make_pair(site_ptr->site_id, site_ptr));
  327. iter = out_site_map.find((*card_ptr)->stationid);
  328. }
  329. if(iter->second->is_call_all())
  330. {
  331. continue;
  332. }
  333. if((*card_ptr)->is_call_all())
  334. {
  335. iter->second->call_type_id = (*card_ptr)->call_type_id;
  336. iter->second->cards.clear();
  337. continue;
  338. }
  339. iter->second->cards[(*card_ptr)->cardid]=(*card_ptr);
  340. }
  341. }
  342. ///发呼叫的报文数据给分站
  343. void send_to_sites(call_site_map& site_map)
  344. {
  345. auto iter_site = site_map.begin();
  346. for(; iter_site != site_map.end(); ++iter_site)
  347. {
  348. if(iter_site->second->cards.size() > 254)
  349. {
  350. log_error("呼叫:分站下卡数量过多:分站id=%d, 卡数=%d",
  351. iter_site->first, iter_site->second->cards.size());
  352. continue;
  353. }
  354. std::vector<char> arr;
  355. //分站地址 4字节,支持大小分站呼叫协议;added by zhuyf 2018/06/04
  356. //大小分站的全员呼叫,分站id要求4字节全为FF
  357. uint32_t anchor_id = (iter_site->second->is_call_all()? 0xffffffff : static_cast<uint32_t>(iter_site->first));
  358. memcpy_uint32(arr,anchor_id);
  359. //卡数
  360. arr.push_back(static_cast<char>(iter_site->second->cards.size()));
  361. //呼叫类型
  362. arr.push_back(static_cast<char>(iter_site->second->call_type_id));
  363. if(!iter_site->second->is_call_all())
  364. {
  365. //循环添加卡信息
  366. auto it_card = iter_site->second->cards.begin();
  367. for(; it_card != iter_site->second->cards.end(); ++it_card)
  368. {
  369. //呼叫类型
  370. arr.push_back(static_cast<char>(it_card->second->call_level_id));
  371. //卡类型
  372. arr.push_back(static_cast<char>(it_card->second->cardtype));
  373. //卡ID长度
  374. int id_len = 4;
  375. arr.push_back(static_cast<char>(id_len));
  376. //卡ID
  377. memcpy_uint32(arr, it_card->second->cardid);
  378. }
  379. }
  380. auto sit_ptr = sit_list::instance()->get(iter_site->first);
  381. if(!sit_ptr || !sit_ptr->m_clt)
  382. {
  383. log_error("在全局分站列表中找不到分站 或者sit_ptr->m_clt==null:分站id=%d", iter_site->first);
  384. continue;
  385. }
  386. //std::vector<char> arr2 = arr;
  387. sit_ptr->m_clt->send(std::move(arr));
  388. //print_test(std::move(arr2), iter_site->first);
  389. }
  390. }
  391. static void print_test(std::vector<char>&& arr, int siteid)
  392. {
  393. std::string str("呼叫发送的数据帧,分站号=");
  394. str.append(std::to_string(siteid));
  395. str.append(":");
  396. char a[4]={0};
  397. for(std::vector<char>::size_type i=0; i<arr.size(); i++)
  398. {
  399. sprintf(a, "%02X ", static_cast<unsigned char>(arr[i]));
  400. str.append(std::string(a));
  401. }
  402. log_info("%s", str.c_str());
  403. }
  404. void static memcpy_uint32(std::vector<char>& arr, uint32_t dwSrc)
  405. {
  406. char bt = HIBYTE(HIWORD(dwSrc));
  407. arr.push_back(bt);
  408. bt = LOBYTE(HIWORD(dwSrc));
  409. arr.push_back(bt);
  410. bt = HIBYTE(LOWORD(dwSrc));
  411. arr.push_back(bt);
  412. bt = LOBYTE(LOWORD(dwSrc));
  413. arr.push_back(bt);
  414. }
  415. static char HIBYTE(uint16_t dwSrc)
  416. {
  417. return static_cast<char>(dwSrc>>8);
  418. }
  419. static char LOBYTE(uint16_t dwSrc)
  420. {
  421. return static_cast<char>(dwSrc);
  422. }
  423. static uint16_t HIWORD(uint32_t dwSrc)
  424. {
  425. return static_cast<uint16_t>(dwSrc>>16);
  426. }
  427. static uint16_t LOWORD(uint32_t dwSrc)
  428. {
  429. return static_cast<uint16_t>(dwSrc);
  430. }
  431. private:
  432. /// 呼叫用户列表
  433. call_user_map _map;
  434. };
  435. #endif // MODULE_CALL_H