agent.py 49 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962
  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. @with_app_context
  235. def checkin(self, req: AgentActionRequest):
  236. agent = self.data_handle_server.get_agent(req.saas_id, req.agent_id)
  237. if not agent or agent.agent_state == AgentState.DISABLE.code:
  238. raise BizException(BizErrorCode.ERROR_NOT_FOLLOW_CHECK_IN)
  239. phone = self.data_handle_server.get_phone(req.saas_id, agent.phone_num)
  240. agent_monitor = self.data_handle_server.get_agent_monitor(req.saas_id, agent.agent_num)
  241. agent_monitor.check_scene = req.scene
  242. self.agent_monitor_service.update_checkin(agent_monitor)
  243. self.agent_actionlog_service.insert_check_state(agent_monitor, AgentCheck.IN, AgentLogState.CHECKIN)
  244. self.agent_state_service.checkin(agent.saas_id, agent.out_id, agent.phone_num)
  245. print("checkin", agent_monitor,agent_monitor.check_state)
  246. if req.scene == AgentScene.MANUAL.code:
  247. # 如果是手动外呼增加置忙
  248. self._handle_idle(req.scene, agent)
  249. return self._push_event_for_checkin(agent, agent_monitor, phone, req.scene)
  250. @with_app_context
  251. def checkout(self, req: AgentActionRequest):
  252. agent = self.data_handle_server.get_agent(req.saas_id, req.agent_id)
  253. if not agent or agent.agent_state == AgentState.DISABLE.code:
  254. raise BizException(BizErrorCode.AGENT_DISABLE_NOT_ALLOW_OPERATE)
  255. agent_monitor = self.data_handle_server.get_agent_monitor(req.saas_id, agent.agent_num)
  256. if not agent_monitor or agent_monitor.service_state == AgentServiceState.CALLING.code:
  257. raise BizException(BizErrorCode.AGENT_CALLING_NOT_ALLOW_OPERATE)
  258. if agent_monitor.check_state == AgentCheck.OUT.code:
  259. return self._push_event_for_checkout(agent, req.scene)
  260. self.agent_monitor_service.update_checkout(agent_monitor)
  261. self.agent_actionlog_service.insert_check_state(agent_monitor, AgentCheck.OUT, AgentLogState.CHECKOUT)
  262. self.agent_state_service.checkout(agent.saas_id, agent.out_id, agent.phone_num)
  263. return self._push_event_for_checkout(agent, req.scene)
  264. @with_app_context
  265. def busy(self, req: AgentActionRequest):
  266. agent = self.data_handle_server.get_agent(req.saas_id, req.agent_id)
  267. if not agent or agent.agent_state == AgentState.DISABLE.code:
  268. raise BizException(BizErrorCode.AGENT_DISABLE_NOT_ALLOW_OPERATE)
  269. agent_monitor = self.data_handle_server.get_agent_monitor(req.saas_id, agent.agent_num)
  270. if not agent_monitor or agent_monitor.check_state == AgentCheck.OUT.code:
  271. raise BizException(BizErrorCode.AGENT_CHECK_OUT_NOT_ALLOW_OPERATE)
  272. if agent_monitor.service_state == AgentServiceState.CALLING.code:
  273. raise BizException(BizErrorCode.AGENT_CALLING_NOT_ALLOW_OPERATE)
  274. self.agent_state_service.busy(agent.saas_id, agent.out_id, agent.phone_num)
  275. if agent_monitor.service_state == AgentServiceState.BUSY.code:
  276. self._push_event_for_busy(agent, req.scene)
  277. return
  278. self.agent_monitor_service.update_busy(agent_monitor)
  279. self.agent_actionlog_service.insert_service_state(agent_monitor, AgentServiceState.BUSY, AgentLogState.BUSY)
  280. self._push_event_for_busy(agent, req.scene)
  281. @with_app_context
  282. def idle(self, req: AgentActionRequest):
  283. agent = self.data_handle_server.get_agent(req.saas_id, req.agent_id)
  284. if not agent or agent.agent_state == AgentState.DISABLE.code:
  285. raise BizException(BizErrorCode.AGENT_DISABLE_NOT_ALLOW_OPERATE)
  286. self._handle_idle(req.scene, agent)
  287. @with_app_context
  288. def assign(self, req: AgentActionRequest):
  289. return self.agent_state_service.assign_agent(req.saas_id, req.service_id)
  290. def idle_agent_exist(self, request: AgentActionRequest):
  291. pass
  292. @with_app_context
  293. def agent_state(self,req: AgentActionRequest):
  294. # agent = _get_agent(req.saas_id, req.agent_id)
  295. agent_monitor = self.data_handle_server.get_agent_monitor(req.saas_id, req.agent_id)
  296. return agent_monitor.service_state
  297. @with_app_context
  298. def turn_on(self, req: AgentActionRequest):
  299. agent = self.data_handle_server.get_agent(req.saas_id, req.agent_id)
  300. agent_monitor = self.data_handle_server.get_agent_monitor(req.saas_id, agent.agent_num)
  301. agent_scene = AgentScene.get_by_code(req.scene)
  302. if not agent_monitor:
  303. raise BizException(BizErrorCode.RECORD_NOT_EXIST_ERROR)
  304. if agent_monitor.service_state == AgentServiceState.CALLING.code:
  305. self.push_handler.push_on_agent_report(agent.saas_id, agent.out_id, agent_scene, AgentServiceState.BUSY)
  306. return
  307. self.agent_monitor_service.update_calling(agent_monitor)
  308. self.agent_state_service.busy(agent.saas_id, agent.out_id, agent.phone_num)
  309. self.push_handler.push_on_agent_report(agent.saas_id, agent.out_id, agent_scene, AgentServiceState.BUSY)
  310. @with_app_context
  311. def _handle_idle(self, scene, agent):
  312. agent_monitor = self.data_handle_server.get_agent_monitor(agent.saas_id, agent.agent_num)
  313. print('agent_monitor:', agent_monitor, agent)
  314. if agent_monitor.check_state == AgentCheck.OUT.code:
  315. raise BizException(BizErrorCode.AGENT_CHECK_OUT_NOT_ALLOW_OPERATE)
  316. if agent_monitor.service_state == AgentServiceState.CALLING.code or agent_monitor.service_state == AgentServiceState.DIALING.code:
  317. raise BizException(BizErrorCode.AGENT_CALLING_NOT_HANG)
  318. if scene == AgentScene.ROBOT.code:
  319. self.agent_state_service.idle(agent.saas_id, agent.out_id, agent.phone_num)
  320. else:
  321. self.agent_state_service.busy(agent.saas_id, agent.out_id, agent.phone_num)
  322. if agent_monitor.service_state == AgentServiceState.IDLE.code:
  323. self._push_event_for_idle(agent, scene)
  324. return
  325. self.agent_monitor_service.update_idle(agent_monitor)
  326. self.agent_actionlog_service.insert_service_state(agent_monitor, AgentServiceState.IDLE, AgentLogState.IDLE)
  327. self._push_event_for_idle(agent, scene)
  328. def _push_event_for_checkin(self, agent, agent_monitor, phone, scene):
  329. """坐席签入事件推送"""
  330. agent_scene = AgentScene.get_by_code(scene)
  331. self.push_handler.push_on_agent_work_report(agent.saas_id, "", agent.out_id, "", agent_scene, WorkStatus.LOGIN_SUCCESS)
  332. event_data = AgentEventData(agent.saas_id, agent.out_id)
  333. event_data.data = {'eventName': DownEvent.ON_INITAL_SUCCESS.code,
  334. 'ext': {'saasId': agent.saas_id,
  335. 'agentId': agent.out_id,
  336. 'phoneNum': phone.phone_num,
  337. 'phonePwd': phone.phone_pwd,
  338. 'sipServer': phone.sip_server
  339. }
  340. }
  341. return event_data.__dict__
  342. def _push_event_for_checkout(self,agent,scene):
  343. """签出事件推送"""
  344. agent_scene = AgentScene.get_by_code(scene)
  345. self.push_handler.push_on_agent_work_report(agent.saas_id, "", agent.out_id, "", agent_scene, WorkStatus.NO_INIT, '签出成功')
  346. event_data = AgentEventData(agent.saas_id, agent.out_id)
  347. event_data.data = {'eventName': DownEvent.ON_INITAL_FAILURE.code,
  348. 'ext': {'saasId': agent.saas_id,
  349. 'agentId': agent.out_id,
  350. }
  351. }
  352. return event_data.__dict__
  353. def _push_event_for_busy(self,agent,scene):
  354. """置忙事件推送"""
  355. agent_scene = AgentScene.get_by_code(scene)
  356. self.push_handler.push_on_agent_work_report(agent.saas_id, "", agent.out_id, "", agent_scene, WorkStatus.AGENT_BUSY)
  357. pass
  358. def _push_event_for_idle(self, agent, scene):
  359. """坐席置闲事件推送"""
  360. agent_scene = AgentScene.get_by_code(scene)
  361. self.push_handler.push_on_agent_work_report(agent.saas_id, "", agent.out_id, "", agent_scene, WorkStatus.AGENT_READY)
  362. class AgentService:
  363. def __init__(self, app):
  364. self.app = app
  365. self.logger = app.logger
  366. self.data_handle_server = DataHandleServer(app)
  367. self.agent_monitor_service = AgentMonitorService(app)
  368. @with_app_context
  369. def get_and_check(self, req: AgentActionRequest):
  370. agent = self.data_handle_server.get_agent(req.saas_id, req.agent_id)
  371. if not agent:
  372. return {}
  373. phone = self.data_handle_server.get_phone(req.saas_id, agent.phone_num)
  374. return phone.to_dict()
  375. @with_app_context
  376. def watch_agent_state(self, req: HumanServiceQueryRequest):
  377. pos = HumanServiceMap.query.filter(HumanServiceMap.is_delete == 0, HumanServiceMap.saas_id == req.saas_id,
  378. HumanServiceMap.service_id == req.serviceId).all()
  379. agent_ids = [x.agent_id for x in pos]
  380. monitors = self.agent_monitor_service.detail_monitor_out_ids(req.saas_id, agent_ids)
  381. return monitors
  382. @with_app_context
  383. def add(self, req: AgentRequest):
  384. new_agent_num = self.data_handle_server.get_newest_agent_number(req.saas_id)
  385. agent = Agent(saas_id=req.saas_id, agent_num=new_agent_num, agent_name=req.agent_name, out_id=req.out_id,
  386. agent_pwd=req.agent_password, agent_type=req.agent_type, phone_num=req.phone_number,
  387. distribute=req.distribute, agent_state=req.agent_state)
  388. db.session.add(agent)
  389. db.session.commit()
  390. agent_monitor = AgentMonitor(saas_id=req.saas_id, agent_num=new_agent_num, out_id=req.out_id)
  391. db.session.add(agent_monitor)
  392. db.session.commit()
  393. return new_agent_num
  394. @with_app_context
  395. def update(self, req: AgentRequest):
  396. agent = self.data_handle_server.get_agent(req.saas_id, req.agent_number, req.out_id)
  397. if not agent:
  398. return
  399. phone_num = agent.phone_num
  400. state_change = agent.agent_state != req.agent_state
  401. disable = req.agent_state == AgentState.DISABLE
  402. agent.agent_name = req.agent_name
  403. agent.agent_pwd = req.agent_password
  404. agent.agent_type = req.agent_type
  405. agent.phone_num = req.phone_number if not disable else ''
  406. agent.distribute = req.distribute
  407. agent.agent_state = req.agent_state
  408. db.session.commit()
  409. if state_change and disable:
  410. phone = Phone.query.filter(Phone.saas_id == req.saas_id, Phone.phone_num == phone_num).first()
  411. if phone:
  412. db.session.delete(phone)
  413. @with_app_context
  414. def detail(self, saas_id, agent_number):
  415. agent = self.data_handle_server.get_agent(saas_id, agent_number=agent_number, out_id='')
  416. return agent
  417. @with_app_context
  418. def count(self, req: AgentQueryRequest):
  419. cnt = Agent.query.filter(Agent.saas_id == req.saas_id,
  420. or_(Agent.agent_num == req.agent_number,
  421. Agent.agent_name.contains(req.agent_name),
  422. Agent.out_id == req.out_id,
  423. Agent.agent_type == req.agent_type
  424. )
  425. ).count()
  426. return cnt
  427. @with_app_context
  428. def query_page(self, req: AgentQueryRequest):
  429. pagination = Agent.query.filter(Agent.saas_id == req.saas_id,
  430. or_(Agent.agent_num == req.agent_number,
  431. Agent.agent_name.contains(req.agent_name),
  432. Agent.out_id == req.out_id,
  433. Agent.agent_type == req.agent_type
  434. )
  435. ).paginate(req.page, req.size)
  436. # data = {
  437. # "page": pagination.page, # 当前页码
  438. # "pages": pagination.pages, # 总页码
  439. # "has_prev": pagination.has_prev, # 是否有上一页
  440. # "prev_num": pagination.prev_num, # 上一页页码
  441. # "has_next": pagination.has_next, # 是否有下一页
  442. # "next_num": pagination.next_num, # 下一页页码
  443. # "items": [{
  444. # "id": item.id,
  445. # "name": item.name,
  446. # "age": item.age,
  447. # "sex": item.sex,
  448. # "money": item.money,
  449. # } for item in pagination.items]
  450. # }
  451. return pagination
  452. @with_app_context
  453. def delete(self, saas_id, agent_number):
  454. agent = self.data_handle_server.get_agent(saas_id, agent_number=agent_number, out_id='')
  455. if not agent:
  456. return
  457. agent.is_delete = 1
  458. agent_monitor = self.data_handle_server.get_agent_monitor(saas_id, agent_number)
  459. agent_monitor.is_delete = 1
  460. db.session.commit()
  461. phone = self.data_handle_server.get_phone(saas_id, agent.phone_num)
  462. phone.is_delete = 1
  463. db.session.commit()
  464. class AgentMonitorService:
  465. def __init__(self, app):
  466. self.app = app
  467. self.logger = app.logger
  468. self.data_handle_server = DataHandleServer(app)
  469. @with_app_context
  470. def detail_monitor_out_ids(self, saas_id, out_ids, check_scene=None):
  471. if not out_ids:
  472. return []
  473. agents = Agent.query.filter(Agent.is_delete == 0, Agent.saas_id == saas_id, Agent.out_id.in_(out_ids)).all()
  474. if not agents:
  475. raise BizException(BizErrorCode.RECORD_NOT_EXIST_ERROR)
  476. agent_num_map = {x.agent_num: x for x in agents}
  477. agent_nums = [x.agent_num for x in agents]
  478. agent_monitors = AgentMonitor.query.filter(AgentMonitor.is_delete == 0, AgentMonitor.agent_num.in_(agent_nums)).all()
  479. res = []
  480. for agent_monitor in agent_monitors:
  481. agent = agent_num_map.get(agent_monitor.agent_num)
  482. data = AgentMonitorData()
  483. data.saas_id = saas_id
  484. data.agent_num = agent_monitor.agent_num
  485. data.agent_name = agent.agent_name
  486. data.out_id = agent_monitor.out_id
  487. data.check_state = agent_monitor.check_state
  488. data.check_scene = agent_monitor.check_scene
  489. data.online_state = 1 if agent_monitor.check_state == AgentCheck.IN.code else 0
  490. if agent_monitor.check_in_time:
  491. data.check_in_time = agent_monitor.check_in_time.timestamp()
  492. day_start = self._get_day_start()
  493. if agent_monitor.check_in_time.timestamp() > day_start and \
  494. agent_monitor.check_state == AgentCheck.OUT.code:
  495. data.online_state = 2
  496. data.check_out_time = agent_monitor.check_out_time.timestamp() if agent_monitor.check_out_time else None
  497. data.service_state = agent_monitor.service_state
  498. data.busy_time = agent_monitor.busy_time.timestamp() if agent_monitor.busy_time else None
  499. data.idle_time = agent_monitor.idle_time.timestamp() if agent_monitor.idle_time else None
  500. data.call_time = agent_monitor.call_time.timestamp() if agent_monitor.call_time else None
  501. data.hang_time = agent_monitor.hang_time.timestamp() if agent_monitor.hang_time else None
  502. data.heart_state = agent_monitor.heart_state
  503. data.heart_time = agent_monitor.heart_time.timestamp() if agent_monitor.heart_time else None
  504. data.update_time = agent_monitor.update_time.timestamp() if agent_monitor.update_time else None
  505. res.append(data.__dict__)
  506. return res
  507. @with_app_context
  508. def update_checkin(self, agent_monitor):
  509. agent_monitor.check_state = AgentCheck.IN.code
  510. agent_monitor.check_in_time = datetime.now()
  511. agent_monitor.heart_state = AgentHeartState.NORMAL.code
  512. agent_monitor.heart_time = datetime.now()
  513. self.logger.info("update_checkin %s", agent_monitor.check_state)
  514. db.session.commit()
  515. db.session.refresh(agent_monitor)
  516. @with_app_context
  517. def update_checkout(self, agent_monitor):
  518. agent_monitor.check_state = AgentCheck.OUT.code
  519. agent_monitor.check_out_time = datetime.now()
  520. agent_monitor.service_state = AgentServiceState.LOGOUT.code
  521. agent_monitor.heart_state = AgentHeartState.DEFAULT.code
  522. agent_monitor.heart_time = datetime.now()
  523. self.logger.info("update_checkout %s", agent_monitor.check_out_time)
  524. db.session.commit()
  525. @with_app_context
  526. def update_idle(self, agent_monitor):
  527. agent_monitor.service_state = AgentServiceState.IDLE.code
  528. agent_monitor.idle_time = datetime.now()
  529. db.session.commit()
  530. @with_app_context
  531. def update_busy(self, agent_monitor):
  532. agent_monitor.service_state = AgentServiceState.BUSY.code
  533. agent_monitor.busy_time = datetime.now()
  534. db.session.commit()
  535. @with_app_context
  536. def update_dialing(self, agent_monitor):
  537. agent_monitor.service_state = AgentServiceState.DIALING.code
  538. db.session.commit()
  539. @with_app_context
  540. def update_calling(self, agent_monitor):
  541. agent_monitor.service_state = AgentServiceState.CALLING.code
  542. agent_monitor.call_time = datetime.now()
  543. db.session.commit()
  544. @with_app_context
  545. def update_processing(self, agent_monitor):
  546. agent_monitor.service_state = AgentServiceState.REPROCESSING.code
  547. agent_monitor.hang_time = datetime.now()
  548. db.session.commit()
  549. @with_app_context
  550. def update_session_id(self, agent_monitor, session_id):
  551. agent_monitor.session_id = session_id
  552. db.session.commit()
  553. @with_app_context
  554. def update_heart_error(self, agent_monitor):
  555. agent_monitor.heart_state = AgentHeartState.ABNORMAL.code
  556. agent_monitor.heart_time = datetime.now()
  557. db.session.commit()
  558. def _get_day_start(self):
  559. today_start = datetime.now().replace(hour=0, minute=0, second=0, microsecond=0)
  560. return int(today_start.timestamp() * 1000) # Return milliseconds
  561. class AgentActionLogService:
  562. def __init__(self, app):
  563. self.app = app
  564. self.logger = app.logger
  565. self.data_handle_server = DataHandleServer(app)
  566. @with_app_context
  567. def insert_check_state(self, agent_monitor, agent_check_enum: AgentCheck, agent_log_enum: AgentLogState):
  568. action_log = AgentActionLog()
  569. action_log.saas_id = agent_monitor.saas_id
  570. action_log.agent_num = agent_monitor.agent_num
  571. action_log.out_id = agent_monitor.out_id
  572. action_log.action_type = 0
  573. action_log.check_state = agent_check_enum.code
  574. action_log.pre_check_state = agent_monitor.check_state
  575. if agent_log_enum:
  576. action_log.event_type = agent_log_enum.code
  577. action_log.event_desc = agent_log_enum.description
  578. now = datetime.now()
  579. pre_date = None
  580. if agent_monitor.check_state == AgentCheck.IN.code:
  581. pre_date = agent_monitor.check_in_time
  582. if agent_monitor.check_state == AgentCheck.OUT.code:
  583. pre_date = agent_monitor.check_out_time
  584. if pre_date is None:
  585. pre_date = agent_monitor.update_time
  586. action_log.check_state_time = now
  587. action_log.pre_check_state_time = pre_date
  588. action_log.check_state_duration = now.timestamp() - pre_date.timestamp()
  589. db.session.add(action_log)
  590. db.session.commit()
  591. @with_app_context
  592. def insert_service_state(self, agent_monitor, agent_service_state: AgentServiceState, agent_log_enum: AgentLogState,
  593. task_id=None, service_id=None):
  594. if agent_monitor.service_state == agent_service_state.code:
  595. self.logger.info("agent action log insert service state same %s %s", agent_monitor.service_state, agent_service_state.code)
  596. action_log = AgentActionLog()
  597. action_log.saas_id = agent_monitor.saas_id
  598. action_log.agent_num = agent_monitor.agent_num
  599. action_log.out_id = agent_monitor.out_id
  600. action_log.action_type = 1
  601. action_log.service_state = agent_service_state.code
  602. action_log.pre_service_state = agent_monitor.service_state
  603. action_log.task_id = task_id
  604. action_log.service_id = service_id
  605. if agent_log_enum:
  606. action_log.event_type = agent_log_enum.code
  607. action_log.event_desc = agent_log_enum.description
  608. now = datetime.now()
  609. pre_date = None
  610. if agent_monitor.service_state == AgentServiceState.IDLE.code:
  611. pre_date = agent_monitor.idle_time
  612. if agent_monitor.service_state == AgentServiceState.BUSY.code:
  613. pre_date = agent_monitor.busy_time
  614. if agent_monitor.service_state == AgentServiceState.CALLING.code:
  615. pre_date = agent_monitor.call_time
  616. if agent_monitor.service_state == AgentServiceState.LOGOUT.code:
  617. pre_date = agent_monitor.check_out_time
  618. if agent_monitor.service_state == AgentServiceState.REPROCESSING.code:
  619. pre_date = agent_monitor.hang_time
  620. action_log.service_state_time = now
  621. if pre_date is None:
  622. pre_date = agent_monitor.update_time
  623. action_log.pre_service_state_time = pre_date
  624. action_log.service_state_duration = now.timestamp() - pre_date.timestamp()
  625. db.session.add(action_log)
  626. db.session.commit()
  627. class AgentStateService:
  628. def __init__(self, app):
  629. self.app = app
  630. self.logger = app.logger
  631. self.redis_handler = RedisHandler()
  632. self.assigned_recycle_millisecond = 60000
  633. self.state_service_id_data_map = defaultdict(dict)
  634. self.executor = ThreadPoolExecutor(max_workers=10)
  635. self.data_handle_server = DataHandleServer(app)
  636. self.agent_monitor_service = AgentMonitorService(app)
  637. self.agent_actionlog_service = AgentActionLogService(app)
  638. def idle(self, saas_id, agent_id, phone_num):
  639. human_service = self.data_handle_server.get_human_service_service(saas_id, agent_id)
  640. if human_service is None:
  641. self.logger.info(f"agent engine idle not have human service {saas_id} {agent_id}") # 使用print替代log
  642. return
  643. self.idle_hash(saas_id, agent_id, phone_num, human_service.service_id)
  644. def busy(self, saas_id, agent_id, phone_num):
  645. human_service = self.data_handle_server.get_human_service_service(saas_id, agent_id)
  646. if human_service is None:
  647. self.logger.info(f"agent engine busy not hava human service {saas_id} {agent_id}") # 使用print替代log
  648. return
  649. self.busy_hash(saas_id, agent_id, phone_num, human_service.service_id)
  650. def idle_by_human(self, saas_id, agent_id, service_id):
  651. agent = self.data_handle_server.get_agent(saas_id, out_id=agent_id)
  652. if not agent:
  653. return
  654. agent_monitor = self.data_handle_server.get_agent_monitor(saas_id, agent_number=agent.agent_num)
  655. if not agent_monitor:
  656. return
  657. if agent_monitor.check_state == AgentCheck.IN.code:
  658. self.agent_monitor_service.update_idle(agent_monitor)
  659. self.agent_actionlog_service.insert_service_state(agent_monitor, AgentServiceState.IDLE, AgentLogState.ACTIVE_HUMAN_SERVICE)
  660. self.idle_hash(saas_id, agent_id, agent.phone_num, service_id)
  661. def busy_by_human(self, saas_id, service_id, agent_id=None):
  662. agent = self.data_handle_server.get_agent(saas_id, out_id=agent_id)
  663. if not agent:
  664. return
  665. agent_monitor = self.data_handle_server.get_agent_monitor(saas_id, agent_number=agent.agent_num)
  666. if not agent_monitor:
  667. return
  668. if agent_monitor.check_state == AgentCheck.IN.code:
  669. self.agent_monitor_service.update_busy(agent_monitor)
  670. self.agent_actionlog_service.insert_service_state(agent_monitor, AgentServiceState.BUSY,
  671. AgentLogState.DEL_HUMAN_SERVICE)
  672. self.busy_hash(saas_id, agent_id, agent.phone_num, service_id)
  673. def checkin(self, saas_id, agent_id, phone_num):
  674. key = self._check_in_key(saas_id)
  675. state_data = AgentStateData()
  676. state_data.status = HumanState.DEFAULT.code
  677. state_data.time = datetime.now().timestamp()
  678. self.redis_handler.redis.hset(key, phone_num, state_data.to_json_string())
  679. self.redis_handler.redis.expire(key, self._get_expire_time())
  680. def checkout(self, saas_id, agent_id, phone_num):
  681. key = self._check_in_key(saas_id)
  682. self.redis_handler.redis.hdel(key, phone_num)
  683. def assign_agent(self, saas_id, service_id, called=None, ivr_id=None, task_id=None, cbp=None):
  684. choose_phone_num = ''
  685. self.logger.info("assignAgent %s %s %s"% (saas_id, service_id, called))
  686. idle_agents = self.idle_agents(saas_id, service_id)
  687. if len(idle_agents) <= 0:
  688. return choose_phone_num
  689. choose_phone_num = self._choose_max_idle_time(idle_agents)
  690. self.handle_assign_time(saas_id, service_id, choose_phone_num)
  691. return choose_phone_num
  692. def handle_assign_time(self, saas_id, service_id, choose_phone_num):
  693. key = self._key(saas_id, service_id)
  694. cache_agent_map = self.get_cache_agent_map(saas_id, service_id)
  695. if cache_agent_map and choose_phone_num in cache_agent_map:
  696. state_data = cache_agent_map[choose_phone_num]
  697. state_data.assign_time = datetime.now().timestamp() * 1000
  698. self.redis_handler.redis.hset(key, choose_phone_num, state_data.to_json_string())
  699. self.redis_handler.redis.expire(key, self._get_expire_time())
  700. # self.update_report_state(saas_id, service_id)
  701. def idle_agents(self, saas_id, service_id):
  702. cache_agent_list = self.get_cache_agent_list(saas_id, service_id)
  703. if not cache_agent_list:
  704. return []
  705. agent_str = '\n'.join(["%s-%s-%s-%s"%(x.phone_num, x.status, x.time, x.assign_time) for x in cache_agent_list])
  706. self.logger.info("assignAgent %s %s idleAgents:%s "% (saas_id, service_id, agent_str))
  707. return self.get_idle_agents(cache_agent_list)
  708. def idle_hash(self, saas_id, agent_id, phone_num, service_id):
  709. key = self._key(saas_id, service_id)
  710. state_data = AgentStateData()
  711. # self.logger.info("idle_hash, key=%s, saas_id=%s, phone_num=%s", key, saas_id, phone_num)
  712. cache_agent_map = self.get_cache_agent_map(saas_id, service_id)
  713. if cache_agent_map and phone_num in cache_agent_map:
  714. state_data = cache_agent_map[phone_num]
  715. state_data.status = HumanState.IDLE.code
  716. state_data.time = datetime.now().timestamp() * 1000
  717. self.redis_handler.redis.hset(key, phone_num, state_data.to_json_string())
  718. self.redis_handler.redis.expire(key, self._get_expire_time())
  719. # self.update_report_state(saas_id, service_id)
  720. self.logger.info("idle_hash, key=%s, saas_id=%s, phone_num=%s, state_data=%s"%(key, saas_id, phone_num, state_data))
  721. def busy_hash(self, saas_id, agent_id, phone_num, service_id):
  722. cache_agent_map = self.get_cache_agent_map(saas_id, service_id)
  723. state_data = cache_agent_map.get(phone_num)
  724. key = self._key(saas_id, service_id)
  725. if state_data is None:
  726. return
  727. state_data.status = HumanState.BUSY.code
  728. self.redis_handler.redis.hset(key, phone_num, state_data.to_json_string())
  729. self.redis_handler.redis.expire(key, self._get_expire_time())
  730. # self.update_report_state(saas_id, service_id)
  731. def get_cache_agent_map(self, saas_id, service_id):
  732. cache_agent_list = self.get_cache_agent_list(saas_id, service_id)
  733. # 检查列表是否为空,如果为空返回空字典
  734. if not cache_agent_list:
  735. return {}
  736. # 使用字典推导式将 cache_agent_list 转换为字典
  737. return {agent.phone_num: agent for agent in cache_agent_list}
  738. def get_cache_agent_list(self, saas_id, service_id):
  739. redis_key = self._key(saas_id, service_id)
  740. map_cache_by_key = self.redis_handler.redis.hgetall(redis_key)
  741. self.logger.info("get_cache_agent_list, redis_key=%s, map_cache_by_key=%s"%(redis_key, map_cache_by_key))
  742. if not map_cache_by_key: # 检查字典是否为空
  743. return [] # 返回空列表
  744. free_agents = []
  745. for phone_num, json_value in map_cache_by_key.items():
  746. agent_status_data_dict = json.loads(json_value)
  747. agent_status_data = AgentStateData(**agent_status_data_dict) # 解析 JSON 为 AgentStateData 对象
  748. agent_status_data.phone_num = phone_num.decode('utf-8') # 设置电话号码
  749. free_agents.append(agent_status_data)
  750. return free_agents
  751. def get_idle_agents(self,cache_agent_list):
  752. current_time =int(datetime.now().timestamp() * 1000) # 获取当前时间的毫秒级时间戳
  753. idle_agents = [
  754. agent for agent in cache_agent_list
  755. if agent.status == 1 and (
  756. agent.assign_time == 0 or
  757. agent.assign_time + self.assigned_recycle_millisecond < current_time
  758. )
  759. ]
  760. return idle_agents
  761. def get_agent_service_idle_size(self, saas_id, service_id):
  762. idle_agents_size = 0
  763. cache_agent_list = self.get_cache_agent_list(saas_id, service_id)
  764. if cache_agent_list: # 检查列表是否非空
  765. idle_agents = self.get_idle_agents(cache_agent_list)
  766. idle_agents_size = len(idle_agents) # 获取空闲代理的数量
  767. return idle_agents_size
  768. def get_agent_service_busy_size(self, saas_id, service_id):
  769. busy_agents_size = 0
  770. cache_agent_list = self.get_cache_agent_list(saas_id, service_id)
  771. if cache_agent_list: # 检查列表是否非空
  772. idle_agents = self.get_idle_agents(cache_agent_list)
  773. busy_agents = [agent for agent in cache_agent_list if agent not in idle_agents] # 计算忙碌代理
  774. busy_agents_size = len(busy_agents) # 获取忙碌代理的数量
  775. return busy_agents_size
  776. def update_report_state(self, saas_id, service_id):
  777. key = self._key(saas_id, service_id)
  778. # data_map 这个地方有疑问
  779. data_map = self.state_service_id_data_map[key]
  780. idle = HumanState.IDLE
  781. if idle.value not in data_map:
  782. data_map[idle.code] = threading.Lock()
  783. self.executor.submit(self.do_report_real_time_human_service_id, saas_id, service_id, idle)
  784. # data_map[idle.code] = self.do_report_real_time_human_service_id(saas_id, service_id, idle)
  785. busy = HumanState.BUSY
  786. if busy.value not in data_map:
  787. data_map[busy.code] = threading.Lock()
  788. self.executor.submit(self.do_report_real_time_human_service_id, saas_id, service_id, busy)
  789. # data_map[busy.code] = self.do_report_real_time_human_service_id(saas_id, service_id, busy)
  790. def do_report_real_time_human_service_id(self, saas_id, service_id, human_state):
  791. name = "cti_center_real_time_human_service_state"
  792. tag_list = {
  793. "vcc_id": saas_id,
  794. "service_id": service_id,
  795. "state": human_state.code,
  796. }
  797. if human_state == HumanState.IDLE:
  798. # meter_registry 这块疑问
  799. self.meter_registry.gauge(name, tag_list, self, lambda ctx: ctx.get_agent_service_busy_size(saas_id, service_id))
  800. elif human_state == HumanState.BUSY:
  801. self.meter_registry.gauge(name, tag_list, self, lambda ctx: ctx.get_agent_service_idle_size(saas_id, service_id))
  802. return 0
  803. def _check_in_key(self, saas_id):
  804. return "CTI:%s:HUMAN:AGENT"%(saas_id.upper())
  805. def _key(self, saas_id, service_id):
  806. return "CTI:%s:HUMAN:%s"%(saas_id.upper(), service_id)
  807. def _get_expire_time(self):
  808. now = datetime.now()
  809. end_of_day = now.replace(hour=23, minute=59, second=59, microsecond=0)
  810. expire_time = (end_of_day - now).total_seconds() * 1000 # Convert to milliseconds
  811. return int(expire_time)
  812. def _choose_max_idle_time(self, idle_agents: List[AgentStateData]) -> str:
  813. idle_agents = sorted(idle_agents, key=lambda agent: agent.assign_time, reverse=False)
  814. return idle_agents[0].phone_num