module_call.cpp 20 KB

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