1
0

module_call.h 14 KB

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