call.py 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168
  1. #!/usr/bin/env python3
  2. # encoding:utf-8
  3. import json
  4. import time
  5. from datetime import datetime
  6. from src.core.callcenter.agent import AgentMonitorService, AgentActionLogService
  7. from src.core.callcenter.cache import Cache
  8. from src.core.callcenter.constant import saasId, HOLD_MUSIC_PATH
  9. from src.core.callcenter.enumeration import CallCause, Direction, NextType, DeviceType, CdrType, AgentServiceState, \
  10. AgentScene, WorkStatus, AgentLogState, ServiceDirect
  11. from src.core.callcenter.api import AgentCallRequest, CallInfo, HangupCallRequest, CheckInCallRequest, \
  12. DeviceInfo, NextCommand, MakeCallContext
  13. from src.core.callcenter.esl.constant.sip_header_constant import sipHeaderServiceId, sipHeaderCtiFlowId
  14. from src.core.callcenter.snowflake import Snowflake
  15. from src.core.callcenter.push import PushHandler
  16. from src.core.callcenter.data_handler import *
  17. class CallService:
  18. def __init__(self, client, app):
  19. self.client = client
  20. self.logger = app.logger
  21. self.cache = Cache(app)
  22. self.snowflake = Snowflake()
  23. self.push_handler = PushHandler(app.logger)
  24. self.data_handle_server=DataHandleServer(app)
  25. self.agent_monitor_service = AgentMonitorService(app)
  26. self.agent_actionlog_service = AgentActionLogService(app)
  27. # self.push_handler = PushHandler(logger)
  28. def call(self, request: AgentCallRequest):
  29. call_id = 'C' + str(self.snowflake.next_id())
  30. device_id = 'D' + str(self.snowflake.next_id())
  31. # now = lambda: int(round(time.time() * 1000))
  32. now = datetime.utcnow().timestamp()
  33. self.logger.info("CallService 人工外呼 agent:%s makecall, ctiFlowId:%s, callId:%s, callerDisplay:%s, called:%s"%
  34. (request.caller, request.cti_flow_id, call_id, request.caller_display, request.called))
  35. agent = self.cache.get_agent_info(request.saas_id, request.agent_id)
  36. route_gateway = self.cache.get_route_gateway(request.saas_id)
  37. call_info = CallInfo(cti_flow_id=request.cti_flow_id, call_id=call_id, agent_key=agent.agent_number, sip_server=agent.sip_server,
  38. caller=agent.agent_number, called=request.called, direction=Direction.OUTBOUND.code,
  39. caller_display=request.caller_display, called_display=request.called_display,
  40. call_type=request.call_type, call_time=now, follow_data=request.follow_data,
  41. uuid1=request.uuid1, uuid2=request.uuid2, saas_id=saasId,
  42. core_uuid=None, conference=None, group_id=None, hidden_customer=0, number_location=None, agent_name=None, login_type=None, ivr_id=None, task_id=None, media_host=None, client_host=None, record=None, record2=None, record_time=None, answer_flag=None, wait_time=None, answer_count=0, hangup_dir=None, sdk_hangup=0, hangup_code=None, answer_time=None, end_time=None, talk_time=None, first_queue_time=None, queue_start_time=None, queue_end_time=None, overflow_count=0, cdr_notify_url=None, queue_level=None, transfer_agent=None, device_list=[], device_info_map = {}, process_data = {}, next_commands=[], call_details=[])
  43. device_info = DeviceInfo(cti_flow_id=request.cti_flow_id, device_id=device_id, call_time=now, call_id=call_id, device_type=DeviceType.AGENT.code,
  44. agent_key=agent.agent_number, caller_display=route_gateway.name, cdr_type=CdrType.INBOUND.code,
  45. conference=None, agent_name=None, from_agent=None, caller=None, called=None, display=None, called_location=None, caller_location=None, ring_start_time=None, ring_end_time=None, answer_time=None, bridge_time=None, end_time=None, talk_time=None, sip_protocol=None, channel_name=None, hangup_cause=None, ring_cause=None, sip_status=None, record=None, record_time=None, record_start_time=None, state=None, apparent_number=None,
  46. )
  47. call_info.device_list.append(device_id)
  48. self.logger.info("liuwei::debugger:1111::call_id=%s, device_id=%s"%(call_id, device_id))
  49. call_info.next_commands.append(NextCommand(device_id, NextType.NEXT_CALL_OTHER.code))
  50. call_info.device_info_map = {device_id: device_info}
  51. self.cache.add_call_info(call_info)
  52. context = MakeCallContext(display=request.called, caller=request.called, called=request.caller,
  53. call_id=call_id, device_id=device_id, device_type=device_info.device_type,
  54. call_type=call_info.call_type, sip_server=call_info.sip_server,
  55. sip_header_map={sipHeaderCtiFlowId: request.cti_flow_id})
  56. self.client.make_call_new(context)
  57. self.do_after_manual_call(call_info, agent.agent_number)
  58. # # 创建一条通话记录
  59. # self.dataHandleServer.create_record({
  60. # "session_id": call_id,
  61. # "time_begin": datetime.utcnow(),
  62. # "category": 1,
  63. # "agent_num":request.agent_id,
  64. # "phone": request.called
  65. # })
  66. # # 变更坐席状态为拨号中
  67. # self.dataHandleServer.update_agent_monitor_service_state(request.agent_id, AgentServiceState.DIALING.code)
  68. # self.push_handler.push_on_agent_work_report(request.saas_id, request.cti_flow_id, request.agent_id, call_id,AgentScene.ROBOT, WorkStatus.AGENT_DIALING)
  69. return call_id
  70. def do_after_manual_call(self, call_info: CallInfo, agent_id):
  71. agent_monitor = self.data_handle_server.get_agent_monitor(call_info.saas_id, agent_number=agent_id)
  72. self.agent_monitor_service.update_dialing(agent_monitor)
  73. self.push_handler.push_on_call_ring(call_info.saas_id, flow_id=call_info.cti_flow_id, user_id=agent_id, scene=AgentScene.MANUAL, call_id=call_info.call_id, service_direct=ServiceDirect.MANUAL_CALL.service_direct)
  74. self.agent_actionlog_service.insert_service_state(agent_monitor, AgentServiceState.DIALING, AgentLogState.DIALING)
  75. def hold(self, call_info: CallInfo, device_id):
  76. devices = call_info.device_list
  77. try:
  78. devices.remove(device_id)
  79. except:
  80. pass
  81. custom_device_id = devices[0]
  82. self.logger.info('hold, custom_device_id=%s'%custom_device_id)
  83. self.client.bridge_break(call_info.call_id, custom_device_id)
  84. self.cache.set_need_play_hold_music(call_info.call_id)
  85. self.logger.info('hold success custom_device_id=%s'%custom_device_id)
  86. def cancel_hold(self, call_info: CallInfo, device_id):
  87. self.client.bridge_call(call_info.call_id, call_info.device_list[0], call_info.device_list[1])
  88. def transfer(self, call_info: CallInfo, agent_number, service_id):
  89. caller = call_info.called
  90. call_id = call_info.call_id
  91. agent = self.cache.get_agent_info(call_info.saas_id, agent_number)
  92. device_id = 'T' + str(self.snowflake.next_id())
  93. # now = lambda: int(round(time.time() * 1000))
  94. now = datetime.utcnow().timestamp()
  95. device_info = DeviceInfo(device_id=device_id, caller=caller, display=caller, called=agent_number, call_id=call_id,
  96. call_time=now, cdr_type=CdrType.TRANSFER.code, device_type=DeviceType.AGENT.code,
  97. cti_flow_id=None, agent_key=None, agent_name=None, from_agent=None, called_location=None, caller_location=None, ring_start_time=None, ring_end_time=None, answer_time=None, bridge_time=None, end_time=None, talk_time=None, sip_protocol=None, channel_name=None, hangup_cause=None, ring_cause=None, sip_status=None, record=None, record_time=None, record_start_time=None, state=None, apparent_number=None, caller_display=None)
  98. call_info.device_list.append(device_id)
  99. self.logger.info("liuwei::debugger:2222::call_id=%s, device_id=%s" % (call_id, device_id))
  100. # call_info.caller = agent_number
  101. call_info.device_info_map[device_id] = device_info
  102. call_info.next_commands.append(NextCommand(device_info.device_id, NextType.NEXT_TRANSFER_CALL.code, call_info.device_list[0]))
  103. call_info.agent_key = agent_number
  104. self.logger.info('transfer, agent_number=%s, device_id=%s, call_info=%s'% (agent_number, device_id, call_info))
  105. # agent.sip_server
  106. self.cache.add_call_info(call_info)
  107. self.cache.add_agent_info(agent=agent, call_id=call_id, device_id=device_id)
  108. sip_header_map = {sipHeaderServiceId: service_id}
  109. context = MakeCallContext(display=call_info.caller, caller=call_info.caller, called=agent_number,
  110. call_id=call_id, device_id=device_id, device_type=device_info.device_type,sip_server=agent.sip_server,
  111. call_type=call_info.call_type, service_id=service_id, sip_header_map=sip_header_map)
  112. self.client.make_call_new(context)
  113. def hangup_by_scene(self, request: HangupCallRequest):
  114. scene = AgentScene.get_by_code(request.scene)
  115. if scene and AgentScene.MANUAL == scene:
  116. self.do_manual_hang(request)
  117. elif scene and not AgentScene.MANUAL == scene:
  118. self.do_robot_hang(request)
  119. def do_manual_hang(self, request: HangupCallRequest):
  120. self.hangup_call(request.call_id)
  121. agent_monitor = self.data_handle_server.get_agent_monitor(saas_id=request.saas_id, agent_number=request.agent_id)
  122. self.agent_actionlog_service.insert_service_state(agent_monitor, AgentServiceState.HANGING, AgentLogState.MANUAL_HANG_UP)
  123. def do_robot_hang(self, request: HangupCallRequest):
  124. self.hangup_call(request.call_id)
  125. def hangup_all(self, call_info: CallInfo, case_enum=CallCause.DEFAULT):
  126. devices = call_info.device_list
  127. if not devices:
  128. self.logger.info('hangupCallAll skip 已全部挂断 callId: %s', call_info.call_id)
  129. return
  130. self.logger.info("hangup_all, call_id:%s, devices=%s, case=%s"%(call_info.call_id, json.dumps(devices), case_enum))
  131. for device in devices:
  132. self.client.kill_call(call_info.call_id, device, case_enum)
  133. def hangup_call(self, call_id):
  134. call_info = self.cache.get_call_info(call_id)
  135. if not call_info:
  136. self.logger.info('hangup call not exist callId: %s', call_id)
  137. return
  138. devices = call_info.device_list
  139. if not devices:
  140. self.logger.info('hangup deviceList is null callId: %s', call_id)
  141. return
  142. self.logger.info("hangup_all, call_id:%s, devices=%s" % (call_info.call_id, json.dumps(devices)))
  143. for device in devices:
  144. self.client.kill_call(call_info.call_id, device, CallCause.RESTART)
  145. def checkin_call(self, request: CheckInCallRequest):
  146. agent = self.cache.get_agent_info(request.saas_id, request.agent_number)
  147. return self.client.show_channel(agent.device_id)