1
0

module_call.h 14 KB

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