agent.py 49 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961
  1. #!/usr/bin/env python3
  2. # encoding:utf-8
  3. import threading
  4. import time
  5. import traceback
  6. from collections import defaultdict
  7. from concurrent.futures import ThreadPoolExecutor
  8. from typing import List
  9. from sqlalchemy import or_
  10. from src.core.callcenter import registry
  11. import src.core.callcenter.esl.utils.esl_event_util as EslEventUtil
  12. from src.core import with_app_context
  13. from src.core.callcenter.api import AgentActionRequest, AgentQueryRequest, AgentRequest, AgentEventData, \
  14. AgentStateData, HumanServiceQueryRequest, AgentMonitorData, CallInfo, DeviceInfo, AgentDelayStateData
  15. from src.core.callcenter.cache import Cache
  16. from src.core.callcenter.dao import *
  17. from src.core.callcenter.data_handler import DataHandleServer
  18. from src.core.callcenter.enumeration import AgentState, AgentCheck, AgentHeartState, AgentServiceState, AgentLogState, \
  19. AgentScene, BizErrorCode, WorkStatus, DownEvent, HumanState, DeviceType, ServiceDirect
  20. from src.core.callcenter.esl.constant.event_names import *
  21. from src.core.callcenter.exception import BizException
  22. from src.core.callcenter.push import PushHandler
  23. from src.core.datasource import RedisHandler
  24. class AgentEventService:
  25. def __init__(self, app):
  26. self.app = app
  27. self.logger = app.logger
  28. self.cache = Cache(app)
  29. self.push_handler = PushHandler(app.logger)
  30. self.data_handle_server = DataHandleServer(app)
  31. self.agent_monitor_service = AgentMonitorService(app)
  32. self.agent_state_service = AgentStateService(app)
  33. self.agent_actionlog_service = AgentActionLogService(app)
  34. def delay_state(self, state_data: AgentDelayStateData):
  35. agent = self.data_handle_server.get_agent(state_data.saas_id, state_data.agent_num)
  36. if not agent:
  37. return
  38. agent_monitor = self.data_handle_server.get_agent_monitor(state_data.saas_id, state_data.agent_num)
  39. if not agent_monitor:
  40. return
  41. #TODO 非最新通话的延迟事件,忽略.
  42. agent_scene = AgentScene.get_by_code(state_data.scene)
  43. self.logger.info("agent event delay state %s %s %s %s", state_data.saas_id, state_data.agent_num, state_data.service_state, agent_monitor.service_state)
  44. if AgentServiceState.REPROCESSING.code == state_data.service_state and AgentServiceState.REPROCESSING.code == agent_monitor.service_state:
  45. self.reprocessing_idle(state_data)
  46. if AgentServiceState.DIALING.code == state_data.service_state and AgentServiceState.DIALING.code == agent_monitor.service_state:
  47. if self.cache.get_call_is_answer(state_data.saas_id, state_data.agent_num):
  48. return
  49. self.agent_monitor_service.update_idle(agent_monitor)
  50. self.push_handler.push_on_call_end(state_data.saas_id, state_data.flow_id, state_data.agent_num, agent_scene, ServiceDirect.MANUAL_CALL.service_direct, '0')
  51. self.push_handler.push_on_agent_work_report(state_data.saas_id, state_data.flow_id, state_data.agent_num, '', agent_scene, WorkStatus.AGENT_HANG_REPROCESSING)
  52. self.push_handler.push_on_agent_work_report(state_data.saas_id, state_data.flow_id, state_data.agent_num, '', agent_scene, WorkStatus.AGENT_HANG_IDLE)
  53. self.agent_actionlog_service.insert_service_state(agent_monitor, AgentServiceState.IDLE, AgentLogState.BIZ_DIALING_IDLE)
  54. if AgentServiceState.HANGING.code == state_data.service_state \
  55. and (AgentServiceState.DIALING.code == agent_monitor.service_state
  56. or AgentServiceState.CALLING.code == agent_monitor.service_state):
  57. self.agent_monitor_service.update_idle(agent_monitor)
  58. if AgentServiceState.DIALING.code == agent_monitor.service_state:
  59. self.push_handler.push_on_call_end(state_data.saas_id, state_data.flow_id, state_data.agent_num, agent_scene, ServiceDirect.MANUAL_CALL.service_direct, '0')
  60. self.push_handler.push_on_agent_work_report(state_data.saas_id, state_data.flow_id, state_data.agent_num, '', agent_scene, WorkStatus.AGENT_HANG_REPROCESSING)
  61. self.push_handler.push_on_agent_work_report(state_data.saas_id, state_data.flow_id, state_data.agent_num, '', agent_scene, WorkStatus.AGENT_HANG_IDLE)
  62. self.agent_actionlog_service.insert_service_state(agent_monitor, AgentServiceState.IDLE, AgentLogState.MANUAL_HANG_UP)
  63. def agent_event_channel(self, event, call_info: CallInfo, device_info: DeviceInfo):
  64. event_name = EslEventUtil.getEventName(event)
  65. saas_id = call_info.saas_id if call_info else None
  66. flow_id = call_info.cti_flow_id if call_info else None
  67. call_id = call_info.call_id if call_info else None
  68. device_id = device_info.device_id if device_info else None
  69. agent_num = device_info.agent_key if device_info else None
  70. caller = device_info.caller if device_info else None
  71. called = device_info.called if device_info else None
  72. is_agent = (device_info and DeviceType.AGENT.code == device_info.device_type) if device_info else False
  73. start_time = time.time()
  74. try:
  75. self.logger.info('agent_event_channel, event_name=%s, call_id=%s, device_id=%s, is_agent=%s', event_name, call_id, device_id, is_agent)
  76. agent = self.data_handle_server.get_agent(saas_id, agent_num)
  77. if not agent:
  78. # self.logger.warn("event service channel agent is null %s %s %s %s %s", saas_id, event_name, caller, called, json.loads(event.serialize('json')))
  79. return
  80. agent_monitor = self.data_handle_server.get_agent_monitor(saas_id, agent_num)
  81. if not agent_monitor:
  82. # self.logger.warn("event service channel agentMonitor is null %s %s %s %s %s", saas_id, event_name, caller, called, json.loads(event.serialize('json')))
  83. return
  84. # 信道发起事件,触发完成发起(或桥)&& 坐席侧
  85. if CHANNEL_ORIGINATE == event_name and is_agent:
  86. self.push_handler.push_on_agent_work_report(saas_id, flow_id, agent_num, call_id, AgentScene.MANUAL, WorkStatus.AGENT_RINGING,phone=call_info.caller)
  87. # 进度事件,外呼时对方提醒。或者入呼时提醒 && 坐席侧
  88. if CHANNEL_PROGRESS == event_name and is_agent:
  89. self.push_handler.push_on_agent_report(saas_id, agent_num, AgentScene.MANUAL, AgentServiceState.DIALING)
  90. self.push_handler.push_on_agent_work_report(saas_id, flow_id, agent_num, call_id, AgentScene.MANUAL, WorkStatus.AGENT_CALLING)
  91. # 媒体进度事件,外呼时对方提醒。或者入呼时提醒 && 用户侧
  92. if CHANNEL_PROGRESS_MEDIA == event_name and not is_agent:
  93. self.push_handler.push_on_agent_work_report(saas_id, flow_id, agent_num, call_id, AgentScene.MANUAL, WorkStatus.AGENT_CALLING_RINGING)
  94. #应答
  95. if CHANNEL_ANSWER == event_name:
  96. if call_id:
  97. if self.cache.get_call_is_end(call_id):
  98. # self.logger.warn("event service channel call is end {} {} {} {} {} {}", saas_id, event_name, caller, called, call_id, json.dumps(event.serialize('json')))
  99. return
  100. self.agent_state_service.busy(saas_id, agent.agent_num, agent.phone_num)
  101. self.data_handle_server.update_record(call_id, status=1)
  102. if is_agent:
  103. # 坐席接起
  104. self.cache.set_call_is_answer(saas_id, flow_id)
  105. self.push_handler.push_on_agent_work_report(saas_id, flow_id, agent_num, call_id, AgentScene.MANUAL, WorkStatus.ANSWER_COMPENSATE)
  106. self.push_handler.push_answer_call(saas_id, flow_id, agent_num, call_id, AgentScene.MANUAL, ServiceDirect.MANUAL_CALL.service_direct, WorkStatus.AGENT_HANG_REPROCESSING)
  107. self.agent_actionlog_service.insert_service_state(agent_monitor, AgentServiceState.CALLING, AgentLogState.CHANNEL_TURN_ON)
  108. else:
  109. # 用户侧接起
  110. self.agent_monitor_service.update_calling(agent_monitor)
  111. self.push_handler.push_on_agent_work_report(saas_id, flow_id, agent_num, call_id, AgentScene.MANUAL, WorkStatus.AGENT_ANSWER_OUTGOING)
  112. self.push_handler.push_on_agent_report(saas_id, agent_num, AgentScene.MANUAL, AgentServiceState.CALLING)
  113. #挂断
  114. if CHANNEL_HANGUP == event_name:
  115. # 坐席侧挂断
  116. if is_agent:
  117. if call_id:
  118. self.cache.set_call_is_end(call_id)
  119. self.agent_monitor_service.update_processing(agent_monitor)
  120. self.reprocessing_idle(AgentDelayStateData(saas_id, flow_id, agent_num, AgentServiceState.REPROCESSING, AgentScene.MANUAL))
  121. self.push_handler.push_on_call_end(saas_id, flow_id, agent_num, AgentScene.MANUAL, ServiceDirect.MANUAL_CALL.service_direct, '0')
  122. self.push_handler.push_on_agent_work_report(saas_id, flow_id, agent_num, call_id, AgentScene.MANUAL, WorkStatus.AGENT_HANG_REPROCESSING)
  123. self.push_handler.push_on_agent_report(saas_id, agent_num, AgentScene.MANUAL, AgentServiceState.REPROCESSING)
  124. self.agent_actionlog_service.insert_service_state(agent_monitor, AgentServiceState.REPROCESSING, AgentLogState.CHANNEL_HANG_UP)
  125. # self.data_handle_server.update_record(call_id, time_end=datetime.now())
  126. # 同步处理后处理置闲
  127. # reprocessingIdle(statusDto);
  128. # agentProducer.pushDelayedStatus(statusDto, reprocessingTimeout);
  129. if (CHANNEL_BRIDGE == event_name or PLAYBACK_START == event_name) and is_agent:
  130. self.push_handler.push_on_ring_start(saas_id, flow_id, agent_num, AgentScene.MANUAL, call_id)
  131. # self.push_handler.push_on_agent_work_report(saas_id, flow_id, agent_num, call_id, AgentScene.MANUAL, WorkStatus.AGENT_ANSWER_OUTGOING)
  132. # self.push_handler.push_on_agent_report(saas_id, agent_num, AgentScene.MANUAL, AgentServiceState.CALLING)
  133. if DETECTED_TONE == event_name and not is_agent:
  134. self.push_handler.push_on_detected_tone(saas_id, flow_id, call_id, AgentScene.MANUAL, call_id)
  135. if (CHANNEL_UNBRIDGE == event_name or PLAYBACK_STOP == event_name) and is_agent:
  136. self.push_handler.push_on_ring_end(saas_id, flow_id, call_id, AgentScene.MANUAL, call_id)
  137. except:
  138. traceback.print_exc()
  139. finally:
  140. latency = (time.time() - start_time)
  141. registry.ESL_EVENT_CALLBACK_LATENCY.labels(event_name, "agent").observe(latency)
  142. def bot_event_channel(self, event, call_info, device_info):
  143. event_name = EslEventUtil.getEventName(event)
  144. saas_id = call_info.saas_id if call_info else None
  145. flow_id = call_info.cti_flow_id if call_info else None
  146. call_id = call_info.call_id if call_info else None
  147. agent_num = device_info.agent_key if device_info else None
  148. is_agent = (device_info and DeviceType.AGENT.code == device_info.device_type) if device_info else False
  149. caller = (device_info.called if is_agent else device_info.caller) if device_info else None
  150. called = (device_info.caller if is_agent else device_info.called) if device_info else None
  151. human_service_id = '00000000000000000'
  152. start_time = time.time()
  153. try:
  154. self.logger.info('bot_event_channel, event_name=%s, call_id=%s, is_agent=%s, agent_num=%s', event_name, call_id, is_agent, agent_num)
  155. agent = self.data_handle_server.get_agent(saas_id, agent_num)
  156. if not agent:
  157. # self.logger.warn("bot event service channel agent is null %s %s %s %s %s", saas_id, event_name, caller, called,
  158. # json.dumps(event.serialize('json')))
  159. return
  160. agent_monitor = self.data_handle_server.get_agent_monitor(saas_id, agent_num)
  161. if not agent_monitor:
  162. # self.logger.warn("bot event service channel agentMonitor is null %s %s %s %s %s", saas_id, event_name, caller,
  163. # called, json.dumps(event.serialize('json')))
  164. return
  165. # 信道发起事件,触发完成发起(或桥)&& 坐席侧
  166. if CHANNEL_ORIGINATE == event_name and is_agent:
  167. self.push_handler.push_on_call_ring(saas_id, flow_id, agent_num, AgentScene.ROBOT, call_id, ServiceDirect.ROBOT_CALL.service_direct, called, caller, human_service_id)
  168. self.push_handler.push_on_agent_work_report(saas_id, flow_id, agent_num, call_id, AgentScene.ROBOT, WorkStatus.AGENT_RINGING,phone=call_info.caller)
  169. if CHANNEL_ANSWER == event_name:
  170. self.agent_state_service.busy(saas_id, agent.agent_num, agent.phone_num)
  171. if is_agent:
  172. self.agent_monitor_service.update_calling(agent_monitor)
  173. self.push_handler.push_on_agent_work_report(saas_id, flow_id, agent_num, call_id, AgentScene.ROBOT, WorkStatus.AGENT_ANSWER_INCOMING, "座席接通呼入电话! internal")
  174. self.push_handler.push_on_agent_report(saas_id, agent_num, AgentScene.ROBOT, AgentServiceState.CALLING)
  175. self.push_handler.push_answer_call(saas_id, flow_id, agent_num, call_id, AgentScene.ROBOT, ServiceDirect.ROBOT_CALL.service_direct, WorkStatus.AGENT_HANG_REPROCESSING)
  176. self.agent_actionlog_service.insert_service_state(agent_monitor, AgentServiceState.CALLING, AgentLogState.CHANNEL_TURN_ON, service_id=human_service_id)
  177. else:
  178. self.push_handler.push_on_agent_work_report(saas_id, flow_id, agent_num, call_id, AgentScene.ROBOT, WorkStatus.AGENT_ANSWER_INCOMING, "座席接通呼入电话! external")
  179. if CHANNEL_HANGUP == event_name and is_agent:
  180. self.agent_monitor_service.update_processing(agent_monitor)
  181. self.reprocessing_idle(AgentDelayStateData(saas_id, flow_id, agent_num, AgentServiceState.REPROCESSING, AgentScene.ROBOT))
  182. self.push_handler.push_on_call_end(saas_id, flow_id, agent_num, AgentScene.ROBOT, ServiceDirect.ROBOT_CALL.service_direct, "0")
  183. self.push_handler.push_on_agent_work_report(saas_id, flow_id, agent_num, call_id, AgentScene.ROBOT, WorkStatus.AGENT_HANG_REPROCESSING)
  184. self.push_handler.push_on_agent_report(saas_id, agent_num, AgentScene.ROBOT, AgentServiceState.REPROCESSING)
  185. self.agent_actionlog_service.insert_service_state(agent_monitor, AgentServiceState.REPROCESSING,
  186. AgentLogState.CHANNEL_HANG_UP, service_id=human_service_id)
  187. # self.data_handle_server.update_record(call_id, time_end=datetime.now())
  188. except:
  189. traceback.print_exc()
  190. finally:
  191. latency = (time.time() - start_time)
  192. registry.ESL_EVENT_CALLBACK_LATENCY.labels(event_name, "agent").observe(latency)
  193. def reprocessing_idle(self, state_data: AgentDelayStateData):
  194. agent = self.data_handle_server.get_agent(state_data.saas_id, state_data.agent_num)
  195. if not agent:
  196. return
  197. agent_monitor = self.data_handle_server.get_agent_monitor(state_data.saas_id, state_data.agent_num)
  198. if not agent_monitor:
  199. return
  200. self.agent_monitor_service.update_idle(agent_monitor)
  201. self.push_handler.push_on_agent_work_report(state_data.saas_id, state_data.flow_id, state_data.agent_num, "", state_data.scene, WorkStatus.AGENT_HANG_IDLE)
  202. self.agent_actionlog_service.insert_service_state(agent_monitor, AgentServiceState.IDLE, AgentLogState.REPROCESSING_IDLE)
  203. class AgentOperService:
  204. def __init__(self, app):
  205. self.app = app
  206. self.logger = app.logger
  207. self.push_handler = PushHandler(app.logger)
  208. self.data_handle_server = DataHandleServer(app)
  209. self.agent_monitor_service = AgentMonitorService(app)
  210. self.agent_actionlog_service = AgentActionLogService(app)
  211. self.agent_state_service = AgentStateService(app)
  212. @with_app_context
  213. def enable(self, req: AgentActionRequest):
  214. agent = self.data_handle_server.get_agent(req.saas_id, req.agent_number, req.out_id)
  215. if agent.agent_state == AgentState.ENABLE.code:
  216. return
  217. agent.agent_state = AgentState.ENABLE.code
  218. db.session.commit()
  219. @with_app_context
  220. def disable(self, req: AgentActionRequest):
  221. agent = self.data_handle_server.get_agent(req.saas_id, req.agent_id)
  222. if agent.agent_state == AgentState.DISABLE.code:
  223. return
  224. agent_monitor = self.data_handle_server.get_agent_monitor(req.saas_id, agent.agent_number)
  225. if agent_monitor.check_state == AgentCheck.IN.code and \
  226. agent_monitor.service_state == AgentServiceState.CALLING.code:
  227. raise BizException(BizErrorCode.AGENT_CALLING_NOT_ALLOW_OPERATE)
  228. agent.phone_num = ''
  229. agent.agent_state = AgentState.DISABLE.code
  230. db.session.commit()
  231. phone = self.data_handle_server.get_phone(req.saas_id, agent.phone_num)
  232. phone.is_delete = 1
  233. db.session.commit()
  234. def checkin(self, req: AgentActionRequest):
  235. agent = self.data_handle_server.get_agent(req.saas_id, req.agent_id)
  236. if not agent or agent.agent_state == AgentState.DISABLE.code:
  237. raise BizException(BizErrorCode.ERROR_NOT_FOLLOW_CHECK_IN)
  238. phone = self.data_handle_server.get_phone(req.saas_id, agent.phone_num)
  239. agent_monitor = self.data_handle_server.get_agent_monitor(req.saas_id, agent.agent_num)
  240. agent_monitor.check_scene = req.scene
  241. self.agent_monitor_service.update_checkin(agent_monitor)
  242. self.agent_actionlog_service.insert_check_state(agent_monitor, AgentCheck.IN, AgentLogState.CHECKIN)
  243. self.agent_state_service.checkin(agent.saas_id, agent.out_id, agent.phone_num)
  244. print("checkin", agent_monitor,agent_monitor.check_state)
  245. if req.scene == AgentScene.MANUAL.code:
  246. # 如果是手动外呼增加置忙
  247. self._handle_idle(req.scene, agent)
  248. return self._push_event_for_checkin(agent, agent_monitor, phone, req.scene)
  249. def checkout(self, req: AgentActionRequest):
  250. agent = self.data_handle_server.get_agent(req.saas_id, req.agent_id)
  251. if not agent or agent.agent_state == AgentState.DISABLE.code:
  252. raise BizException(BizErrorCode.AGENT_DISABLE_NOT_ALLOW_OPERATE)
  253. agent_monitor = self.data_handle_server.get_agent_monitor(req.saas_id, agent.agent_num)
  254. if not agent_monitor or agent_monitor.service_state == AgentServiceState.CALLING.code:
  255. raise BizException(BizErrorCode.AGENT_CALLING_NOT_ALLOW_OPERATE)
  256. if agent_monitor.check_state == AgentCheck.OUT.code:
  257. return self._push_event_for_checkout(agent, req.scene)
  258. self.agent_monitor_service.update_checkout(agent_monitor)
  259. self.agent_actionlog_service.insert_check_state(agent_monitor, AgentCheck.OUT, AgentLogState.CHECKOUT)
  260. self.agent_state_service.checkout(agent.saas_id, agent.out_id, agent.phone_num)
  261. return self._push_event_for_checkout(agent, req.scene)
  262. @with_app_context
  263. def busy(self, req: AgentActionRequest):
  264. agent = self.data_handle_server.get_agent(req.saas_id, req.agent_id)
  265. if not agent or agent.agent_state == AgentState.DISABLE.code:
  266. raise BizException(BizErrorCode.AGENT_DISABLE_NOT_ALLOW_OPERATE)
  267. agent_monitor = self.data_handle_server.get_agent_monitor(req.saas_id, agent.agent_num)
  268. if not agent_monitor or agent_monitor.check_state == AgentCheck.OUT.code:
  269. raise BizException(BizErrorCode.AGENT_CHECK_OUT_NOT_ALLOW_OPERATE)
  270. if agent_monitor.service_state == AgentServiceState.CALLING.code:
  271. raise BizException(BizErrorCode.AGENT_CALLING_NOT_ALLOW_OPERATE)
  272. self.agent_state_service.busy(agent.saas_id, agent.out_id, agent.phone_num)
  273. if agent_monitor.service_state == AgentServiceState.BUSY.code:
  274. self._push_event_for_busy(agent, req.scene)
  275. return
  276. self.agent_monitor_service.update_busy(agent_monitor)
  277. self.agent_actionlog_service.insert_service_state(agent_monitor, AgentServiceState.BUSY, AgentLogState.BUSY)
  278. self._push_event_for_busy(agent, req.scene)
  279. @with_app_context
  280. def idle(self, req: AgentActionRequest):
  281. agent = self.data_handle_server.get_agent(req.saas_id, req.agent_id)
  282. if not agent or agent.agent_state == AgentState.DISABLE.code:
  283. raise BizException(BizErrorCode.AGENT_DISABLE_NOT_ALLOW_OPERATE)
  284. self._handle_idle(req.scene, agent)
  285. @with_app_context
  286. def assign(self, req: AgentActionRequest):
  287. return self.agent_state_service.assign_agent(req.saas_id, req.service_id)
  288. def idle_agent_exist(self, request: AgentActionRequest):
  289. pass
  290. @with_app_context
  291. def agent_state(self,req: AgentActionRequest):
  292. # agent = _get_agent(req.saas_id, req.agent_id)
  293. agent_monitor = self.data_handle_server.get_agent_monitor(req.saas_id, req.agent_id)
  294. return agent_monitor.service_state
  295. @with_app_context
  296. def turn_on(self, req: AgentActionRequest):
  297. agent = self.data_handle_server.get_agent(req.saas_id, req.agent_id)
  298. agent_monitor = self.data_handle_server.get_agent_monitor(req.saas_id, agent.agent_num)
  299. agent_scene = AgentScene.get_by_code(req.scene)
  300. if not agent_monitor:
  301. raise BizException(BizErrorCode.RECORD_NOT_EXIST_ERROR)
  302. if agent_monitor.service_state == AgentServiceState.CALLING.code:
  303. self.push_handler.push_on_agent_report(agent.saas_id, agent.out_id, agent_scene, AgentServiceState.BUSY)
  304. return
  305. self.agent_monitor_service.update_calling(agent_monitor)
  306. self.agent_state_service.busy(agent.saas_id, agent.out_id, agent.phone_num)
  307. self.push_handler.push_on_agent_report(agent.saas_id, agent.out_id, agent_scene, AgentServiceState.BUSY)
  308. @with_app_context
  309. def _handle_idle(self, scene, agent):
  310. agent_monitor = self.data_handle_server.get_agent_monitor(agent.saas_id, agent.agent_num)
  311. print('agent_monitor:', agent_monitor, agent)
  312. if agent_monitor.check_state == AgentCheck.OUT.code:
  313. raise BizException(BizErrorCode.AGENT_CHECK_OUT_NOT_ALLOW_OPERATE)
  314. if agent_monitor.service_state == AgentServiceState.CALLING.code or agent_monitor.service_state == AgentServiceState.DIALING.code:
  315. raise BizException(BizErrorCode.AGENT_CALLING_NOT_HANG)
  316. if scene == AgentScene.ROBOT.code:
  317. self.agent_state_service.idle(agent.saas_id, agent.out_id, agent.phone_num)
  318. else:
  319. self.agent_state_service.busy(agent.saas_id, agent.out_id, agent.phone_num)
  320. if agent_monitor.service_state == AgentServiceState.IDLE.code:
  321. self._push_event_for_idle(agent, scene)
  322. return
  323. self.agent_monitor_service.update_idle(agent_monitor)
  324. self.agent_actionlog_service.insert_service_state(agent_monitor, AgentServiceState.IDLE, AgentLogState.IDLE)
  325. self._push_event_for_idle(agent, scene)
  326. def _push_event_for_checkin(self, agent, agent_monitor, phone, scene):
  327. """坐席签入事件推送"""
  328. agent_scene = AgentScene.get_by_code(scene)
  329. self.push_handler.push_on_agent_work_report(agent.saas_id, "", agent.out_id, "", agent_scene, WorkStatus.LOGIN_SUCCESS)
  330. event_data = AgentEventData(agent.saas_id, agent.out_id)
  331. event_data.data = {'eventName': DownEvent.ON_INITAL_SUCCESS.code,
  332. 'ext': {'saasId': agent.saas_id,
  333. 'agentId': agent.out_id,
  334. 'phoneNum': phone.phone_num,
  335. 'phonePwd': phone.phone_pwd,
  336. 'sipServer': phone.sip_server
  337. }
  338. }
  339. return event_data.__dict__
  340. def _push_event_for_checkout(self,agent,scene):
  341. """签出事件推送"""
  342. agent_scene = AgentScene.get_by_code(scene)
  343. self.push_handler.push_on_agent_work_report(agent.saas_id, "", agent.out_id, "", agent_scene, WorkStatus.NO_INIT, '签出成功')
  344. event_data = AgentEventData(agent.saas_id, agent.out_id)
  345. event_data.data = {'eventName': DownEvent.ON_INITAL_FAILURE.code,
  346. 'ext': {'saasId': agent.saas_id,
  347. 'agentId': agent.out_id,
  348. }
  349. }
  350. return event_data.__dict__
  351. def _push_event_for_busy(self,agent,scene):
  352. """置忙事件推送"""
  353. agent_scene = AgentScene.get_by_code(scene)
  354. self.push_handler.push_on_agent_work_report(agent.saas_id, "", agent.out_id, "", agent_scene, WorkStatus.AGENT_BUSY)
  355. pass
  356. def _push_event_for_idle(self, agent, scene):
  357. """坐席置闲事件推送"""
  358. agent_scene = AgentScene.get_by_code(scene)
  359. self.push_handler.push_on_agent_work_report(agent.saas_id, "", agent.out_id, "", agent_scene, WorkStatus.AGENT_READY)
  360. class AgentService:
  361. def __init__(self, app):
  362. self.app = app
  363. self.logger = app.logger
  364. self.data_handle_server = DataHandleServer(app)
  365. self.agent_monitor_service = AgentMonitorService(app)
  366. @with_app_context
  367. def get_and_check(self, req: AgentActionRequest):
  368. agent = self.data_handle_server.get_agent(req.saas_id, req.agent_id)
  369. if not agent:
  370. return {}
  371. phone = self.data_handle_server.get_phone(req.saas_id, agent.phone_num)
  372. return phone.to_dict()
  373. @with_app_context
  374. def watch_agent_state(self, req: HumanServiceQueryRequest):
  375. pos = HumanServiceMap.query.filter(HumanServiceMap.is_delete == 0, HumanServiceMap.saas_id == req.saas_id,
  376. HumanServiceMap.service_id == req.serviceId).all()
  377. agent_ids = [x.agent_id for x in pos]
  378. monitors = self.agent_monitor_service.detail_monitor_out_ids(req.saas_id, agent_ids)
  379. return monitors
  380. @with_app_context
  381. def add(self, req: AgentRequest):
  382. new_agent_num = self.data_handle_server.get_newest_agent_number(req.saas_id)
  383. agent = Agent(saas_id=req.saas_id, agent_num=new_agent_num, agent_name=req.agent_name, out_id=req.out_id,
  384. agent_pwd=req.agent_password, agent_type=req.agent_type, phone_num=req.phone_number,
  385. distribute=req.distribute, agent_state=req.agent_state)
  386. db.session.add(agent)
  387. db.session.commit()
  388. agent_monitor = AgentMonitor(saas_id=req.saas_id, agent_num=new_agent_num, out_id=req.out_id)
  389. db.session.add(agent_monitor)
  390. db.session.commit()
  391. return new_agent_num
  392. @with_app_context
  393. def update(self, req: AgentRequest):
  394. agent = self.data_handle_server.get_agent(req.saas_id, req.agent_number, req.out_id)
  395. if not agent:
  396. return
  397. phone_num = agent.phone_num
  398. state_change = agent.agent_state != req.agent_state
  399. disable = req.agent_state == AgentState.DISABLE
  400. agent.agent_name = req.agent_name
  401. agent.agent_pwd = req.agent_password
  402. agent.agent_type = req.agent_type
  403. agent.phone_num = req.phone_number if not disable else ''
  404. agent.distribute = req.distribute
  405. agent.agent_state = req.agent_state
  406. db.session.commit()
  407. if state_change and disable:
  408. phone = Phone.query.filter(Phone.saas_id == req.saas_id, Phone.phone_num == phone_num).first()
  409. if phone:
  410. db.session.delete(phone)
  411. @with_app_context
  412. def detail(self, saas_id, agent_number):
  413. agent = self.data_handle_server.get_agent(saas_id, agent_number=agent_number, out_id='')
  414. return agent
  415. @with_app_context
  416. def count(self, req: AgentQueryRequest):
  417. cnt = Agent.query.filter(Agent.saas_id == req.saas_id,
  418. or_(Agent.agent_num == req.agent_number,
  419. Agent.agent_name.contains(req.agent_name),
  420. Agent.out_id == req.out_id,
  421. Agent.agent_type == req.agent_type
  422. )
  423. ).count()
  424. return cnt
  425. @with_app_context
  426. def query_page(self, req: AgentQueryRequest):
  427. pagination = Agent.query.filter(Agent.saas_id == req.saas_id,
  428. or_(Agent.agent_num == req.agent_number,
  429. Agent.agent_name.contains(req.agent_name),
  430. Agent.out_id == req.out_id,
  431. Agent.agent_type == req.agent_type
  432. )
  433. ).paginate(req.page, req.size)
  434. # data = {
  435. # "page": pagination.page, # 当前页码
  436. # "pages": pagination.pages, # 总页码
  437. # "has_prev": pagination.has_prev, # 是否有上一页
  438. # "prev_num": pagination.prev_num, # 上一页页码
  439. # "has_next": pagination.has_next, # 是否有下一页
  440. # "next_num": pagination.next_num, # 下一页页码
  441. # "items": [{
  442. # "id": item.id,
  443. # "name": item.name,
  444. # "age": item.age,
  445. # "sex": item.sex,
  446. # "money": item.money,
  447. # } for item in pagination.items]
  448. # }
  449. return pagination
  450. @with_app_context
  451. def delete(self, saas_id, agent_number):
  452. agent = self.data_handle_server.get_agent(saas_id, agent_number=agent_number, out_id='')
  453. if not agent:
  454. return
  455. agent.is_delete = 1
  456. agent_monitor = self.data_handle_server.get_agent_monitor(saas_id, agent_number)
  457. agent_monitor.is_delete = 1
  458. db.session.commit()
  459. phone = self.data_handle_server.get_phone(saas_id, agent.phone_num)
  460. phone.is_delete = 1
  461. db.session.commit()
  462. class AgentMonitorService:
  463. def __init__(self, app):
  464. self.app = app
  465. self.logger = app.logger
  466. self.data_handle_server = DataHandleServer(app)
  467. @with_app_context
  468. def detail_monitor_out_ids(self, saas_id, out_ids, check_scene=None):
  469. if not out_ids:
  470. return []
  471. agents = Agent.query.filter(Agent.is_delete == 0, Agent.saas_id == saas_id, Agent.out_id.in_(out_ids)).all()
  472. if not agents:
  473. raise BizException(BizErrorCode.RECORD_NOT_EXIST_ERROR)
  474. agent_num_map = {x.agent_num: x for x in agents}
  475. agent_nums = [x.agent_num for x in agents]
  476. agent_monitors = AgentMonitor.query.filter(AgentMonitor.is_delete == 0, AgentMonitor.agent_num.in_(agent_nums)).all()
  477. res = []
  478. for agent_monitor in agent_monitors:
  479. agent = agent_num_map.get(agent_monitor.agent_num)
  480. data = AgentMonitorData()
  481. data.saas_id = saas_id
  482. data.agent_num = agent_monitor.agent_num
  483. data.agent_name = agent.agent_name
  484. data.out_id = agent_monitor.out_id
  485. data.check_state = agent_monitor.check_state
  486. data.check_scene = agent_monitor.check_scene
  487. data.online_state = 1 if agent_monitor.check_state == AgentCheck.IN.code else 0
  488. if agent_monitor.check_in_time:
  489. data.check_in_time = agent_monitor.check_in_time.timestamp()
  490. day_start = self._get_day_start()
  491. if agent_monitor.check_in_time.timestamp() > day_start and \
  492. agent_monitor.check_state == AgentCheck.OUT.code:
  493. data.online_state = 2
  494. data.check_out_time = agent_monitor.check_out_time.timestamp() if agent_monitor.check_out_time else None
  495. data.service_state = agent_monitor.service_state
  496. data.busy_time = agent_monitor.busy_time.timestamp() if agent_monitor.busy_time else None
  497. data.idle_time = agent_monitor.idle_time.timestamp() if agent_monitor.idle_time else None
  498. data.call_time = agent_monitor.call_time.timestamp() if agent_monitor.call_time else None
  499. data.hang_time = agent_monitor.hang_time.timestamp() if agent_monitor.hang_time else None
  500. data.heart_state = agent_monitor.heart_state
  501. data.heart_time = agent_monitor.heart_time.timestamp() if agent_monitor.heart_time else None
  502. data.update_time = agent_monitor.update_time.timestamp() if agent_monitor.update_time else None
  503. res.append(data.__dict__)
  504. return res
  505. def update_checkin(self, agent_monitor):
  506. agent_monitor = db.session.query(AgentMonitor).get(agent_monitor.id)
  507. agent_monitor.check_state = AgentCheck.IN.code
  508. agent_monitor.check_in_time = datetime.now()
  509. agent_monitor.heart_state = AgentHeartState.NORMAL.code
  510. agent_monitor.heart_time = datetime.now()
  511. self.logger.info("update_checkin %s", agent_monitor.check_state)
  512. db.session.commit()
  513. def update_checkout(self, agent_monitor):
  514. agent_monitor = db.session.query(AgentMonitor).get(agent_monitor.id)
  515. agent_monitor.check_state = AgentCheck.OUT.code
  516. agent_monitor.check_out_time = datetime.now()
  517. agent_monitor.service_state = AgentServiceState.LOGOUT.code
  518. agent_monitor.heart_state = AgentHeartState.DEFAULT.code
  519. agent_monitor.heart_time = datetime.now()
  520. self.logger.info("update_checkout %s", agent_monitor.check_out_time)
  521. db.session.commit()
  522. def update_idle(self, agent_monitor):
  523. agent_monitor = db.session.query(AgentMonitor).get(agent_monitor.id)
  524. agent_monitor.service_state = AgentServiceState.IDLE.code
  525. agent_monitor.idle_time = datetime.now()
  526. db.session.commit()
  527. def update_busy(self, agent_monitor):
  528. agent_monitor = db.session.query(AgentMonitor).get(agent_monitor.id)
  529. agent_monitor.service_state = AgentServiceState.BUSY.code
  530. agent_monitor.busy_time = datetime.now()
  531. db.session.commit()
  532. def update_dialing(self, agent_monitor):
  533. agent_monitor = db.session.query(AgentMonitor).get(agent_monitor.id)
  534. agent_monitor.service_state = AgentServiceState.DIALING.code
  535. db.session.commit()
  536. def update_calling(self, agent_monitor):
  537. agent_monitor = db.session.query(AgentMonitor).get(agent_monitor.id)
  538. agent_monitor.service_state = AgentServiceState.CALLING.code
  539. agent_monitor.call_time = datetime.now()
  540. db.session.commit()
  541. def update_processing(self, agent_monitor):
  542. agent_monitor = db.session.query(AgentMonitor).get(agent_monitor.id)
  543. agent_monitor.service_state = AgentServiceState.REPROCESSING.code
  544. agent_monitor.hang_time = datetime.now()
  545. db.session.commit()
  546. def update_session_id(self, agent_monitor, session_id):
  547. agent_monitor = db.session.query(AgentMonitor).get(agent_monitor.id)
  548. agent_monitor.session_id = session_id
  549. db.session.commit()
  550. def update_heart_error(self, agent_monitor):
  551. agent_monitor = db.session.query(AgentMonitor).get(agent_monitor.id)
  552. agent_monitor.heart_state = AgentHeartState.ABNORMAL.code
  553. agent_monitor.heart_time = datetime.now()
  554. db.session.commit()
  555. def _get_day_start(self):
  556. today_start = datetime.now().replace(hour=0, minute=0, second=0, microsecond=0)
  557. return int(today_start.timestamp() * 1000) # Return milliseconds
  558. class AgentActionLogService:
  559. def __init__(self, app):
  560. self.app = app
  561. self.logger = app.logger
  562. self.data_handle_server = DataHandleServer(app)
  563. @with_app_context
  564. def insert_check_state(self, agent_monitor, agent_check_enum: AgentCheck, agent_log_enum: AgentLogState):
  565. action_log = AgentActionLog()
  566. action_log.saas_id = agent_monitor.saas_id
  567. action_log.agent_num = agent_monitor.agent_num
  568. action_log.out_id = agent_monitor.out_id
  569. action_log.action_type = 0
  570. action_log.check_state = agent_check_enum.code
  571. action_log.pre_check_state = agent_monitor.check_state
  572. if agent_log_enum:
  573. action_log.event_type = agent_log_enum.code
  574. action_log.event_desc = agent_log_enum.description
  575. now = datetime.now()
  576. pre_date = None
  577. if agent_monitor.check_state == AgentCheck.IN.code:
  578. pre_date = agent_monitor.check_in_time
  579. if agent_monitor.check_state == AgentCheck.OUT.code:
  580. pre_date = agent_monitor.check_out_time
  581. if pre_date is None:
  582. pre_date = agent_monitor.update_time
  583. action_log.check_state_time = now
  584. action_log.pre_check_state_time = pre_date
  585. action_log.check_state_duration = now.timestamp() - pre_date.timestamp()
  586. db.session.add(action_log)
  587. db.session.commit()
  588. @with_app_context
  589. def insert_service_state(self, agent_monitor, agent_service_state: AgentServiceState, agent_log_enum: AgentLogState,
  590. task_id=None, service_id=None):
  591. if agent_monitor.service_state == agent_service_state.code:
  592. self.logger.info("agent action log insert service state same %s %s", agent_monitor.service_state, agent_service_state.code)
  593. action_log = AgentActionLog()
  594. action_log.saas_id = agent_monitor.saas_id
  595. action_log.agent_num = agent_monitor.agent_num
  596. action_log.out_id = agent_monitor.out_id
  597. action_log.action_type = 1
  598. action_log.service_state = agent_service_state.code
  599. action_log.pre_service_state = agent_monitor.service_state
  600. action_log.task_id = task_id
  601. action_log.service_id = service_id
  602. if agent_log_enum:
  603. action_log.event_type = agent_log_enum.code
  604. action_log.event_desc = agent_log_enum.description
  605. now = datetime.now()
  606. pre_date = None
  607. if agent_monitor.service_state == AgentServiceState.IDLE.code:
  608. pre_date = agent_monitor.idle_time
  609. if agent_monitor.service_state == AgentServiceState.BUSY.code:
  610. pre_date = agent_monitor.busy_time
  611. if agent_monitor.service_state == AgentServiceState.CALLING.code:
  612. pre_date = agent_monitor.call_time
  613. if agent_monitor.service_state == AgentServiceState.LOGOUT.code:
  614. pre_date = agent_monitor.check_out_time
  615. if agent_monitor.service_state == AgentServiceState.REPROCESSING.code:
  616. pre_date = agent_monitor.hang_time
  617. action_log.service_state_time = now
  618. if pre_date is None:
  619. pre_date = agent_monitor.update_time
  620. action_log.pre_service_state_time = pre_date
  621. action_log.service_state_duration = now.timestamp() - pre_date.timestamp()
  622. db.session.add(action_log)
  623. db.session.commit()
  624. class AgentStateService:
  625. def __init__(self, app):
  626. self.app = app
  627. self.logger = app.logger
  628. self.redis_handler = RedisHandler()
  629. self.assigned_recycle_millisecond = 60000
  630. self.state_service_id_data_map = defaultdict(dict)
  631. self.executor = ThreadPoolExecutor(max_workers=10)
  632. self.data_handle_server = DataHandleServer(app)
  633. self.agent_monitor_service = AgentMonitorService(app)
  634. self.agent_actionlog_service = AgentActionLogService(app)
  635. def idle(self, saas_id, agent_id, phone_num):
  636. human_service = self.data_handle_server.get_human_service_service(saas_id, agent_id)
  637. if human_service is None:
  638. self.logger.info(f"agent engine idle not have human service {saas_id} {agent_id}") # 使用print替代log
  639. return
  640. self.idle_hash(saas_id, agent_id, phone_num, human_service.service_id)
  641. def busy(self, saas_id, agent_id, phone_num):
  642. human_service = self.data_handle_server.get_human_service_service(saas_id, agent_id)
  643. if human_service is None:
  644. self.logger.info(f"agent engine busy not hava human service {saas_id} {agent_id}") # 使用print替代log
  645. return
  646. self.busy_hash(saas_id, agent_id, phone_num, human_service.service_id)
  647. def idle_by_human(self, saas_id, agent_id, service_id):
  648. agent = self.data_handle_server.get_agent(saas_id, out_id=agent_id)
  649. if not agent:
  650. return
  651. agent_monitor = self.data_handle_server.get_agent_monitor(saas_id, agent_number=agent.agent_num)
  652. if not agent_monitor:
  653. return
  654. if agent_monitor.check_state == AgentCheck.IN.code:
  655. self.agent_monitor_service.update_idle(agent_monitor)
  656. self.agent_actionlog_service.insert_service_state(agent_monitor, AgentServiceState.IDLE, AgentLogState.ACTIVE_HUMAN_SERVICE)
  657. self.idle_hash(saas_id, agent_id, agent.phone_num, service_id)
  658. def busy_by_human(self, saas_id, service_id, agent_id=None):
  659. agent = self.data_handle_server.get_agent(saas_id, out_id=agent_id)
  660. if not agent:
  661. return
  662. agent_monitor = self.data_handle_server.get_agent_monitor(saas_id, agent_number=agent.agent_num)
  663. if not agent_monitor:
  664. return
  665. if agent_monitor.check_state == AgentCheck.IN.code:
  666. self.agent_monitor_service.update_busy(agent_monitor)
  667. self.agent_actionlog_service.insert_service_state(agent_monitor, AgentServiceState.BUSY,
  668. AgentLogState.DEL_HUMAN_SERVICE)
  669. self.busy_hash(saas_id, agent_id, agent.phone_num, service_id)
  670. def checkin(self, saas_id, agent_id, phone_num):
  671. key = self._check_in_key(saas_id)
  672. state_data = AgentStateData()
  673. state_data.status = HumanState.DEFAULT.code
  674. state_data.time = datetime.now().timestamp()
  675. self.redis_handler.redis.hset(key, phone_num, state_data.to_json_string())
  676. self.redis_handler.redis.expire(key, self._get_expire_time())
  677. def checkout(self, saas_id, agent_id, phone_num):
  678. key = self._check_in_key(saas_id)
  679. self.redis_handler.redis.hdel(key, phone_num)
  680. def assign_agent(self, saas_id, service_id, called=None, ivr_id=None, task_id=None, cbp=None):
  681. choose_phone_num = ''
  682. self.logger.info("assignAgent %s %s %s"% (saas_id, service_id, called))
  683. idle_agents = self.idle_agents(saas_id, service_id)
  684. if len(idle_agents) <= 0:
  685. return choose_phone_num
  686. choose_phone_num = self._choose_max_idle_time(idle_agents)
  687. self.handle_assign_time(saas_id, service_id, choose_phone_num)
  688. return choose_phone_num
  689. def handle_assign_time(self, saas_id, service_id, choose_phone_num):
  690. key = self._key(saas_id, service_id)
  691. cache_agent_map = self.get_cache_agent_map(saas_id, service_id)
  692. if cache_agent_map and choose_phone_num in cache_agent_map:
  693. state_data = cache_agent_map[choose_phone_num]
  694. state_data.assign_time = datetime.now().timestamp() * 1000
  695. self.redis_handler.redis.hset(key, choose_phone_num, state_data.to_json_string())
  696. self.redis_handler.redis.expire(key, self._get_expire_time())
  697. # self.update_report_state(saas_id, service_id)
  698. def idle_agents(self, saas_id, service_id):
  699. cache_agent_list = self.get_cache_agent_list(saas_id, service_id)
  700. if not cache_agent_list:
  701. return []
  702. agent_str = '\n'.join(["%s-%s-%s-%s"%(x.phone_num, x.status, x.time, x.assign_time) for x in cache_agent_list])
  703. self.logger.info("assignAgent %s %s idleAgents:%s "% (saas_id, service_id, agent_str))
  704. return self.get_idle_agents(cache_agent_list)
  705. def idle_hash(self, saas_id, agent_id, phone_num, service_id):
  706. key = self._key(saas_id, service_id)
  707. state_data = AgentStateData()
  708. # self.logger.info("idle_hash, key=%s, saas_id=%s, phone_num=%s", key, saas_id, phone_num)
  709. cache_agent_map = self.get_cache_agent_map(saas_id, service_id)
  710. if cache_agent_map and phone_num in cache_agent_map:
  711. state_data = cache_agent_map[phone_num]
  712. state_data.status = HumanState.IDLE.code
  713. state_data.time = datetime.now().timestamp() * 1000
  714. self.redis_handler.redis.hset(key, phone_num, state_data.to_json_string())
  715. self.redis_handler.redis.expire(key, self._get_expire_time())
  716. # self.update_report_state(saas_id, service_id)
  717. self.logger.info("idle_hash, key=%s, saas_id=%s, phone_num=%s, state_data=%s"%(key, saas_id, phone_num, state_data))
  718. def busy_hash(self, saas_id, agent_id, phone_num, service_id):
  719. cache_agent_map = self.get_cache_agent_map(saas_id, service_id)
  720. state_data = cache_agent_map.get(phone_num)
  721. key = self._key(saas_id, service_id)
  722. if state_data is None:
  723. return
  724. state_data.status = HumanState.BUSY.code
  725. self.redis_handler.redis.hset(key, phone_num, state_data.to_json_string())
  726. self.redis_handler.redis.expire(key, self._get_expire_time())
  727. # self.update_report_state(saas_id, service_id)
  728. def get_cache_agent_map(self, saas_id, service_id):
  729. cache_agent_list = self.get_cache_agent_list(saas_id, service_id)
  730. # 检查列表是否为空,如果为空返回空字典
  731. if not cache_agent_list:
  732. return {}
  733. # 使用字典推导式将 cache_agent_list 转换为字典
  734. return {agent.phone_num: agent for agent in cache_agent_list}
  735. def get_cache_agent_list(self, saas_id, service_id):
  736. redis_key = self._key(saas_id, service_id)
  737. map_cache_by_key = self.redis_handler.redis.hgetall(redis_key)
  738. self.logger.info("get_cache_agent_list, redis_key=%s, map_cache_by_key=%s"%(redis_key, map_cache_by_key))
  739. if not map_cache_by_key: # 检查字典是否为空
  740. return [] # 返回空列表
  741. free_agents = []
  742. for phone_num, json_value in map_cache_by_key.items():
  743. agent_status_data_dict = json.loads(json_value)
  744. agent_status_data = AgentStateData(**agent_status_data_dict) # 解析 JSON 为 AgentStateData 对象
  745. agent_status_data.phone_num = phone_num.decode('utf-8') # 设置电话号码
  746. free_agents.append(agent_status_data)
  747. return free_agents
  748. def get_idle_agents(self,cache_agent_list):
  749. current_time =int(datetime.now().timestamp() * 1000) # 获取当前时间的毫秒级时间戳
  750. idle_agents = [
  751. agent for agent in cache_agent_list
  752. if agent.status == 1 and (
  753. agent.assign_time == 0 or
  754. agent.assign_time + self.assigned_recycle_millisecond < current_time
  755. )
  756. ]
  757. return idle_agents
  758. def get_agent_service_idle_size(self, saas_id, service_id):
  759. idle_agents_size = 0
  760. cache_agent_list = self.get_cache_agent_list(saas_id, service_id)
  761. if cache_agent_list: # 检查列表是否非空
  762. idle_agents = self.get_idle_agents(cache_agent_list)
  763. idle_agents_size = len(idle_agents) # 获取空闲代理的数量
  764. return idle_agents_size
  765. def get_agent_service_busy_size(self, saas_id, service_id):
  766. busy_agents_size = 0
  767. cache_agent_list = self.get_cache_agent_list(saas_id, service_id)
  768. if cache_agent_list: # 检查列表是否非空
  769. idle_agents = self.get_idle_agents(cache_agent_list)
  770. busy_agents = [agent for agent in cache_agent_list if agent not in idle_agents] # 计算忙碌代理
  771. busy_agents_size = len(busy_agents) # 获取忙碌代理的数量
  772. return busy_agents_size
  773. def update_report_state(self, saas_id, service_id):
  774. key = self._key(saas_id, service_id)
  775. # data_map 这个地方有疑问
  776. data_map = self.state_service_id_data_map[key]
  777. idle = HumanState.IDLE
  778. if idle.value not in data_map:
  779. data_map[idle.code] = threading.Lock()
  780. self.executor.submit(self.do_report_real_time_human_service_id, saas_id, service_id, idle)
  781. # data_map[idle.code] = self.do_report_real_time_human_service_id(saas_id, service_id, idle)
  782. busy = HumanState.BUSY
  783. if busy.value not in data_map:
  784. data_map[busy.code] = threading.Lock()
  785. self.executor.submit(self.do_report_real_time_human_service_id, saas_id, service_id, busy)
  786. # data_map[busy.code] = self.do_report_real_time_human_service_id(saas_id, service_id, busy)
  787. def do_report_real_time_human_service_id(self, saas_id, service_id, human_state):
  788. name = "cti_center_real_time_human_service_state"
  789. tag_list = {
  790. "vcc_id": saas_id,
  791. "service_id": service_id,
  792. "state": human_state.code,
  793. }
  794. if human_state == HumanState.IDLE:
  795. # meter_registry 这块疑问
  796. self.meter_registry.gauge(name, tag_list, self, lambda ctx: ctx.get_agent_service_busy_size(saas_id, service_id))
  797. elif human_state == HumanState.BUSY:
  798. self.meter_registry.gauge(name, tag_list, self, lambda ctx: ctx.get_agent_service_idle_size(saas_id, service_id))
  799. return 0
  800. def _check_in_key(self, saas_id):
  801. return "CTI:%s:HUMAN:AGENT"%(saas_id.upper())
  802. def _key(self, saas_id, service_id):
  803. return "CTI:%s:HUMAN:%s"%(saas_id.upper(), service_id)
  804. def _get_expire_time(self):
  805. now = datetime.now()
  806. end_of_day = now.replace(hour=23, minute=59, second=59, microsecond=0)
  807. expire_time = (end_of_day - now).total_seconds() * 1000 # Convert to milliseconds
  808. return int(expire_time)
  809. def _choose_max_idle_time(self, idle_agents: List[AgentStateData]) -> str:
  810. idle_agents = sorted(idle_agents, key=lambda agent: agent.assign_time, reverse=False)
  811. return idle_agents[0].phone_num