暂无描述

CallSession.cpp 11KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350
  1. #include "StdAfx.h"
  2. #include "CallSession.h"
  3. #include "LogicLine.h"
  4. #include "Record.h"
  5. #include "TaskMgr.h"
  6. #include "IvrFlow.h"
  7. #include "IvrFlowHolder.h"
  8. #include "StatisticsMgr.h"
  9. #include "LineHolder.h"
  10. #include "CtiCore.h"
  11. #include "NetworkCti.h"
  12. #include "LogicLineExt.h"
  13. CCallSession::CCallSession(ULONG CallId) : m_CallId(CallId), m_RecFile(_T("")), m_Data(_T(""))
  14. {
  15. }
  16. CCallSession::~CCallSession(void)
  17. {
  18. }
  19. /*****************************************************************
  20. **【函数名称】 __procLineRecord
  21. **【函数功能】 处理线路录音
  22. **【参数】 pRecLine: 录音线路
  23. RecFile:录音文件
  24. **【返回值】
  25. ****************************************************************/
  26. void CCallSession::__procLineRecord( CLogicLine* pRecLine, CString& RecFile )
  27. {
  28. // 设置会话中录音文件名
  29. m_RecFile = RecFile;
  30. // 设置录音线路的录音文件名
  31. pRecLine->setRecordFile(RecFile);
  32. // 设置关联线路的录音文件名
  33. CLogicLine* pLine = NULL;
  34. CLineHolder& LineHolder = CLineHolder::GetInstance();
  35. POSITION pos = m_LineList.GetHeadPosition();
  36. while(pos != NULL)
  37. {
  38. pLine = LineHolder.getLogicLine(m_LineList.GetNext(pos));
  39. ASSERT(pLine != NULL);
  40. if(pLine != NULL && pRecLine->type() != pLine->type() && pLine->recordFile() == _T(""))
  41. pLine->setRecordFile(RecFile);
  42. }
  43. }
  44. /*****************************************************************
  45. **【函数名称】 __onLineTalking
  46. **【函数功能】 开始通话后的处理函数
  47. **【参数】 pLine:通话线路
  48. **【返回值】
  49. ****************************************************************/
  50. void CCallSession::__onLineTalking( CLogicLine* pLine )
  51. {
  52. ASSERT(pLine != NULL);
  53. //如果是监听 不再产生录音文件文件路径
  54. if (pLine->opType() == PDU_CMD_AGENT_MONI_LISTEN) return;
  55. // 对该线路进行录音处理
  56. CString RecFile = "";
  57. if(CRecord::GetInstance().record(pLine, RecFile))
  58. __procLineRecord(pLine, RecFile);
  59. else
  60. pLine->setRecordFile(m_RecFile);
  61. }
  62. /*****************************************************************
  63. **【函数名称】 __onLineHangUp
  64. **【函数功能】 处理线路挂机(离开会话)
  65. **【参数】 pLine:通话线路
  66. **【返回值】
  67. ****************************************************************/
  68. void CCallSession::__onLineHangUp( CLogicLine* pLine )
  69. {
  70. // 通知IVR线路挂机
  71. if (pLine->type() == DEV_RES_TYPE_TRUNK || pLine->type() == DEV_RES_TYPE_VOIP)
  72. {
  73. // 显示日志
  74. ILogger::getInstance().log(LOG_CLASS_BUSI, LOG_LEVEL_NORMAL, _T("{CallSession}: [外线挂机], Trunk = %d,CallId = %d,caller=%s,callee=%s"), pLine->lineId(), m_CallId, pLine->callerNum(), pLine->calleeNum());
  75. // 通知IVR外线挂机
  76. CIvrFlow* pIvrFlow = CIvrFlowHolder::GetInstance().getFlowByLineId(pLine->lineId());
  77. if (pIvrFlow != NULL)
  78. {
  79. ILogger::getInstance().log(LOG_CLASS_BUSI, LOG_LEVEL_NORMAL, _T("{CallSession}: [外线挂机], Trunk = %d,IvrFlow_assoLineId=%d"), pLine->lineId(), pIvrFlow->assoLineId());
  80. pIvrFlow->procLineHangUp();
  81. }
  82. if (pLine->opType() != PDU_CMD_AGENT_FAX && pLine->opType() != PDU_CMD_IVR_FAX)
  83. {
  84. // 通知外线关联Task销毁,不在向IVR发返回消息
  85. CTaskMgr::GetInstance().removeTask(pLine->taskIdBinded());
  86. }
  87. }
  88. // 2021-12-21 返回市民挂机状态
  89. if (pLine->type() == DEV_RES_TYPE_TRUNK && pLine->isFirstTrunk())
  90. {
  91. auto pAssoLine = getAssoLine(pLine);
  92. if (pAssoLine != NULL && pAssoLine->type() == DEV_RES_TYPE_EXT && pAssoLine->isMeeting())
  93. {
  94. CPduEntity cmd(PDU_CMD_AGENT_HANGUP);
  95. cmd.SetToExecReturn();
  96. cmd.SetDataUInt(1, pAssoLine->lineId());
  97. cmd.SetDataUInt(2, pAssoLine->getAgentNum());
  98. CString lineId;
  99. lineId.Format("%u", pAssoLine->lineId());
  100. if (pAssoLine->callerNum() == lineId)
  101. cmd.SetDataString(3, pLine->calleeNum());
  102. else
  103. cmd.SetDataString(3, pLine->callerNum());
  104. CNetworkCti::GetInstance().send2ACD(cmd);
  105. }
  106. }
  107. // 统计挂机
  108. T_EvtHangUp repInfo;
  109. memset(&repInfo, 0, sizeof(repInfo));
  110. repInfo.nHangUpFlag = (m_LineList.GetCount() > 1 ? 0:1); // 0本端先挂机 1对端先挂机
  111. CStatisticsMgr::GetInstance().onCallDetail(m_CallId, REP_EVENT_HANG_UP, pLine->lineId(), &repInfo);
  112. // 移除线路
  113. POSITION pos = m_LineList.Find(pLine->lineId());
  114. if(pos != NULL)
  115. m_LineList.RemoveAt(pos);
  116. // 任务(协商转移)失败时,不重置发起方的逻辑线路信息
  117. if (pLine->opResult() == FALSE && pLine->opType() == PDU_CMD_AGENT_CONSULTATION_CALL)
  118. pLine->opType() = PDU_CMD_UNKNOWN;
  119. else
  120. {
  121. pLine->resetLine(); // 重置线路信息
  122. }
  123. }
  124. void CCallSession::__onSetChannelVar(CLogicLine * pLine, CLogicLine * pTempLine)
  125. {
  126. if (pLine == NULL || pTempLine == NULL) return;
  127. if (pLine->type() == DEV_RES_TYPE_EXT && pLine->isMeeting()) return;
  128. if (pTempLine->type() == DEV_RES_TYPE_EXT && pTempLine->isMeeting()) return; // 会议中不识别
  129. LineOpParam pLineOpParam;
  130. memset(&pLineOpParam, 0, sizeof(pLineOpParam));
  131. sprintf_s(pLineOpParam.szParam3, "%ld", pLine->callId()); // 保存callid
  132. AsrInfo asrInfo;
  133. asrInfo.callId = std::to_string(m_CallId); // callid
  134. asrInfo.callStartTime = GetUnixTime(); // 接通时间
  135. CString str = m_RecFile;
  136. str.Replace('\\',','); // 替换路径中的 \为,,解析时再由逗号转为反斜杠
  137. asrInfo.reference = str.GetBuffer(0); // 录音路径
  138. if (pLine->type() == DEV_RES_TYPE_EXT) //
  139. {
  140. CString lineId;
  141. lineId.Format("%d", pLine->lineId());
  142. if (lineId == pLine->calleeNum()) // 该线路被叫,坐席呼入
  143. {
  144. asrInfo.direction = "CallIn"; // 呼叫方向
  145. }
  146. else
  147. {
  148. asrInfo.direction = "CallOut";
  149. }
  150. asrInfo.agentId = std::to_string(pLine->getAgentNum()); // 坐席id
  151. CString str;
  152. str = pLine->calleeNum();
  153. asrInfo.calledId = str.GetBuffer(0); // 被叫号码
  154. str.ReleaseBuffer();
  155. str = pLine->callerNum();
  156. asrInfo.callerId = str.GetBuffer(0); // 主叫号码
  157. str.ReleaseBuffer();
  158. asrInfo.vocDirect = "Agent"; //角色标识,坐席
  159. std::string asrStr;
  160. asrInfo >> asrStr;
  161. memset(pLineOpParam.szParam4, 0, sizeof(pLineOpParam.szParam4));
  162. strcpy_s(pLineOpParam.szParam4, asrStr.c_str());
  163. CCtiCore::GetInstance().getDevLink().exec(-1, LINE_OP_SET_CHANNEL_VARIABLE, pLine->lineId(), &pLineOpParam);
  164. if (pTempLine->type() == DEV_RES_TYPE_TRUNK) // 另一条是外线
  165. asrInfo.vocDirect = "User"; //角色标识,坐席
  166. asrStr.clear();
  167. asrInfo >> asrStr;
  168. memset(pLineOpParam.szParam4, 0, sizeof(pLineOpParam.szParam4));
  169. strcpy_s(pLineOpParam.szParam4, asrStr.c_str());
  170. CCtiCore::GetInstance().getDevLink().exec(-1, LINE_OP_SET_CHANNEL_VARIABLE, pTempLine->lineId(), &pLineOpParam);
  171. }
  172. else // 当前线路是外线
  173. {
  174. if (pTempLine->type() == DEV_RES_TYPE_TRUNK) return; // 如果两条线路都是外线,不处理
  175. CString lineId;
  176. lineId.Format("%d", pTempLine->lineId());
  177. if (lineId == pTempLine->calleeNum()) // 该线路被叫,坐席呼入
  178. {
  179. asrInfo.direction = "CallIn"; // 呼叫方向
  180. }
  181. else
  182. {
  183. asrInfo.direction = "CallOut";
  184. }
  185. asrInfo.agentId = std::to_string(pTempLine->getAgentNum()); // 坐席id
  186. CString str;
  187. str = pTempLine->calleeNum();
  188. asrInfo.calledId = str.GetBuffer(0); // 被叫号码
  189. str.ReleaseBuffer();
  190. str = pTempLine->callerNum();
  191. asrInfo.callerId = str.GetBuffer(0); // 主叫号码
  192. str.ReleaseBuffer();
  193. asrInfo.vocDirect = "Agent"; //角色标识,坐席
  194. std::string asrStr;
  195. asrInfo >> asrStr;
  196. memset(pLineOpParam.szParam4, 0, sizeof(pLineOpParam.szParam4));
  197. strcpy_s(pLineOpParam.szParam4, asrStr.c_str());
  198. CCtiCore::GetInstance().getDevLink().exec(-1, LINE_OP_SET_CHANNEL_VARIABLE, pTempLine->lineId(), &pLineOpParam);
  199. asrInfo.vocDirect = "User"; //角色标识,坐席
  200. asrStr.clear();
  201. asrInfo >> asrStr;
  202. memset(pLineOpParam.szParam4, 0, sizeof(pLineOpParam.szParam4));
  203. strcpy_s(pLineOpParam.szParam4, asrStr.c_str());
  204. CCtiCore::GetInstance().getDevLink().exec(-1, LINE_OP_SET_CHANNEL_VARIABLE, pLine->lineId(), &pLineOpParam);
  205. }
  206. }
  207. /*****************************************************************
  208. **【函数名称】 getAssoLineExt
  209. **【函数功能】 获取指定线路的关联线路
  210. **【参数】 pHostLine:指定线路
  211. **【返回值】 关联线路对象
  212. ****************************************************************/
  213. CLogicLine* CCallSession::getAssoLine( CLogicLine* pHostLine )
  214. {
  215. if(m_LineList.GetCount() < 2)
  216. return NULL;
  217. CLineHolder& LineHolder = CLineHolder::GetInstance();
  218. POSITION pos = m_LineList.GetHeadPosition();
  219. while(pos != NULL)
  220. {
  221. CLogicLine* pLine = LineHolder.getLogicLine(m_LineList.GetNext(pos));
  222. ASSERT(pLine != NULL);
  223. if(pLine != NULL && pLine != pHostLine)
  224. return pLine;
  225. }
  226. return NULL;
  227. }
  228. /*****************************************************************
  229. **【函数名称】 addLine
  230. **【函数功能】 会话中添加线路
  231. **【参数】 pLine: 待添加线路
  232. **【返回值】
  233. ****************************************************************/
  234. void CCallSession::addLine( CLogicLine *pLine )
  235. {
  236. ASSERT(pLine != NULL);
  237. if(pLine == NULL)
  238. return;
  239. if(m_LineList.Find(pLine->lineId()) != NULL)
  240. return;
  241. m_LineList.AddTail(pLine->lineId());
  242. pLine->callId() = m_CallId;
  243. }
  244. /*****************************************************************
  245. **【函数名称】 onLineStatusUpdated
  246. **【函数功能】 处理线路状态改变
  247. **【参数】 pLine: 状态变化的线路
  248. **【返回值】
  249. ****************************************************************/
  250. void CCallSession::onLineStatusUpdated( CLogicLine* pLine )
  251. {
  252. // 保持状态过滤
  253. UINT Status = (HELD_STATE_FILTER_MASK & pLine->status());
  254. ILogger::getInstance().log(LOG_CLASS_BUSI, LOG_LEVEL_NORMAL, _T("{CallSession}:onLineStatusUpdated, Trunk = %d,CallId = %d,Status=%d"), pLine->lineId(), m_CallId, Status);
  255. // 通话
  256. if(Status == TRUNK_STATE_TALKING || Status == INNER_STATE_TALKING)
  257. {
  258. // 有至少两个线路通话时才录音
  259. CLogicLine* pTempLine = NULL;
  260. CLineHolder& LineHolder = CLineHolder::GetInstance();
  261. POSITION pos = m_LineList.GetHeadPosition();
  262. while(pos)
  263. {
  264. pTempLine = LineHolder.getLogicLine(m_LineList.GetNext(pos));
  265. ASSERT(pTempLine != NULL);
  266. if(pTempLine == NULL || pTempLine == pLine)
  267. continue;
  268. UINT nTempStatus = (HELD_STATE_FILTER_MASK & pTempLine->status());
  269. if(nTempStatus == INNER_STATE_TALKING || nTempStatus == TRUNK_STATE_TALKING)
  270. {
  271. __onLineTalking(pLine);
  272. __onLineTalking(pTempLine);
  273. // 2021-12-3 设置asr通道变量
  274. __onSetChannelVar(pLine, pTempLine);
  275. break;
  276. }
  277. }
  278. }
  279. // 空闲状态当挂机处理
  280. if(Status == TRUNK_STATE_FREE || Status == INNER_STATE_FREE)
  281. __onLineHangUp(pLine);
  282. }
  283. void CCallSession::onRecord(CLogicLine* pLine) {
  284. //ILogger::getInstance().log(LOG_CLASS_SOCKET, LOG_LEVEL_NORMAL, _T("{CallSession},onRecord"));
  285. CLogicLine* pTempLine = NULL;
  286. CLineHolder& LineHolder = CLineHolder::GetInstance();
  287. POSITION pos = m_LineList.GetHeadPosition();
  288. while (pos)
  289. {
  290. pTempLine = LineHolder.getLogicLine(m_LineList.GetNext(pos));
  291. ASSERT(pTempLine != NULL);
  292. if (pTempLine == NULL || pTempLine == pLine)
  293. continue;
  294. UINT nTempStatus = (HELD_STATE_FILTER_MASK & pTempLine->status());
  295. if (nTempStatus == INNER_STATE_TALKING || nTempStatus == TRUNK_STATE_TALKING)
  296. {
  297. ILogger::getInstance().log(LOG_CLASS_SOCKET, LOG_LEVEL_NORMAL, _T("{CallSession},onRecord,%d"), nTempStatus);
  298. __onLineTalking(pLine);
  299. __onLineTalking(pTempLine);
  300. break;
  301. }
  302. else
  303. {
  304. ILogger::getInstance().log(LOG_CLASS_SOCKET, LOG_LEVEL_NORMAL, _T("{CallSession},onRecord,nTempStatus=%d"), nTempStatus);
  305. }
  306. }
  307. }